まあ、それは無理だと思うので、本当に新しいインスタンスを使わなければなりませんでした。
結果は次のとおりです。
error_reporting(E_ALL^E_STRICT);//removes the "Strict Standards" warning, don't do this
final class session implements ArrayAccess {
//session data
private static $s=null;
//function (setters and getters)
private static $f=array('s'=>array(),'g'=>array());//general setter/getter
private $c=array('s'=>array(),'g'=>array());//private setter/getter
function setter($k,$f=null)
{
if(($a=func_num_args())<=1)return!$a?false:((@$this)?(@$this->c['s'][$k]):(@self::$f['s'][$k]));
else if($f==null){if(@$this)unset($this->c['s'][$k]);else unset(self::$f['s'][$k]);return true;}
else return is_callable($f)&&(@$this?$this->c['s'][$k]=$f:self::$f['s'][$k]=$f);
}
function getter($k,$f=null)
{
if(($a=func_num_args())<=1)return!$a?false:((@$this)?(@$this->c['g'][$k]):(@self::$f['g'][$k]));
else if($f==null){if(@$this)unset($this->c['g'][$k]);else unset(self::$f['g'][$k]);return true;}
else return is_callable($f)&&(@$this?$this->c['g'][$k]=$f:self::$f['g'][$k]=$f);
}
//==== arrayaccess ====
//extra: setters and getters implemented when accessing $session['offset']
// setter: gives the value, expects a value in return
// getter: gives the value, returns the value returned by the getter
function offsetSet($o,$v){@self::$s[$o]=(@$this->c['s'][$o])?$this->c['s'][$o]($v,$o):(@self::$f['s'][$o]?call_user_func(self::$f['s'][$o],$v,$o):@self::$s[$o]);}
function offsetExists($o){return isset(self::$s[$o]);}
function offsetUnset($o){unset(self::$s[$o]);}
function offsetGet($o){return(@$this->c['g'][$o])?$this->c['g'][$o](@self::$s[$o]):(@self::$f['g'][$o]?call_user_func(self::$f['g'][$o],@self::$s[$o]):@self::$s[$o]);}
//==== arrayaccess ====
}
$s=new session();
$s->setter('test',function($ob){return$ob>6?'bigger than 6':'lower than 6';});
session::setter('test',function($st){return$st<5?'lower than 5':'bigger than 5';});
session::setter('test',null);//*deletes the setter,*/ session::setter('test');//return setter
$s['test']=5;
var_dump($s->setter('test'),session::setter('test'),$s['test']);
//expected output (php 5.3.0): object(Closure)#2 (1) { ["parameter"]=> array(1) { ["$ob"]=> string(10) "" } } NULL string(12) "lower than 6"
見た目はひどいですが、意図したとおりに機能します。
すべてのデータはすべての新しいインスタンスで共有され、一般的なコンテキストのすべてのセッターは期待どおりに実行されます。
このコードを使用する場合は、次のことを覚えておいてください。
session::[sg]etter('test',null);
と同じではありませんsession::[sg]etter('test');
!
セッターには 2 つのパラメーター (値とオフセット) がありますが、ゲッターには 1 つ (値) しかありません。
セッターとゲッターの両方が値を返さなければなりません!
コメントの1つで、私はこれを言いました:
セッター メソッドがありますが、 $class=new my_class(); を実行することは不可能だと思います。$class->setter($param); および my_class::setter($param);
まあ、それは不可能ではありません、それは標準ではありません (そのため、最初の行は ですerror_reporting(E_ALL^E_STRICT);
)。この行がないと、PHP を実行しようとするとおかしくなりますsession::setter('test');
。