記錄這個經歷不是為了說我多厲害,多牛逼,而是為了能夠加深自己對所學知識的理解。如果可以幫助到別人學習這個框架的話,小生不勝榮幸。
關於 laravel 的教程很多,我也是通過大量閱讀別人寫的教程對 laravel 有了一個比較表面的認識。我也不打算把這個當做教程來寫,因為我沒寫過教程,不知道怎麼寫,而且我個人是比較隨意的,請原諒。
既然我是走進去的,就把我當做apache吧,我看到的第一個檔案應該是專案中 public 的 index.php 檔案。
1.
define('LARAVEL_START', microtime(true));
解讀:在執行時定義一個常量 LARAVEL_START 為當前 Unix 時間戳。
2.
require __DIR__.'/../vendor/autoload.php';
解讀:引入一個自動載入器
2.1進入到 autoload.php 中
require_once __DIR__ . '/composer/autoload_real.php';
return ComposerAutoloaderInit94ec56b468e10706582fb576e66d9831::getLoader();
解讀:從 autoload_real.php 中引入自載入入口 getLoader
public static function getLoader()
{
//如果 $loader 非空,返回 $loader
if (null !== self::$loader) {
return self::$loader;
}
//spl_autoload_register:註冊指定的函式 loadClassLoader 作為__autoload的實現, throw:丟擲異常,prepend:新增到佇列之首
spl_autoload_register(array('ComposerAutoloaderInit94ec56b468e10706582fb576e66d9831', 'loadClassLoader'), true, true);
// 建立 \Composer\Autoload\ClassLoader() 物件,並物件化類屬性 $loader
self::$loader = $loader = new \Composer\Autoload\ClassLoader();
// 登出已註冊的指定的__autoload()函式loadClassLoader
spl_autoload_unregister(array('ComposerAutoloaderInit94ec56b468e10706582fb576e66d9831', 'loadClassLoader'));
// 判斷 php 版本是否大於5.6,且沒有定義 HHVM 版本(一個高效能 PHP 執行引擎,類似於 apache ),且 (不存在函式 zend_loader_file_encoded 或未執行 zend_loader_file_encoded )
$useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
if ($useStaticLoader) {
// 載入 autoload_static.php 的 ComposerStaticInit94ec56b468e10706582fb576e66d9831 類,一個靜態資訊檔案,包含了需要自動載入的files路徑、psr 規範資訊、laravel 框架類的路徑
require_once __DIR__ . '/autoload_static.php';
//執行 ComposerStaticInit94ec56b468e10706582fb576e66d9831 類中的 getInitializer 靜態函式。call_user_func:執行作為第一個引數的回撥函式,如果執行失敗返回 false,避免發生錯誤
/**
public static function getInitializer(ClassLoader $loader)
{
return \Closure::bind(function () use ($loader) {
$loader->prefixLengthsPsr4 = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$prefixLengthsPsr4;
$loader->prefixDirsPsr4 = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$prefixDirsPsr4;
$loader->prefixesPsr0 = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$prefixesPsr0;
$loader->classMap = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$classMap;
}, null, ClassLoader::class);
}
*/
// getInitializer 其實就是在返回 files 路徑、psr 規範資訊、laravel 框架類的路徑
call_user_func(\Composer\Autoload\ComposerStaticInit94ec56b468e10706582fb576e66d9831::getInitializer($loader));
} else {
//載入 autoload_namespaces.php,返回符合 prs0 自載入外掛的名稱空間-路徑陣列
$map = require __DIR__ . '/autoload_namespaces.php';
foreach ($map as $namespace => $path) {
//向 $loader 新增自載入外掛
$loader->set($namespace, $path);
}
//載入 autoload_psr4.php,返回符合 prs4 自載入外掛的名稱空間-路徑陣列
$map = require __DIR__ . '/autoload_psr4.php';
foreach ($map as $namespace => $path) {
//向 $loader 新增自載入外掛
$loader->setPsr4($namespace, $path);
}
//載入 autoload_classmap.php,返回各種需自載入類的名稱空間-路徑陣列
$classMap = require __DIR__ . '/autoload_classmap.php';
if ($classMap) {
//向 $loader 新增自載入類
$loader->addClassMap($classMap);
}
}
// 預載入 autoloader
$loader->register(true);
// 載入框架檔案
if ($useStaticLoader) {
$includeFiles = Composer\Autoload\ComposerStaticInit94ec56b468e10706582fb576e66d9831::$files;
} else {
$includeFiles = require __DIR__ . '/autoload_files.php';
}
foreach ($includeFiles as $fileIdentifier => $file) {
composerRequire94ec56b468e10706582fb576e66d9831($fileIdentifier, $file);
}
return $loader;
}
解讀:檢視程式碼註釋
3.
$app = require_once __DIR__.'/../bootstrap/app.php';
解讀:引入laravel框架例項
4.
$kernel = $app->make(Illuminate\Contracts\Http\Kernel::class);
$response = $kernel->handle(
$request = Illuminate\Http\Request::capture()
);
$response->send();
$kernel->terminate($request, $response);
解讀:啟動laravel應用與 http 互動,完成 http 請求和響應
/**
* Resolve the given type from the container.
* 對映指定型別的容器
* (Overriding Container::make)
*
* @param string $abstract
* @param array $parameters
* @return mixed
*/
public function make($abstract, array $parameters = [])
{
$abstract = $this->getAlias($abstract);
if (isset($this->deferredServices[$abstract]) && ! isset($this->instances[$abstract])) {
$this->loadDeferredProvider($abstract);
}
return parent::make($abstract, $parameters);
}
// Illuminate\Container\Container::class
/**
* Get the alias for an abstract if available.
* 獲取可信賴的類組
* @param string $abstract
* @return string
*
* @throws \LogicException
*/
public function getAlias($abstract)
{
if (! isset($this->aliases[$abstract])) {
return $abstract;
}
if ($this->aliases[$abstract] === $abstract) {
throw new LogicException("[{$abstract}] is aliased to itself.");
}
return $this->getAlias($this->aliases[$abstract]);
}
// Illuminate\Foundation\Application::class
public function loadDeferredProvider($service)
{
if (! isset($this->deferredServices[$service])) {
return;
}
$provider = $this->deferredServices[$service];
// If the service provider has not already been loaded and registered we can
// register it with the application and remove the service from this list
// of deferred services, since it will already be loaded on subsequent.
// 當服務提供者即將被載入,如果服務提供者還沒有被載入和註冊,
// 我們可以在這個應用中註冊它,並且去除
// 這個列表中已經快取的服務
if (! isset($this->loadedProviders[$provider])) {
$this->registerDeferredProvider($provider, $service);
}
}
public function registerDeferredProvider($provider, $service = null)
{
// Once the provider that provides the deferred service has been registered we
// will remove it from our local list of the deferred services with related
// providers so that this container does not try to resolve it out again.
// 一旦服務提供者提供被註冊的快取的服務,我們將從我們本地快取的服務列表中去除它和
//與它相關聯的服務提供者,如此這個容器就不會再嘗試對映它。
if ($service) {
unset($this->deferredServices[$service]);
}
$this->register($instance = new $provider($this));
if (! $this->booted) {
$this->booting(function () use ($instance) {
$this->bootProvider($instance);
});
}
}