Метод цепочки с переменными переменными
API, который я использую (ConnectPHP Oracle Service Cloud), использует подход цепочки. Например:
$incident = new Incident();
$incident->CustomFields->c->make = "Same value";
$incident->StatusWithType->Status->ID = 34;
$incident->save();
Как бы я пошел на достижение того же, если последующие свойства $incident
объект генерируется динамически? Например:
$data = array();
$data[0]['parts'] = array('CustomFields', 'c', 'make');
$data[0]['value'] = "Some value";
$data[1]['parts'] = array('StatusWithType', 'Status', 'ID');
$data[1]['value'] = 34;
$incident = new Incident();
foreach($data as $array)
{
foreach($array['parts'] as $key)
{
// how will I generate
// (1) $incident->CustomFields->c->make = $array['value']
// (2) $incident->StatusWithType->Status->ID = $array['value']
}
}
$incident->save();
Что я пробовал
$incident = new Incident();
foreach($data as $array)
{
$parts = implode('->', $array['parts']);
$incident->{$parts} = $array['value']; // this doesn't work even though $parts is coming out with the expected pattern because I think it is converting it into a string representation
}
$incident->save();
2 ответа
Если нет риска пользовательского ввода, вы можете создать строку всех ключей объекта и использовать eval, как это
$incident = new stdClass();
foreach($data as $key=>$chain){
$str = "{'".implode("'}->{'",$chain['parts'])."'}";
eval("@\$incident->$str = '$chain[value]';");
}
print_r($incident);
Живая демоверсия: https://eval.in/923232
ВЫХОД как
stdClass Object
(
[CustomFields] => stdClass Object
(
[c] => stdClass Object
(
[make] => Some value
)
)
[StatusWithType] => stdClass Object
(
[Status] => stdClass Object
(
[ID] => 34
)
)
)
и теперь вы можете легко получить доступ, как $incident->CustomFields->c->make
@kranthi технически прав (в комментарии), я дал реализацию.
Итак, Кранти был на правильном пути.
$incident = new Incident();
foreach($data as $array)
{
$this->setDynamicFields($incident, $array['parts'], $array['value']);
}
$incident->save();
function setDynamicFields($obj, $parts, $value)
{
if(is_array($parts) && count($parts) == 3)
{
$obj->{$parts[0]}->{$parts[1]}->{$parts[2]} = ($parts[0] == 'StatusWithType' ? (int) $value: $value);
}
}
Хитрость заключалась в том, чтобы пройти весь $incident
объект как аргумент функции (я думаю, это называется внедрением зависимости, если я не ошибаюсь) и использование ->
как литерал вместо строки, находящейся внутри переменной.