作者:ude816 | 来源:互联网 | 2023-10-13 09:13
这次又为大家带来一个好玩的扩大。咱们晓得,在PHP运行的时候,也就是部署实现后,咱们是不能批改常量的值,也不能批改办法体外部的实现的。也就是说,咱们编码实现后,将代码上传到服务器,这时候,咱们想在不批改代码的状况去批改一个常量的值是不行的。常量自身就是不容许批改的。然而,runkit扩大却能够帮忙咱们
这次又为大家带来一个好玩的扩大。咱们晓得,在 PHP 运行的时候,也就是部署实现后,咱们是不能批改常量的值,也不能批改办法体外部的实现的。也就是说,咱们编码实现后,将代码上传到服务器,这时候,咱们想在不批改代码的状况去批改一个常量的值是不行的。常量自身就是不容许批改的。然而,runkit 扩大却能够帮忙咱们实现这个性能。
动静批改常量
define('A', 'TestA');
runkit_constant_redefine('A', 'NewTestA');
echo A; // NewTestA
是不是很神奇。这个 runkit 扩大就是在运行时能够让咱们来动静的批改一些常量、办法体及类的性能扩大。当然,从系统安全的角度来来,这个扩大并不是很举荐。因为自身常量的含意就是不变的量,自身就不应该批改的。同理,在运行时动静的扭转函数体或者类定义的内容都是会有可能影响到其它调用到这些函数或类的代码,所以,这个扩大是一个危险的扩大。
除了动静地批改常量外,咱们还能够应用 runkit_constant_add() 、 runkit_constant_remove() 函数来动静地减少或者删除常量。
装置
runkit 扩大的装置是须要在 github 下载而后进行失常的扩大编译即可,pecl 下载的曾经过期了。
PHP5: http://github.com/zenovich/runkit
PHP7:https://github.com/runkit7/runkit7.git
clone 胜利后进行失常的扩大编译装置步骤即可。
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
各自媒体平台均可搜寻【硬核项目经理】