需要分析PHP代碼的效能,或者說實現同樣功能的代碼到底哪個更好呢?或者說想知道底層的實現可以使用VLD查看opcode
下載與安裝VLD
# wget http://pecl.php.net/get/vld-0.11.2.tgz
# tar zxvf vld-0.11.2.tgz
# cd ./vld-0.11.2
# /usr/local/php/bin/phpize 或者直接phpize
# ./configure --with-php-config=/usr/local/php/bin/php-config --enable-vld
# make && make install
---------------------------------
編輯php.ini文件啟用vld擴充:
php.ini位置
#cd /usr/local/php/lib
增加
extension=vld.so
重啟Apache:
# /usr/local/apache2/bin/apachectl restart
---------------------------------
查看phpinfo()資訊
--------------------------------
至此,VLD就安裝完了。寫個簡單的test.php
$a='123'; echo $a;
# php -dvld.active=1 ./test.php
如果沒有設定php環境變數的話
#/usr/local/php/bin/php -dvld.active=1 test.php
查看結果
如上為VLD輸出的PHP代碼產生的中間代碼的資訊,說明如下:
- Branch analysis from position 這條資訊多在分析數組時使用。
- Return found 是否返回,這個基本上有都有。
- filename 分析的檔案名稱
- function name 函數名,針對每個函數VLD都會產生一段如上的獨立的資訊,這裡顯示當前函數的名稱
- number of ops 產生的運算元
- compiled vars 編譯期間的變數,這些變數是在PHP5後添加的,它是一個緩衝最佳化。這樣的變數在PHP源碼中以IS_CV標記。
- op list 產生的中間代碼的變數列表
使用-dvld.active參數輸出的是VLD預設設定,如果想看更加詳細的內容。可以使用-dvld.verbosity參數。
#php -dvld.active=1 -dvld.verbosity=3 text.php
-dvld.verbosity=3是VLD在目前的版本可以顯示的最詳細的資訊.
如果我們只是想要看輸出的中間代碼,並不想執行這段PHP代碼,可以使用-dvld.execute=0來禁用代碼的執行
#php -dvld.active=1 -dvld.execute=0 text.php
VLD擴充的參數列表:
- -dvld.active 是否在執行PHP時啟用VLD掛鈎,預設為0,表示禁用。可以使用-dvld.active=1啟用。
- -dvld.skip_prepend 是否跳過php.ini設定檔中auto_prepend_file指定的檔案, 預設為0,即不跳過包含的檔案,顯示這些包含的檔案中的代碼所產生的中間代碼。此參數生效有一個前提條件:-dvld.execute=0
- -dvld.skip_append 是否跳過php.ini設定檔中auto_append_file指定的檔案, 預設為0,即不跳過包含的檔案,顯示這些包含的檔案中的代碼所產生的中間代碼。此參數生效有一個前提條件:-dvld.execute=0
- -dvld.execute 是否執行這段PHP指令碼,預設值為1,表示執行。可以使用-dvld.execute=0,表示只顯示中間代碼,不執行產生的中間代碼。
- -dvld.format 是否以自訂的格式顯示,預設為0,表示否。可以使用-dvld.format=1,表示以自己定義的格式顯示。這裡自訂的格式輸出是以-dvld.col_sep指定的參數間隔
- -dvld.col_sep 在-dvld.format參數啟用時此函數才會有效,預設為 “\t”。
- -dvld.verbosity 是否顯示更詳細的資訊,預設為1,其值可以為0,1,2,3 其實比0小的也可以,只是效果和0一樣,比如0.1之類,但是負數除外,負數和效果和3的效果一樣 比3大的值也是可以的,只是效果和3一樣。
- -dvld.save_dir 指定檔案輸出的路徑,預設路徑為/tmp。
- -dvld.save_paths 控制是否輸出檔案,預設為0,表示不輸出檔案
- -dvld.dump_paths 控制輸出的內容,現在只有0和1兩種情況,預設為1,輸出內容
使用VLD比較代碼差異
代碼text1.php
$var = 111;$str = "AAA " . $var . " BBB";
代碼text2.php
$var = 111;$str = "AAA $var BBB";
從結果很清晰的看出第一段代碼比第二段代碼多了concat
第一個串連操作,將“test string begin ”和$var串連起來,得到“AAA 111”,然後再執行第二個串連操作,將上一個操作得到的結果“AAA 111”和” BBB”串連起來,並將結果儲存在另一個臨時變數,最後將第二個串連操作的結果賦值給$str。
串連操作對應的opcode為ZEND_CONCAT,對於所給的兩個運算元,其最終通過concat_function函數將兩個字串串連起來,如果所給的變數的類型不是字串,則會通過zend_make_printable_zval將其轉換成字串。concat_function函數會根據兩個字串的長度重新分配記憶體,並執行兩次拷貝操作,將兩個字串拷貝到新的記憶體空間。這裡針對兩個字串相同的情況有一個特殊處理。
if (result==op1) {/* special case, perform operations on result */uint res_len = Z_STRLEN_P(op1) + Z_STRLEN_P(op2); Z_STRVAL_P(result) = erealloc(Z_STRVAL_P(result), res_len+1); memcpy(Z_STRVAL_P(result)+Z_STRLEN_P(result), Z_STRVAL_P(op2), Z_STRLEN_P(op2));Z_STRVAL_P(result)[res_len]=0;Z_STRLEN_P(result) = res_len;} else {Z_STRLEN_P(result) = Z_STRLEN_P(op1) + Z_STRLEN_P(op2);Z_STRVAL_P(result) = (char *) emalloc(Z_STRLEN_P(result) + 1);memcpy(Z_STRVAL_P(result), Z_STRVAL_P(op1), Z_STRLEN_P(op1));memcpy(Z_STRVAL_P(result)+Z_STRLEN_P(op1), Z_STRVAL_P(op2), Z_STRLEN_P(op2));Z_STRVAL_P(result)[Z_STRLEN_P(result)] = 0;Z_TYPE_P(result) = IS_STRING;}
而直接在字串中插入變數,其所有的操作都是添加操作,將字串添加到傳回值,將變數添加到傳回值,
所有的結果返回都是在一個臨時變數中,如我們的樣本,首先會將”AAA “添加到臨時變數,然後將臨時變數和$var變數添加到臨時變數,之後將臨時變數和” BBB”添加到臨時變數,最後將此此時變數賦值給$str。這裡添加將字串添加到臨時變數,其對應的opcode為ZEND_ADD_STRING,將變數添加到臨時變數,其對應的opcode為ZEND_ADD_VAR,雖然這兩個操作的opcode不同,但其最終調用都是add_string_to_string,他們所不同的調用此函數的第三個參數,一個是作業碼儲存的ZVAL變數,一個是通過變更列表擷取的ZVAL變數。
如果覺得需要看C語音層級的php源碼,可以參考:使用strace查看C語言層級的php源碼
如果你要查看memcpy可以去這個網站
http://linux.about.com/od/commands/l/blcmdl.htm
搜尋結果在:http://linux.about.com/library/cmd/blcmdl3_memcpy.htm
========================
延伸閱讀參考:
http://www.phppan.com/2011/05/vld-extension/
PHP中的字串串連操作
http://blog.csdn.net/phpkernel/article/details/5718519
http://www.laruence.com/2008/08/19/338.html