PHP: проверьте, является ли объект/массив ссылкой
Извините, пожалуйста, поздно, и я не могу понять, как это сделать... любой может помочь?
$users = array(
array(
"name" => "John",
"age" => "20"
),
array(
"name" => "Betty",
"age" => "22"
)
);
$room = array(
"furniture" => array("table","bed","chair"),
"objects" => array("tv","radio","book","lamp"),
"users" => &$users
);
var_dump $показывает комнаты:
...
'users' => &
...
Это означает, что "пользователи" являются ссылкой.
Я хотел бы сделать что-то вроде этого:
foreach($room as $key => $val) {
if(is_reference($val)) unset($room[$key]);
}
Основная цель - скопировать массив без каких-либо ссылок.
Возможно ли это?
Спасибо.
Ответы
Ответ 1
Вы можете протестировать ссылки в многомерном массиве, сделав копию массива, а затем изменив и протестировав каждую запись по очереди:
$roomCopy = $room;
foreach ($room as $key => $val) {
$roomCopy[$key]['_test'] = true;
if (isset($room[$key]['_test'])) {
// It a reference
unset($room[$key]);
}
}
unset($roomCopy);
С вашими данными примера $room['furniture']
и $roomCopy['furniture']
будут отдельными массивами (поскольку $roomCopy
- это копия $room
), поэтому добавление нового ключа к одному не повлияет на другое. Но $room['users']
и $roomCopy['users']
будут ссылками на тот же массив $users
(как на ссылку, скопированную, а не на массив), поэтому, когда мы добавляем ключ в $roomCopy['users']
, он отображается в $room['users']
.
Ответ 2
Лучшее, что я могу сделать, это тест двух переменных, чтобы определить, является ли ссылка ссылкой на другую:
$x = "something";
$y = &$x;
$z = "something else";
function testReference(&$xVal,&$yVal) {
$temp = $xVal;
$xVal = "I am a reference";
if ($yVal == "I am a reference") { echo "is reference<br />"; } else { echo "is not reference<br />"; }
$xVal = $temp;
}
testReference($x,$y);
testReference($y,$x);
testReference($x,$z);
testReference($z,$x);
testReference($y,$z);
testReference($z,$y);
но я сомневаюсь, что это поможет
Действительно грязный метод (не очень хорошо протестирован):
$x = "something";
$y = &$x;
$z = "something else";
function isReference(&$xVal) {
ob_start();
debug_zval_dump(&$xVal);
$dump = ob_get_clean();
preg_match('/refcount\((\d*)\)/',$dump,$matches);
if ($matches[1] > 4) { return true; } else { return false; }
}
var_dump(isReference($x));
var_dump(isReference($y));
var_dump(isReference($z));
Чтобы использовать этот последний метод в вашем коде, вам нужно сделать что-то вроде:
foreach($room as $key => $val) {
if(isReference($room[$key])) unset($room[$key]);
}
потому что $val никогда не является ссылкой, поскольку это копия исходного элемента массива; и использование & $val делает его всегда ссылкой
Ответ 3
что-то рекурсивное возможно.
function removeReferences($inbound)
{
foreach($inbound as $key => $context)
{
if(is_array($context))
{
$inbound[$key] = removeReferences($context)
}elseif(is_object($context) && is_reference($context))
{
unset($inbound[$key]); //Remove the entity from the array.
}
}
return $inbound;
}
Ответ 4
function var_reference_count(&$xVal) {
$ao = is_array($xVal)||is_object($xVal);
if($ao) { $temp= $xVal; $xVal=array(); }
ob_start();
debug_zval_dump(&$xVal);
$dump = ob_get_clean();
if($ao) $xVal=$temp;
preg_match('/refcount\((\d*)\)/',$dump,$matches);
return $matches[1] - 3;
}
//-------------------------------------------------------------------------------------------
Это работает с объектами и массивами HUDGE.
Ответ 5
если вы хотите избавиться от рекурсивных элементов:
<?php
$arr=(object)(NULL); $arr->a=3; $arr->b=&$arr;
//$arr=array('a'=>3, 'b'=>&$arr);
print_r($arr);
$arr_clean=eval('return '.strtr(var_export($arr, true), array('stdClass::__set_state'=>'(object)')).';');
print_r($arr_clean);
?>
выход:
stdClass Object ( [a] => 3 [b] => stdClass Object *RECURSION* )
stdClass Object ( [a] => 3 [b] => )