36

您可以在 PHP 名称中为变量、常量、函数、方法、类...使用哪些有效字符?

该手册有一些 关于正则表达式的提及[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*。此限制何时适用,何时不适用?

4

2 回答 2

57

[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"
于 2013-07-31T14:36:21.640 回答
2

NikiC 拥有除功能之外的所有功能。使用变量函数,您可以使用任何您想要的名称,尽管语法很烂。

${'=^^='}=function(){
   echo "Hello Kitty!";
};

${'=^^='}(); // Outputs "Hello Kitty!"
于 2016-03-03T15:58:25.993 回答