萬盛學電腦網

 萬盛學電腦網 >> 網絡編程 >> php編程 >> php中include_once和require_once性能分析

php中include_once和require_once性能分析

本文章來給各位朋友介紹php中include_once和require_once性能分析,有需要了解的朋友不防參考參考。


我們知道, PHP去判斷一個文件是否被加載, 是需要得到這個文件的opened_path的, 意思是說, 比如:

 代碼如下 復制代碼

 <?php

    set_include_path("/tmp/:/tmp2/");

    include_once("2.php");

    ?>

當PHP看到include_once “2.php”的時候, 他並不知道這個文件的實際路徑是什麼, 也就無法從已加載的文件列表去判斷是否已經加載, 所以在include_once的實現中, 會首先嘗試解析這個文件的真實路徑(對於普通文件這個解析僅僅類似是檢查getcwd和文件路徑, 所以如果是相對路徑, 一般是不會成功), 如果解析成功, 則查找EG(include_files), 如果存在則說明包含過了, 返回, 否則open這個文件, 從而得到這個文件的opened_path. 比如上面的例子, 這個文件存在於 “/tmp2/2.php”.

 

然後, 得到了這個opened_path以後, PHP去已加載的文件列表去查找, 是否已經包含, 如果沒有包含, 那麼就直接compile, 不再需要open file了.


 

1. 嘗試解析文件的絕對路徑, 如果能解析成功, 則檢查EG(included_files), 存在則返回, 不存在繼續

 

2. 打開文件, 得到文件的打開路徑(opened path)

 

3. 拿opened path去EG(included_files)查找, 是否存在, 如果存在則返回, 不存在繼續
 

4. 編譯文件(compile_file)

 
 

這個在大多數情況下, 不是問題, 然而問題出在當你使用APC的時候…

 
 

在使用APC的時候, APC劫持了compile_file這個編譯文件的指針, 從而直接從cache中得到編譯結果, 避免了對實際文件的open, 避免了對open的system call.

 

然而, 當你在代碼中使用include_once的時候, 在compile_file之前, PHP已經嘗試去open file了, 然後才進入被APC劫持的compile file中, 這樣一來, 就會產生一次額外的open操作. 而APC正是為了解決這個問題, 引入了include_once_override, 在include_once_override開啟的情況下, APC會劫持PHP的ZEND_INCLUDE_OR_EVAL opcode handler, 通過stat來確定文件的絕對路徑, 然後如果發現沒有被加載, 就改寫opcode為include, 做一個tricky解決方案.


 

但是, 很可惜, 如我所說, APC的include_once_override實現的一直不好, 會有一些未定義的問題, 比如:

 

 代碼如下 復制代碼

    <?php

    set_include_path("/tmp");

    function a($arg = array()) {

        include_once("b.php");

    }

    a();

    a();

    ?>

然後, 我們的b.php放置在”/tmp/b.php”, 內容如下:

 

 代碼如下 復制代碼

    <?php

      class B {}

    ?>

那麼在打開apc.include_once_override的情況下, 連續訪問就會得到如下錯誤:

 
Fatal error – include() : Cannot redeclare class

 

排除這些技術因素, 我也一直認為, 我們應該使用include, 而不是include_once, 因為我們完全能做到自己規劃, 一個文件只被加載一次. 還可以借助自動加載, 來做到這一點.

 

你使用include_once,只能證明, 你對自己的代碼沒信心.

 

所以, 建議大家, 不要再使用include_once,不過我建義大家使用autoload和spl_autoload自動加載

(1) autoload機制概述

在使用PHP的OO模式開發系統時,通常大家習慣上將每個類的實現都存放在一個單獨的文件裡,這樣會很容易實現對類進行復用,同時將來維護時也很便利。這 也是OO設計的基本思想之一。在PHP5之前,如果需要使用一個類,只需要直接使用include/require將其包含進來即可。下面是一個實際的例 子:

 代碼如下 復制代碼

/* Person.class.php */
<?php
class Person {
var $name, $age;

function __construct ($name, $age)
{
$this->name = $name;
$this->age = $age;
}
}
?>

/* no_autoload.php */
<?php
require_once (”Person.class.php”);

$person = new Person(”Altair”, 6);
var_dump ($person);
?>

在這個例子中,no-autoload.php文件需要使用Person類,它使用了require_once將其包含,然後就可以直接使用Person類來實例化一個對象。

但 隨著項目規模的不斷擴大,使用這種方式會帶來一些隱含的問題:如果一個PHP文件需要使用很多其它類,那麼就需要很多的require/include語 句,這樣有可能會造成遺漏或者包含進不必要的類文件。如果大量的文件都需要使用其它的類,那麼要保證每個文件都包含正確的類文件肯定是一個噩夢。

PHP5為這個問題提供了一個解決方案,這就是類的自動裝載(autoload)機制。autoload機制可以使得PHP程序有可能在使用類時才自動包含類文件,而不是一開始就將所有的類文件include進來,這種機制也稱為lazy loading。

下面是使用autoload機制加載Person類的例子:

 代碼如下 復制代碼

/* autoload.php */
<?php
function __autoload($classname) {
require_once ($classname . “class.php”);
}

$person = new Person(”Altair”, 6);
var_dump ($person);
?>

copyright © 萬盛學電腦網 all rights reserved