getter mágica __get para propriedades estáticas em PHP
-
16-09-2019 - |
Pergunta
public static function __get($value)
não trabalho, e mesmo que o fizesse, acontece que eu já precisa do getter mágica __get para propriedades de instância na mesma classe.
Este é provavelmente um sim ou não pergunta, então, é possível?
Solução
Não, não é possível.
Citando o href="http://php.net/manual/en/language.oop5.overloading.php#language.oop5.overloading.members" rel="noreferrer"> página do manual :
Membro sobrecarga só funciona em contexto objecto. Estes métodos mágicos não será acionado em estática contexto. Por conseguinte, estes métodos pode não ser declarado estático.
No PHP 5.3, __callStatic
tem sido adicionado; mas não há __getStatic
nem __setStatic
ainda; mesmo se a idéia de ter / codificação-los muitas vezes vem de volta os internos php @ mailling-lista.
Existe ainda um Request for Comments: Classes estáticas em PHP
Mas, ainda assim, não implementado (ainda?)
Outras dicas
Talvez alguém ainda precisa isto:
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");
}
}
Very nice mbrzuchalski. Mas parece funcionar apenas em variáveis ??públicas. Basta alterar a sua mudança para isso para permitir o acesso privadas / protegidas:
switch($match[1]) {
case 'get': return self::${$property->name};
case 'set': return self::${$property->name} = $args[0];
}
E você provavelmente gostaria de alterar a instrução if
para limitar as variáveis ??que são acessíveis, ou então ele iria derrotar o propósito de tê-los ser privada ou protegida.
if ($reflector->hasProperty($property) && in_array($property, array("allowedBVariable1", "allowedVariable2"))) {...)
Assim, por exemplo, eu tenho uma classe concebido para puxar vários dados para mim a partir de um servidor remoto usando um módulo ssh pêra, e eu quero fazer certas suposições sobre o diretório de destino com base em qual servidor ele está sendo convidado a olhar a. Uma versão beliscada do método de mbrzuchalski é perfeito para isso.
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.");
}
}
tente o seguinte:
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);
Além disso, você pode obter propriedades estáticas acessá-las como propriedades do membro, usando __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"
Combinando __callStatic
e call_user_func
ou call_user_func_array
pode dar acesso a propriedades estáticas na classe PHP
Exemplo:
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');