您好,登錄后才能下訂單哦!
小編給大家分享一下ThinkPHP如何自動加載Loader源碼,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!
?想了很久終于要開始系列文章的編寫了,期望是寫出提升和面試都可以搞定的系列文章。 當你看到本文時,如果你發現咔咔沒有編寫到的面試熱點問題或者技術難點,期待評論區指出,一起完善。
?
目前在整理PHP進階路線圖,如有好的建議咔咔會第一時間進行收錄。
從上圖咔咔給的解析圖,在base.php中首先加載了loader類,接著調用了register這個方法。
來到thinkphp\library\think\Loader.php
有一個register的方法,在這個方法里邊,我們先學習第一個知識點spl_autoload_register()
聊聊spl_autoload_register前世今生和簡單使用,直接點擊即可查看。
緊接著就是項目的根路徑和composer的路徑。
從這里開始就是在加載composer文件,過程也是很簡單
接著可以去vendor\composer\autoload_static.php文件中可以看到這倆個屬性
這里有一段代碼估計有一部分同學會在這里繞一下self::${$attr} = $composerClass::${$attr};
,這里的$attr
就是'prefixLengthsPsr4', 'prefixDirsPsr4', 'fallbackDirsPsr4', 'prefixesPsr0', 'fallbackDirsPsr0', 'classMap', 'files'
這些數據,外層在加一個$
符號。
從而在ComposerStaticInit30742487e00917c888d89ba216f165b9
這個類中直接獲取對應的屬性值,也就是上圖的倆個屬性值。
文件還是thinkphp\library\think\Loader.php
的register
方法
在這里注冊了倆個命令空間,分別為think和traits。然后會進入到addNamespace這個方法中在addNamespace
方法中,添加了Psr4空間
接著來到addPsr4這個方法,會把這倆個命名空間都注冊到ComposerStaticInit1e269472f484e157e90227b420ffca7a類的$prefixLengthsPsr4和$prefixDirsPsr4這倆個屬性中
為了驗證上面做一個斷點調試,看到這些數據就應該清晰了,至于traits
也是一樣的注冊方式。
截止到這里命名空間就注冊完成了,接下來研究一下psr4命名空間是個什么東東。
psr是簡單的理解就是文件路徑、自動加載對應類的相關規范、目前TP5.1使用的是psr4規范
此處的類是指class、接口、超類結構
一個完整的類需要一下結構\<命名空間>(\<子命名空間>)*\<類名>
以下規范來源于PHP文檔
完整的類名必須要有一個頂級命名空間,被稱為 "vendor namespace";
完整的類名可以有一個或多個子命名空間;
完整的類名必須有一個最終的類名;
完整的類名中任意一部分中的下滑線都是沒有特殊含義的;
完整的類名可以由任意大小寫字母組成;
所有類名都必須是大小寫敏感的。
以下是官方給的一個例子,這個psr規范能理解就盡量去理解它
到這里,肯定會有一個疑問,這里怎么沒有classmap.php這個文件。不急不慌,先執行php think optimize:autoload
把文件弄出來最終會走到addClassMap
這個方法,在這個方法中,只是把classmap.php
這個文件的數據賦值給$classMap
而已,沒有什么其它的用法
extend這個目錄用過TP框架的都多少用過的,在這個目錄里邊可以存放一下自定義的類庫文件。
根據下圖可以看到就是使用addAutoLoadDir
這個方法進行加載的
在方法中也僅僅是把extend的路徑賦值給了$fallbackDirsPsr4
這個屬性。
截止到這里Loader::register();
這部分就結束了,接著我們深入的看一下內部實現和實踐案例。
在以上閱讀源碼中有四個屬性,簡單的整理一下
在剛剛開始解析這里的源碼時就有一個函數spl_autoload_register
當需要使用的類沒有被引入時,這個函數會在PHP報錯前被觸發,未定義的類名會被當作參數傳入這里會直接去執行think\\Loader::autoload
這個方法
經過斷點第一個未加載的類就是think\Error
為什么是think\Error呢!可以在回到thinkphp/base.php
看一下,當自動加載完執行完成后第一個執行的類就是Error
可以簡單的做個測試,將這Error改為Kaka,進行打印一下,這時的類就改變為Kaka。到這里大家對這個類的自動加載機制就有一定的了解了。
當使用的類沒有被引入時會把這個類當做參數傳到thinkphp/library/think/Loader.php
的autoload
方法中。
到這里在進行看一下autoload這個方法
先從findFile這個方法走,把未因為的類傳入這個方法中,在findFile這個方法中會直接從classMap這個屬性中直接把think\Error這個類映射的文件直接返回出來
將think\Error這個類的完整路徑返回給autoload
的file
變量后,把win環境的大小寫給判斷了一次。
然后直接使用include
引入文件即可,直到返回。
直到這里就是一次完整的類的自動加載解析。
雖然到這里結束了,但是還是得在提一點就是$classMap
這個屬性,這個屬性是基于文件classmap.php
來到,這個文件的生成也是需要執行命令php think optimize:autoload
生成的。
當沒有生成這個文件時程序是如何執行的呢!
之前的所有流程都是一樣的,只有在findFile
這里不一樣,接下來進行簡單的梳理一下。
這時代碼肯定不會走classMap
先獲取think\Error文件
然后經過Composer自動加載中的倆個屬性進行獲取命名空間,在把think\Error.php文件進行拼接
最終返回的結果也是D:\phpstudy_pro\WWW\ThinkPHPSourceCodeAnalysis\thinkphp\library\think\Error.php
這個文件。
這里的代碼需要好好的閱讀一下。
類的自動加載到這里就是完全結束了。
先創建一個文件夾kaka
這時在控制器index中引入文件Kaka.php
直接進行訪問,這時這個類肯定會報錯,那么我們應該怎么操作一下,就可以直接訪問呢!
這個時候就提現到源碼的重要性了,還記得在自動加載的register
函數中,加載過extend目錄
這時在加一個kaka這個目錄,直接進行訪問一下
沒毛病,直接就出來了。一切OK在這里在聊一下關于extent的加載方式
在之前聊注冊自動加載類庫目錄只是說明了一下只是把路徑存到了$fallbackDirsPsr4
屬性,沒有細細說,接下來就是說明這些了。
閱讀源碼只能是實現那然后查看那
只要是定義的類都會進去到autoload進行自動加載
同樣也會進入到findFile
這個方法
在findFile這個方法中可以看到這段代碼,這個屬性是不是很熟悉,就是自動加載extend目錄時添加到$fallbackDirsPsr4
屬性的。
當在findFile中打印參數class時看一下數據
很清楚地可以看到test\Kaka
這個類
此時在打印一下這個$fallbackDirsPsr4
屬性里邊返回的file
然后就是使用__include_file
來直接includeD:\phpstudy_pro\WWW\ThinkPHPSourceCodeAnalysis\kaka\test\Kaka.php
我們定義的文件。
以上的這個自定義文件如何實現類的自動加載,并且也就是extend
的加載方式
看完了這篇文章,相信你對“ThinkPHP如何自動加載Loader源碼”有了一定的了解,如果想了解更多相關知識,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。