您可以在 PHP 名称中为变量、常量、函数、方法、类...使用哪些有效字符?
该手册有一些 关于正则表达式的提及[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*
。此限制何时适用,何时不适用?
正[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*
则表达式仅在名称直接用于某些特殊语法元素时适用。一些例子:
$varName // <-- varName needs to satisfy the regex
$foo->propertyName // <-- propertyName needs to satisfy the regex
class ClassName {} // <-- ClassName needs to satisfy the regex
// and can't be a reserved keyword
请注意,此正则表达式是逐字节应用的,而不考虑编码。这就是为什么它还允许许多奇怪的 Unicode 名称。
但是正则表达式仅限制这些“直接”使用名称。通过 PHP 提供的各种动态特性,几乎可以使用任意名称。
一般来说,您不应该假设PHP 中的名称可以包含哪些字符。在大多数情况下,它们只是任意字符串。做诸如“验证某事是否是有效的类名”之类的事情在 PHP 中是毫无意义的。
在下文中,我将提供如何为不同类别创建奇怪名称的示例。
变量名可以是任意字符串:
${''} = 'foo';
echo ${''}; // foo
${"\0"} = 'bar';
echo ${"\0"}; // bar
全局常量也可以是任意字符串:
define('', 'foo');
echo constant(''); // foo
define("\0", 'bar');
echo constant("\0"); // bar
没有办法动态定义我知道的类常量,所以这些不能是任意的。创建奇怪的类常量的唯一方法似乎是通过扩展代码。
属性不能是空字符串,也不能以 NUL 字节开头,但除此之外它们是任意的:
$obj = new stdClass;
$obj->{''} = 'foo'; // Fatal error: Cannot access empty property
$obj->{"\0"} = 'foo'; // Fatal error: Cannot access property started with '\0'
$obj->{'*'} = 'foo';
echo $obj->{'*'}; // foo
方法名称是任意的,可以通过__call
魔术处理:
class Test {
public function __call($method, $args) {
echo "Called method \"$method\"";
}
}
$obj = new Test;
$obj->{''}(); // Called method ""
$obj->{"\0"}(); // Called method "\0"
可以使用class_alias
空字符串创建任意类名:
class Test {}
class_alias('Test', '');
$className = '';
$obj = new $className; // Fatal error: Class '' not found
class_alias('Test', "\0");
$className = "\0";
$obj = new $className; // Works!
我不知道有一种从用户空间创建任意函数名称的方法,但是在某些情况下内部代码会产生“奇怪”的名称:
var_dump(create_function('',''));
// string(9) "\0lambda_1"
NikiC 拥有除功能之外的所有功能。使用变量函数,您可以使用任何您想要的名称,尽管语法很烂。
${'=^^='}=function(){
echo "Hello Kitty!";
};
${'=^^='}(); // Outputs "Hello Kitty!"