Magic __get get для статических свойств в PHP
public static function __get($value)
не работает, и даже если это сработало, так получилось, что мне уже нужен магический метод получения __get для свойств экземпляра в том же классе.
Вероятно, это вопрос да или нет, так что, возможно?
5 ответов
Нет, это невозможно.
Цитирую справочную страницу __get:
Перегрузка членов работает только в контексте объекта. Эти магические методы не будут запускаться в статическом контексте. Поэтому эти методы не могут быть объявлены статическими.
В PHP 5.3 __callStatic
был добавлен; но нет __getStatic
ни __setStatic
еще; даже если идея их / кодирования часто возвращается в php internals@ mailling-list.
Есть даже запрос комментариев: статические классы для PHP
Но все же не реализовано (пока?)
Может, кому-то еще это нужно
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");
}
}
Очень мило мбзучальски. Но, похоже, он работает только с общими переменными. Просто измените ваш переключатель на этот, чтобы разрешить ему доступ к частным / защищенным:
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.");
}
}
Попробуй это:
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);
Объединяя __callStatic
а также call_user_func
или же call_user_func_array
может дать доступ к статическим свойствам в классе PHP
Пример:
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');
Кроме того, вы можете получить статические свойства, обращаясь к ним как к свойствам элемента, используя __get():
class ClassName {
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"