您好,登錄后才能下訂單哦!
本篇內(nèi)容介紹了“不使用include_once和require_once的原因有哪些”的有關(guān)知識,在實(shí)際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
PHP去判斷一個文件是否被加載, 是需要得到這個文件的opened_path的, 意思是說, 比如:
復(fù)制代碼 代碼如下:
<?php
set_include_path("/tmp/:/tmp2/");
include_once("2.php");
?>
當(dāng)PHP看到include_once “2.php”的時候, 他并不知道這個文件的實(shí)際路徑是什么, 也就無法從已加載的文件列表去判斷是否已經(jīng)加載, 所以在include_once的實(shí)現(xiàn)中, 會首先嘗試解析這個文件的真實(shí)路徑(對于普通文件這個解析僅僅類似是檢查getcwd和文件路徑, 所以如果是相對路徑, 一般是不會成功), 如果解析成功, 則查找EG(include_files), 如果存在則說明包含過了, 返回, 否則open這個文件, 從而得到這個文件的opened_path. 比如上面的例子, 這個文件存在于 “/tmp2/2.php”.
然后, 得到了這個opened_path以后, PHP去已加載的文件列表去查找, 是否已經(jīng)包含, 如果沒有包含, 那么就直接compile, 不再需要open file了.
1. 嘗試解析文件的絕對路徑, 如果能解析成功, 則檢查EG(included_files), 存在則返回, 不存在繼續(xù)
2. 打開文件, 得到文件的打開路徑(opened path)
3. 拿opened path去EG(included_files)查找, 是否存在, 如果存在則返回, 不存在繼續(xù)
4. 編譯文件(compile_file)
這個在大多數(shù)情況下, 不是問題, 然而問題出在當(dāng)你使用APC的時候…
在使用APC的時候, APC劫持了compile_file這個編譯文件的指針, 從而直接從cache中得到編譯結(jié)果, 避免了對實(shí)際文件的open, 避免了對open的system call.
然而, 當(dāng)你在代碼中使用include_once的時候, 在compile_file之前, PHP已經(jīng)嘗試去open file了, 然后才進(jìn)入被APC劫持的compile file中, 這樣一來, 就會產(chǎn)生一次額外的open操作. 而APC正是為了解決這個問題, 引入了include_once_override, 在include_once_override開啟的情況下, APC會劫持PHP的ZEND_INCLUDE_OR_EVAL opcode handler, 通過stat來確定文件的絕對路徑, 然后如果發(fā)現(xiàn)沒有被加載, 就改寫opcode為include, 做一個tricky解決方案.
但是, 很可惜, 如我所說, APC的include_once_override實(shí)現(xiàn)的一直不好, 會有一些未定義的問題, 比如:
復(fù)制代碼 代碼如下:
<?php
set_include_path("/tmp");
function a($arg = array()) {
include_once("b.php");
}
a();
a();
?>
然后, 我們的b.php放置在”/tmp/b.php”, 內(nèi)容如下:
復(fù)制代碼 代碼如下:
<?php
class B {}
?>
那么在打開apc.include_once_override的情況下, 連續(xù)訪問就會得到如下錯誤:
Fatal error - include() : Cannot redeclare class
排除這些技術(shù)因素, 我也一直認(rèn)為, 我們應(yīng)該使用include, 而不是include_once, 因?yàn)槲覀兺耆茏龅阶约阂?guī)劃, 一個文件只被加載一次. 還可以借助自動加載, 來做到這一點(diǎn).
你使用include_once,只能證明, 你對自己的代碼沒信心.
所以, 建議大家, 不要再使用include_once
“不使用include_once和require_once的原因有哪些”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。