我知道多维数组上有很多答案,但我找不到我到底在找什么。我是PHP的新手,不能完全理解其他一些示例来修改它们。如果有人能给我指路,我将不胜感激。
一个外部服务传递给我以下多维数组
$mArray = Array (
[success] => 1
[errors] => 0
[data] => Array (
[0] => Array (
[email] => me@example.com
[id] => 123456
[email_type] => html
[ip_opt] => 10.10.1.1
[ip_signup] =>
[member_rating] => X
[info_changed] => 2011-08-17 08:56:51
[web_id] => 123456789
[language] =>
[merges] => Array (
[EMAIL] => me@example.com
[NAME] => Firstname
[LNAME] => Lastname
[ACCOUNT] => ACME Ltd
[ACCMANID] => 123456adc
[ACCMANTEL] => 1234 123456
[ACCMANMAIL] => an.other@example.com
[ACCMANFN] => Humpty
[ACCMANLN] => Dumpty
)
[status] => unknown
[timestamp] => 2011-08-17 08:56:51
[lists] => Array ( )
[geo] => Array ( )
[clients] => Array ( )
[static_segments] => Array ( )
)
)
)
我唯一感兴趣的信息是键/值对,在键名"合并"下保存在数组中。大概是第三列纵深。数组的键名总是称为merges,但不能保证它在数组中的位置不会被移动。merge数组中的键/值对的数量也是可以改变的。
我认为我需要的是array_walk_recursive($mArray,"myfunction",$search)的函数;,其中$search保存了我正在寻找的键名(合并)的字符串。它需要遍历数组,直到找到键,检查它是否保存一个数组,然后(保留键),将每个键/值对返回到单个数组中。
因此,为了清晰起见,函数的输出将返回: $sArray = Array (
[EMAIL] => me@example.com
[NAME] => Firstname
[LNAME] => Lastname
[ACCOUNT] => ACME Ltd
[ACCMANID] => 123456adc
[ACCMANTEL] => 1234 123456
[ACCMANMAIL] => an.other@example.com
[ACCMANFN] => Humpty
[ACCMANLN] => Dumpty
)
然后我可以进入项目的下一步,将单个merge数组中的键与从HTML DOM Parser获得的元素id进行比较,并用单个数组中包含的属性值替换属性值。
我可能需要foreach循环。我知道我可以使用is_array来验证$search是否是一个数组。这让我很纠结,谢谢你的帮助。
这样行吗?
function find_merges($arr)
{
foreach($arr as $key => $value){
if($key == "merges") return $value;
if(is_array($value)){
$ret = find_merges($value);
if($ret) return $ret;
}
}
return false;
}
它会做深度优先搜索,直到你用完键或找到一个值为merges
的键。它不会检查merges
是否是一个数组。试一下,告诉我是否有效。
这是一个通用函数,它将通过嵌套数组工作,并返回与所提供键的第一次出现相关联的值。它允许整数或字符串键。如果没有找到匹配的键,则返回false。
// return the value a key in the supplied array
function get_keyval($arr,$mykey)
{
foreach($arr as $key => $value){
if((gettype($key) == gettype($mykey)) && ($key == $mykey)) {
return $value;
}
if(is_array($value)){
return get_keyval($value,$mykey);
}
}
return false;
}
// test it out
$myArray = get_keyval($suppliedArray, "merges");
foreach($myArray as $key => $value){
echo "$key = $valuen";
}
递归函数可以做到这一点。失败时返回array或FALSE
。
function search_sub_array ($array, $search = 'merges') {
if (!is_array($array)) return FALSE; // We're not interested in non-arrays
foreach ($array as $key => $val) { // loop through array elements
if (is_array($val)) { // We're still not interested in non-arrays
if ($key == $search) {
return $val; // We found it, return it
} else if (($result = search_sub_array($array)) !== FALSE) { // We found a sub-array, search that as well
return $result; // We found it, return it
}
}
}
return FALSE; // We didn't find it
}
// Example usage
if (($result = search_sub_array($myArray,'merges')) !== FALSE) {
echo "I found it! ".print_r($result,TRUE);
} else {
echo "I didn't find it :-(";
}
你想在数组中访问数组中的数组吗?
$mergeArray = NULL;
foreach($mArray['data'] as $mmArray)
$mergeArray[] = $mmArray['merges'];
之类的?如果merges
始终是3,我不明白为什么需要递归。
这是另一种方法,主要是因为我今天还没有用完我的迭代器配额。
$search = new RegexIterator(
new RecursiveIteratorIterator(
new ParentIterator(new RecursiveArrayIterator($array)),
RecursiveIteratorIterator::SELF_FIRST),
'/^merges$/D', RegexIterator::MATCH, RegexIterator::USE_KEY
);
$search->rewind();
$merges = $search->current();
array_walk_recursive()
非常适合这个任务!它不关心键值对在什么级别,它只迭代"叶节点",所以不需要检查元素是否包含字符串。在函数内部,我只是对键和针数组进行比较,以生成一维结果数组($sArray
)。
为了清楚起见,我假设您的merges
子数组中有可预测的键。
代码(演示):
$needles=['EMAIL','NAME','LNAME','ACCOUNT','ACCMANID','ACCMANTEL','ACCMANMAIL','ACCMANFN','ACCMANLN'];
array_walk_recursive($mArray,function($v,$k)use(&$sArray,$needles){if(in_array($k,$needles))$sArray[$k]=$v;});
var_export($sArray);
输出:array (
'EMAIL' => 'me@example.com',
'NAME' => 'Firstname',
'LNAME' => 'Lastname',
'ACCOUNT' => 'ACME Ltd',
'ACCMANID' => '123456adc',
'ACCMANTEL' => '1234 123456',
'ACCMANMAIL' => 'an.other@example.com',
'ACCMANFN' => 'Humpty',
'ACCMANLN' => 'Dumpty',
)