跟着源码课程5-8学习,学到App类初始化$this->initialize( )方法的时候,其中有一步是加载route目录下的route.php文件,发现这个文件里面可以调用:
Route::get("test","test/index"); Route::post("test","test/index" ) ; Route::rule("test","test/index" ) ;
Route静态访问方法这些形式,打开think\Route文件类,发现think\Route类里面get( ),post( ),rule( )方法都不是静态的,为什么能使用Route::get( )这种形式去访问呢?很显然,我们会联想到之前讲的门面模式,只有实例化think\facade\Route这个类,才能使用门面模式(访问不存在的静态方法时,会调用__callStatic( )),那么为什么在 route.php能直接使用门面模式呢? 我们可以借助 get_clared_classes( )这个函数(这个函数包含所有被加载到的类的集合数组)去App类里面的初始化路由的地方,打印一下,发现在加载route.php之前,think\facade\Route 类并没有被加载过,那么 直接使用Route::get( ),此时相当于调用不能存在的类Route, ,那么我们结合之前学的自动加载类,发现在base.php里面有个类的别名注册:
Loader::addClassAlias([ 'App' => facade\App::class, 'Build' => facade\Build::class, 'Cache' => facade\Cache::class, 'Config' => facade\Config::class, 'Cookie' => facade\Cookie::class, 'Db' => Db::class, 'Debug' => facade\Debug::class, 'Env' => facade\Env::class, 'Facade' => Facade::class, 'Hook' => facade\Hook::class, 'Lang' => facade\Lang::class, 'Log' => facade\Log::class, 'Request' => facade\Request::class, 'Response'=> facade\Response::class, 'Route' => facade\Route::class, 'Session' => facade\Session::class, 'Url' => facade\Url::class, 'Validate'=> facade\Validate::class, 'View' => facade\View::class, ]);
即 "Route" => facade\Route , 所以触发Loader类里面auto方法的时候会在别名映射里面去找到 think\facade\Route ,并实例化, 然后 此时的 Route类是继承了 think\Facade,静态访问Route::get( )方法的时候,会触发__callStatic,然后去实例化 think\Route类,并调用该类下的get( )方法
课程详情:<<BAT资深工程师由浅入深分析Tp5和Tp6底层源码>>