❝想了很久終於要開始系列文章的編寫了,期望是寫出提升和麵試都可以搞定的系列文章。 當你看到本文時,如果你發現咔咔沒有編寫到的面試熱點問題或技術難點,期待評論區指出,一起完善。
❞
目前在整理PHP進階路線圖,如有好的建議咔咔會第一時間進行收錄。
從上圖咔咔給的解析圖,在base.php中先載入了loader類,接著呼叫了register這個方法。
來到thinkphp\library\think\Loader.php
有一個register的方法,在這個方法裡邊,我們先學習第一個知識點spl_autoload_register()
聊聊spl_autoload_register前世今生和簡單使用,直接點擊即可查看。
緊接著就是專案的根路徑和composer的路徑。
#從這裡開始就是在載入composer文件,過程也是很簡單
#接著可以去vendor\composer\phpload_statstatic.看到這兩個屬性
這裡有一段程式碼估計有一部分同學會在這裡繞一下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和$prefixDirs#P#4這兩個屬性中的$#1# #為了驗證上面做一個斷點調試,看到這些資料就應該清晰了,至於
traits
截止到這裡命名空間就註冊完成了,接下來研究一下psr4命名空間是個什麼東東。
一個完整的類別需要一下結構
\(\)*\以下規格來自PHP文檔<ul data-tool="mdnice编辑器" style="margin-top: 8px; margin-bottom: 8px; padding-left: 25px; font-size: 15px; color: #595959; list-style-type: circle;">
<li><section style="margin-top: 5px; margin-bottom: 5px; line-height: 26px; font-size: 14px;"><p style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; color: rgb(43, 43, 43); margin-top: 10px; margin-bottom: 10px; word-spacing: 2px;">完整的類別名稱必須要有一個頂層命名空間,被稱為"vendor namespace";</p></section></li>
<li><section style="margin-top: 5px; margin-bottom: 5px; line-height: 26px; font-size: 14px;"><p style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; color: rgb(43, 43, 43); margin-top: 10px; margin-bottom: 10px; word-spacing: 2px;">完整的類別名稱可以有一個或多個子命名空間;</p></section></li>
<li><section style="margin-top: 5px; margin-bottom: 5px; line-height: 26px; font-size: 14px;"><p style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; color: rgb(43, 43, 43); margin-top: 10px; margin-bottom: 10px; word-spacing: 2px;"> 完整的類別名稱必須有一個最終的類別名稱;</p></section></li>
<li><section style="margin-top: 5px; margin-bottom: 5px; line-height: 26px; font-size: 14px;"><p style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; color: rgb(43, 43, 43); margin-top: 10px; margin-bottom: 10px; word-spacing: 2px;">#完整的類別名稱中任一部分的下滑線都是沒有特殊意義的;</p></section></li>
<li><section style="margin-top: 5px; margin-bottom: 5px; line-height: 26px; font-size: 14px;"><p style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; color: rgb(43, 43, 43); margin-top: 10px; margin-bottom: 10px; word-spacing: 2px;">完整的類名可以由任意大小寫字母組成;</p></section></li>
<li><section style="margin-top: 5px; margin-bottom: 5px; line-height: 26px; font-size: 14px;"><p style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; color: rgb(43, 43, 43); margin-top: 10px; margin-bottom: 10px; word-spacing: 2px;">所有類別名稱都必須是大小寫敏感的。 </p></section></li>
</ul>
<p data-tool="mdnice编辑器" style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; margin-top: 10px; margin-bottom: 10px; font-size: 14px; word-spacing: 2px;">以下是官方給的一個例子,這個psr規範能理解就盡量去理解它<img src="https://img-blog.csdnimg.cn/20200914110757812.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2ZhbmdrYW5nNw==,size_16,color_FFFFFF,t_70#pic_center" alt="ThinkPHP自動載入Loader源碼解析" style="max-width:90%"></p>
<h2 data-tool="mdnice编辑器" style="margin-top: 30px; margin-bottom: 15px; padding: 0px; color: black; font-size: 22px; border-bottom: 4px solid rgb(64, 184, 250);">
<span class="prefix" style="display: flex; width: 20px; height: 20px; background-size: 20px 20px; background-image: url(https://my-wechat.mdnice.com/fullstack-1.png); margin-bottom: -22px;"></span><span class="content" style="display: flex; color: #40B8FA; font-size: 20px; margin-left: 25px;">1-5 載入類別庫映射檔</span><span class="suffix" style="display: flex; box-sizing: border-box; width: 200px; height: 10px; border-top-left-radius: 20px; background: RGBA(64, 184, 250, .5); color: rgb(255, 255, 255); font-size: 16px; letter-spacing: 0.544px; justify-content: flex-end; float: right; margin-top: -10px; box-sizing: border-box !important; overflow-wrap: break-word !important;"></span>
</h2>
<p data-tool="mdnice编辑器" style="padding-top: 8px; padding-bottom: 8px; line-height: 26px; margin-top: 10px; margin-bottom: 10px; font-size: 14px; word-spacing: 2px;">#到這裡,一定會有一個疑問,這裡怎麼沒有classmap.php這個文件。 <img src="https://img-blog.csdnimg.cn/20200914112731364.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2ZhbmdrYW5nNw==,size_16,color_FFFFFF,t_70#pic_center" alt="ThinkPHP自動載入Loader源碼解析" style="max-width:90%">不急不慌,先執行<code style="overflow-wrap: break-word; margin: 0px 2px; font-family: " operator mono consolas monaco menlo monospace word-break: break-all color: rgb background: rgba padding: border-radius: height: line-height:>php think optimize:autoload
把檔案弄出來最後會走到addClassMap
這個方法,在這個方法中,只是把classmap.php
這個檔案的資料賦值給$classMap
而已,沒有什麼它的用法
根據下圖可以看到就是使用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
來直接include D:\phpstudy_pro\WWW\ThinkPHPSourceCodeAnalysis\kaka\test\Kaka.php
我們定義的檔案。
以上的這個自訂檔案如何實作類別的自動加載,也就是extend
的載入方式
關於類別自動載入的所有流程就完成了,如有錯誤之處可以在評論區哦!
❝堅持學習、堅持寫博、堅持分享是咔咔從業以來一直所秉持的信念。希望在偌大互聯網中咔咔的文章能帶給你一絲絲幫助。我是咔咔,下期見。
❞
以上是ThinkPHP自動載入Loader源碼解析的詳細內容。更多資訊請關注PHP中文網其他相關文章!