文字列に基づいてPHPのプロパティを取得するにはどうすればよいですか? magic
と呼びます。 magic
とは何ですか?
$obj->Name = 'something';
$get = $obj->Name;
のようになります...
magic($obj, 'Name', 'something');
$get = magic($obj, 'Name');
このような
<?php
$prop = 'Name';
echo $obj->$prop;
または、クラスを制御できる場合は、 ArrayAccess インターフェイスを実装し、これを実行します
echo $obj['Name'];
中間変数を作成せずにプロパティにアクセスする場合は、{}
表記を使用します。
$something = $object->{'something'};
また、たとえば、ループ内でプロパティ名を作成することもできます。
for ($i = 0; $i < 5; $i++) {
$something = $object->{'something' . $i};
// ...
}
あなたが求めているのは Variable Variables と呼ばれるものです。必要なことは、文字列を変数に保存し、次のようにアクセスすることだけです。
$Class = 'MyCustomClass';
$Property = 'Name';
$List = array('Name');
$Object = new $Class();
// All of these will echo the same property
echo $Object->$Property; // Evaluates to $Object->Name
echo $Object->{$List[0]}; // Use if your variable is in an array
このようなもの?テストしていませんが、正常に動作するはずです。
function magic($obj, $var, $value = NULL)
{
if($value == NULL)
{
return $obj->$var;
}
else
{
$obj->$var = $value;
}
}
プロパティ名を変数に保存し、変数を使用してプロパティにアクセスするだけです。このような:
$name = 'Name';
$obj->$name = 'something';
$get = $obj->$name;
シンプルです。$ obj-> {$ obj-> Name}中括弧は、変数変数のようにプロパティをラップします。
これはトップ検索でした。しかし、$ thisを使用していた私の質問は解決しませんでした。私の状況では、中括弧を使用することも助けました...
code Igniter get instanceの例
親クラスインスタンスを持つ何かと呼ばれるソースライブラリクラス
$this->someClass='something';
$this->someID=34;
親インスタンスを使用して別のクラスからソースする必要があるライブラリクラス
echo $this->CI->{$this->someClass}->{$this->someID};
この質問には答えがあるかもしれませんが、これらのPHP 7への移行を見たいかもしれません
ソース: php.net
他の誰かが深さ不明の深いプロパティを見つけたい場合、すべての子のすべての既知のプロパティをループする必要なく、以下を思いつきました。
たとえば、$ Foo-> Bar-> baz、または$ Foo-> baz、または$ Foo-> Bar-> Baz-> daveを見つけるには、$ pathは「foo/bar/baz」のような文字列です。
public function callModule($pathString, $delimiter = '/'){
//split the string into an array
$pathArray = explode($delimiter, $pathString);
//get the first and last of the array
$module = array_shift($pathArray);
$property = array_pop($pathArray);
//if the array is now empty, we can access simply without a loop
if(count($pathArray) == 0){
return $this->{$module}->{$property};
}
//we need to go deeper
//$tmp = $this->Foo
$tmp = $this->{$module};
foreach($pathArray as $deeper){
//re-assign $tmp to be the next level of the object
// $tmp = $Foo->Bar --- then $tmp = $Bar->baz
$tmp = $tmp->{$deeper};
}
//now we are at the level we need to be and can access the property
return $tmp->{$property};
}
そして、次のようなもので呼び出します:
$propertyString = getXMLAttribute('string'); // '@Foo/Bar/baz'
$propertyString = substr($propertyString, 1);
$moduleCaller = new ModuleCaller();
echo $moduleCaller->callModule($propertyString);
これが私の試みです。一般的な「愚かさ」チェックが組み込まれているため、利用できないメンバーを設定または取得しようとしないでください。
これらの「property_exists」チェックをそれぞれ__setおよび__getに移動し、magic()内で直接呼び出すことができます。
<?php
class Foo {
public $Name;
public function magic($member, $value = NULL) {
if ($value != NULL) {
if (!property_exists($this, $member)) {
trigger_error('Undefined property via magic(): ' .
$member, E_USER_ERROR);
return NULL;
}
$this->$member = $value;
} else {
if (!property_exists($this, $member)) {
trigger_error('Undefined property via magic(): ' .
$member, E_USER_ERROR);
return NULL;
}
return $this->$member;
}
}
};
$f = new Foo();
$f->magic("Name", "Something");
echo $f->magic("Name") , "\n";
// error
$f->magic("Fame", "Something");
echo $f->magic("Fame") , "\n";
?>
この関数は、子のいずれかのこのクラスにプロパティが存在するかどうかをチェックし、存在する場合は値を取得し、そうでない場合はnullを返します。そのため、プロパティはオプションで動的になりました。
/**
* check if property is defined on this class or any of it's childes and return it
*
* @param $property
*
* @return bool
*/
private function getIfExist($property)
{
$value = null;
$propertiesArray = get_object_vars($this);
if(array_has($propertiesArray, $property)){
$value = $propertiesArray[$property];
}
return $value;
}
使用法:
const CONFIG_FILE_PATH_PROPERTY = 'configFilePath';
$configFilePath = $this->getIfExist(self::CONFIG_FILE_PATH_PROPERTY);