我有这样的代码:

private static $dates = array(
  'start' => mktime( 0,  0,  0,  7, 30, 2009),  // Start date
  'end'   => mktime( 0,  0,  0,  8,  2, 2009),  // End date
  'close' => mktime(23, 59, 59,  7, 20, 2009),  // Date when registration closes
  'early' => mktime( 0,  0,  0,  3, 19, 2009),  // Date when early bird discount ends
);

这给了我以下错误:

解析错误:语法错误,在/home/user/ sites /site/registration/inc/registration.class中出现意外的'(',期望')'。第19行

所以,我想我做错了什么…但如果不是那样,我怎么能这样做呢?如果我用常规字符串改变mktime,它就能工作。所以我知道我可以这样做。

有人有什么建议吗?


当前回答

如果你可以控制类加载,你可以从那里进行静态初始化。

例子:

class MyClass { public static function static_init() { } }

在你的类装入器中,执行以下操作:

include($path . $klass . PHP_EXT);
if(method_exists($klass, 'static_init')) { $klass::staticInit() }

一个更重量级的解决方案是使用ReflectionClass接口:

interface StaticInit { public static function staticInit() { } }
class MyClass implements StaticInit { public static function staticInit() { } }

在你的类装入器中,执行以下操作:

$rc = new ReflectionClass($klass);
if(in_array('StaticInit', $rc->getInterfaceNames())) { $klass::staticInit() }

其他回答

我结合了Tjeerd Visser和porneL的答案。

class Something
{
    private static $foo;

    private static getFoo()
    {
        if ($foo === null)
            $foo = [[ complicated initializer ]]
        return $foo;
    }

    public static bar()
    {
        [[ do something with self::getFoo() ]]
    }
}

但是更好的解决方案是去掉静态方法,使用单例模式。然后在构造函数中进行复杂的初始化。或者让它成为一个“服务”,并使用DI将其注入到任何需要它的类中。

在这部分代码中不能进行函数调用。如果你让init()类型的方法在其他代码执行之前执行,那么你就可以填充变量。

我更喜欢简单地创建一个getter函数,而不是寻找一种让静态变量工作的方法。如果需要属于特定类的数组,也很有帮助,实现起来也简单得多。

class MyClass
{
   public static function getTypeList()
   {
       return array(
           "type_a"=>"Type A",
           "type_b"=>"Type B",
           //... etc.
       );
   }
}

在需要该列表的任何地方,只需调用getter方法。例如:

if (array_key_exists($type, MyClass::getTypeList()) {
     // do something important...
}

如果你可以控制类加载,你可以从那里进行静态初始化。

例子:

class MyClass { public static function static_init() { } }

在你的类装入器中,执行以下操作:

include($path . $klass . PHP_EXT);
if(method_exists($klass, 'static_init')) { $klass::staticInit() }

一个更重量级的解决方案是使用ReflectionClass接口:

interface StaticInit { public static function staticInit() { } }
class MyClass implements StaticInit { public static function staticInit() { } }

在你的类装入器中,执行以下操作:

$rc = new ReflectionClass($klass);
if(in_array('StaticInit', $rc->getInterfaceNames())) { $klass::staticInit() }

这太复杂了,无法定义。你可以将定义设置为null,然后在构造函数中检查它,如果它没有被更改-设置它:

private static $dates = null;
public function __construct()
{
    if (is_null(self::$dates)) {  // OR if (!is_array(self::$date))
         self::$dates = array( /* .... */);
    }
}