web-dev-qa-db-ja.com

PHPの静的プロパティのマジック__getゲッター

public static function __get($value)

動作せず、動作したとしても、同じクラスのインスタンスプロパティに魔法の__get getterがすでに必要になっています。

これはおそらく「はい」または「いいえ」の質問なので、可能ですか?

58
treznik

いいえ、できません。

__ getのマニュアルページ を引用:

メンバーのオーバーロードは、オブジェクトコンテキストでのみ機能します。これらのマジックメソッドは、静的コンテキストではトリガーされません。したがって、これらのメソッドは静的に宣言できません。


PHP 5.3、 __callStatic が追加されましたが、__getStatic__setStaticもまだありません;それらを持っている/コーディングするというアイデアがphp internals @ mailling-listにしばしば戻ってきたとしても。

コメントのリクエスト:PHPの静的クラス もあります
しかし、まだ、実装されていません(yet?)

61
Pascal MARTIN

誰かがまだこれを必要としているかもしれません:

static public function __callStatic($method, $args) {

  if (preg_match('/^([gs]et)([A-Z])(.*)$/', $method, $match)) {
    $reflector = new \ReflectionClass(__CLASS__);
    $property = strtolower($match[2]). $match[3];
    if ($reflector->hasProperty($property)) {
      $property = $reflector->getProperty($property);
      switch($match[1]) {
        case 'get': return $property->getValue();
        case 'set': return $property->setValue($args[0]);
      }     
    } else throw new InvalidArgumentException("Property {$property} doesn't exist");
  }
}
16
mbrzuchalski

とても素敵なmbrzuchalski。ただし、パブリック変数でのみ機能するようです。スイッチをこれに変更して、プライベート/保護されたものにアクセスできるようにします。

switch($match[1]) {
   case 'get': return self::${$property->name};
   case 'set': return self::${$property->name} = $args[0];
}

そして、おそらくアクセス可能な変数を制限するためにifステートメントを変更する必要があります。そうしないと、変数をプライベートまたは保護する目的が失われます。

if ($reflector->hasProperty($property) && in_array($property, array("allowedBVariable1", "allowedVariable2"))) {...)

したがって、たとえば、ssh pearモジュールを使用してリモートサーバーからさまざまなデータを引き出すために設計されたクラスがあり、どのサーバーを見るように求められているかに基づいてターゲットディレクトリについて特定の仮定を立てたいです。 mbrzuchalskiの方法を微調整したバージョンは、これに最適です。

static public function __callStatic($method, $args) {
    if (preg_match('/^([gs]et)([A-Z])(.*)$/', $method, $match)) {
        $reflector = new \ReflectionClass(__CLASS__);
        $property = strtolower($match[2]). $match[3];
        if ($reflector->hasProperty($property)) {
            if ($property == "server") {
                $property = $reflector->getProperty($property);
                switch($match[1]) {
                    case 'set':
                        self::${$property->name} = $args[0];
                        if ($args[0] == "server1") self::$targetDir = "/mnt/source/";
                        elseif($args[0] == "server2") self::$targetDir = "/source/";
                        else self::$targetDir = "/";
                    case 'get': return self::${$property->name};
                }
            } else throw new InvalidArgumentException("Property {$property} is not publicly accessible.");
        } else throw new InvalidArgumentException("Property {$property} doesn't exist.");
    }
}
4
Claymore

これを試して:

class nameClass{
    private static $_sData = [];
    private static $object = null;
    private $_oData = [];

    public function __construct($data=[]){
        $this->_oData = $data;
    }

    public static function setData($data=[]){
        self::$_sData = $data;
    }

    public static function Data(){
        if( empty( self::$object ) ){
            self::$object = new self( self::$_sData ); 
        }
        return self::$object;
    }

    public function __get($key) {
        if( isset($this->_oData[$key] ){
            return $this->_oData[$key];
        }
    }

    public function __set($key, $value) {
        $this->_oData[$key] = $value;
    }
}

nameClass::setData([
    'data1'=>'val1',
    'data2'=>'val2',
    'data3'=>'val3',
    'datan'=>'valn'
]);

nameClass::Data()->data1 = 'newValue';
echo(nameClass::Data()->data1);
echo(nameClass::Data()->data2);
0
Gilberto

__callStaticおよびcall_user_funcまたはcall_user_func_arrayは、PHP classの静的プロパティへのアクセスを与えることができます

例:

class myClass {

    private static $instance;

    public function __construct() {

        if (!self::$instance) {
            self::$instance = $this;
        }

        return self::$instance;
    }

    public static function __callStatic($method, $args) {

        if (!self::$instance) {
            new self();
        }

        if (substr($method, 0, 1) == '$') {
            $method = substr($method, 1);
        }

        if ($method == 'instance') {
            return self::$instance;
        } elseif ($method == 'not_exist') {
            echo "Not implemented\n";
        }
    }

    public function myFunc() {
        echo "myFunc()\n";
    }

}

// Getting $instance
$instance = call_user_func('myClass::$instance');
$instance->myFunc();

// Access to undeclared
call_user_func('myClass::$not_exist');
0
Jonny

また、__ get()を使用して、メンバープロパティのように静的プロパティにアクセスできます。

class ClassName {    
    private static $data = 'smth';

    function __get($field){
        if (isset($this->$field)){
            return $this->$field;
        }
        if(isset(self::$$field)){  
            return self::$$field;  // here you can get value of static property
        }
        return NULL;
    }
}

$obj = new ClassName();
echo $obj->data; // "smth"
0
Jokerius