一塊兒學習PHP的runkit擴展如何使用

此次又爲你們帶來一個好玩的擴展。咱們知道,在 PHP 運行的時候,也就是部署完成後,咱們是不能修改常量的值,也不能修改方法體內部的實現的。也就是說,咱們編碼完成後,將代碼上傳到服務器,這時候,咱們想在不修改代碼的狀況去修改一個常量的值是不行的。常量自己就是不容許修改的。可是,runkit 擴展卻能夠幫助咱們完成這個功能。php

動態修改常量

define('A', 'TestA');

runkit_constant_redefine('A', 'NewTestA');

echo A; // NewTestA

是否是很神奇。這個 runkit 擴展就是在運行時可讓咱們來動態的修改一些常量、方法體及類的功能擴展。固然,從系統安全的角度來來,這個擴展並非很推薦。由於自己常量的含義就是不變的量,自己就不該該修改的。同理,在運行時動態的改變函數體或者類定義的內容都是會有可能影響到其它調用到這些函數或類的代碼,因此,這個擴展是一個危險的擴展。git

除了動態地修改常量外,咱們還可使用 runkit_constant_add() 、 runkit_constant_remove() 函數來動態地增長或者刪除常量。github

安裝

runkit 擴展的安裝是須要在 github 下載而後進行正常的擴展編譯便可,pecl 下載的已通過時了。shell

PHP5: http://github.com/zenovich/runkit安全

PHP7:https://github.com/runkit7/runkit7.git服務器

clone 成功後進行正常的擴展編譯安裝步驟便可。ide

phpize
./configure
make
make install

不一樣的 PHP 版本須要安裝不一樣版本的擴展,同時,runkit7 還在開發中,有一些函數尚未支持,好比:函數

  • runkit_class_adopt
  • runkit_class_emancipate
  • runkit_import
  • runkit_lint_file
  • runkit_lint
  • runkit_sandbox_output_handler
  • runkit_return_value_used
  • Runkit_Sandbox
  • Runkit_Sandbox_Parent

在寫這篇文章的測試代碼時,上述函數或者類都是不支持的。你們能夠用 PHP5 的環境測試下原版的擴展是否都能正常使用。測試

查看超全局變量鍵

print_r(runkit_superglobals());
//Array
//(
//    [0] => GLOBALS
//    [1] => _GET
//    [2] => _POST
//    [3] => _COOKIE
//    [4] => _SERVER
//    [5] => _ENV
//    [6] => _REQUEST
//    [7] => _FILES
//    [8] => _SESSION
//)

這個函數其實就是查看下當前運行環境中的全部超全局變量鍵名。這些都是咱們經常使用的一些超全局變量,就不一一解釋了。編碼

方法相關操做

方法操做就和常量操做同樣,咱們能夠動態地添加、修改、刪除以及重命名各類方法。首先仍是來看一下咱們最關心的在動態運行時來修改方法體裏面的邏輯代碼。

function testme() {
  echo "Original Testme Implementation\n";
}
testme(); // Original Testme Implementation
runkit_function_redefine('testme','','echo "New Testme Implementation\n";');
testme(); // New Testme Implementation

定義了一個 testme() 方法,而後經過 runkit_function_redefine() 來修改它的實現,最後再次調用 testme() 時輸出的就是新修改後的實現了。那麼,咱們能不能修改 PHP 自帶的那些方法呢?

// php.ini runkit.internal_override=1
runkit_function_redefine('str_replace', '', 'echo "str_replace changed!\n";');
str_replace(); // str_replace changed!

runkit_function_rename ('implode', 'joinArr' );
var_dump(joinArr(",", ['a', 'b', 'c'])); 
// string(5) "a,b,c"


array_map(function($v){
   echo $v,PHP_EOL;
},[1,2,3]);
// 1
// 2
// 3
runkit_function_remove ('array_map');

// array_map(function($v){
//   echo $v;
// },[1,2,3]);
// PHP Fatal error:  Uncaught Error: Call to undefined function array_map()

代碼裏的註釋說的很清楚了,咱們只須要在 php.ini 中設置 runkit.internal_override=1 ,就能夠動態地修改 PHP 自帶的那些方法函數了。好比第一段咱們修改了 str_replace() 方法,讓他直接就輸出了一段文字。而後咱們將 implode() 更名爲 joinArr() ,就能夠像 implode() 同樣來使用這個 joinArr() 。最後,咱們刪除了 array_map() 方法,若是再次調用這個方法,就會報錯。

類方法相關操做

類內部方法函數的操做和上面變量方法操做是相似的,不過對於 PHP 自帶的類咱們沒法進行修改之類的操做。這個你們能夠本身嘗試一下。

//runkit_method_add('PDO', 'testAddPdo', '', 'echo "This is PDO new Func!\n";');
//PDO::testAddPdo();
// PHP Warning:  runkit_method_add(): class PDO is not a user-defined class

從報錯信息能夠看出,PDO 類不是用戶定義的類,因此沒法使用 runkit 函數進行相關操做。那咱們就來看看咱們自定義的類是如何使用 runkit 來進行動態操做的吧。

class Example{
}

runkit_method_add('Example', 'func1', '', 'echo "This is Func1!\n";');
runkit_method_add('Example', 'func2', function(){
    echo "This is Func2!\n";
});
$e = new Example;
$e->func1(); // This is Func1!
$e->func2(); // This is Func2!

runkit_method_redefine('Example', 'func1', function(){
    echo "New Func1!\n";
});
$e->func1(); // New Func1!

runkit_method_rename('Example', 'func2', 'func22');
$e->func22(); // This is Func2!

runkit_method_remove('Example', 'func1');
//$e->func1();
// PHP Fatal error:  Uncaught Error: Call to undefined method Example::func1()

咱們定義了一個空類,而後動態給它添加了兩個方法,以後修改了方法1,重命名了方法2,最後刪除了方法1,一系列的操做其實和上面的普通方法的操做基本是同樣的。

總結

就像上面說過的同樣,這個擴展是比較危險的一個擴展,特別是若是開啓了 runkit.internal_override 後,咱們還可以修改 PHP 的原生函數。不過若是是必需要使用它的話,那麼它的這些功能就很是有用。就像 訪問者模式 同樣,「大多時候你並不須要訪問者模式,但當一旦你須要訪問者模式時,那就是真的須要它了」,這一套 runkit 擴展也是同樣的道理。

測試代碼:

https://github.com/zhangyue0503/dev-blog/blob/master/php/202006/source/%E4%B8%80%E8%B5%B7%E5%AD%A6%E4%B9%A0PHP%E7%9A%84runkit%E6%89%A9%E5%B1%95%E5%A6%82%E4%BD%95%E4%BD%BF%E7%94%A8.php

參考文檔:

https://www.php.net/manual/zh/book.runkit.php

https://www.php.net/manual/zh/book.runkit7.php

各自媒體平臺都可搜索【硬核項目經理】

相關文章
相關標籤/搜索