소개
일반적으로 foreach는 1차원 배열의 각 키 => 값을 순서대로 인쇄할 수 있지만, 다차원 배열인 경우 중첩 루프로 루프를 반복하거나 재귀적으로 구현해야 합니다. , 그러나 이러한 방법 중 어느 것도 충분히 유연하지 않습니다. 왜냐하면 배열의 차원 수를 알지 못하면 끝없는 중첩 루프를 가질 수 없기 때문입니다. 재귀를 사용하면 해결할 수 있지만 foreach를 사용하여 루프하려는 경우 모든 루프를 어떻게 구현하나요?
구현 방법 1
PHP 자체 반복자 클래스 RecursiveIteratorIterator 사용
$test_arr = array(1,2,3,array(4,'aa'=>5,6,array(7,'bb'=>8),9,10),11,12); $arrayiter = new RecursiveArrayIterator($test_arr); $iteriter = new RecursiveIteratorIterator($arrayiter); //直接打印即可按照横向顺序打印出来 foreach ($iteriter as $key => $val){ echo $key.'=>'.$val; } //结果 /* 0=>1 1=>2 2=>3 0=>4 aa=>5 2=>6 0=>7 bb=>8 4=>9 5=>10 4=>11 5=>12 */
구현 방법 2
구현 RecursiveIteratorIterator와 유사한 iterator 클래스를 사용하여 다차원 배열의 가로 인쇄 기능을 직접 구현
class foreachPrintfArr implements Iterator { //当前数组作用域 private $_items; private $_old_items; //保存每次执行数组环境栈 private $_stack = array(); public function __construct($data=array()){ $this->_items = $data; } private function _isset(){ $val = current($this->_items); if (empty($this->_stack) && !$val) { return false; } else { return true; } } public function current() { $this->_old_items = null; $val = current($this->_items); //如果是数组则保存当前执行环境,然后切换到新的数组执行环境 if (is_array($val)){ array_push($this->_stack,$this->_items); $this->_items = $val; return $this->current(); } //判断当前执行完成后是否需要切回上次执行环境 //(1) 如果存在跳出继续执行 //(2) 如果不存在且环境栈为空,则表示当前执行到最后一个元素 //(3) 如果当前数组环境下一个元素不存在,则保存一下当前执行数组环境 $this->_old_items = $this->_items; //然后切换上次执行环境 $this->_items = array_pop($this->_stack) 继续循环, 直到当前数组环境下一个 //元素不为空为止 while (1) { if (next($this->_items)) { prev($this->_items); break; } elseif (empty($this->_stack)) { end($this->_items); break; } else { end($this->_items); if (!$this->_old_items) $this->_old_items = $this->_items; $this->_items = array_pop($this->_stack); } } return $val; } public function next() { next($this->_items); } public function key() { // 由于 key() 函数执行在 current() 函数之后 // 所以在 current() 函数切换执行环境 , 会导致切换之前的执行环境最后一个 key // 变成切换之后的key , 所以 $this->_old_items 保存一下切换之前的执行环境 // 防止key打印出错 return $this->_old_items ? key($this->_old_items) : key($this->_items); } public function rewind() { reset($this->_items); } public function valid() { return $this->_isset(); } }
내부 실행 방법
1. Foreach는 사용자 정의된 foreachPrintfArr 클래스를 반복합니다. 이 클래스는 5가지 내부 메서드(valid(), rewind(), key(), next() 및 current()))를 자동으로 호출합니다.
2. 호출 순서:
되감기 -> 현재 -> 키
다음 -> -> 현재 -> 키
$test_arr = array(1,2,3,array(4,'aa'=>5,6,array(7,'bb'=>8),9,10),11,12); $iteriter = new foreachPrintfArr($test_arr); foreach ($iteriter as $key => $val){ echo $key.'=>'.$val; } //结果: /* 0=>1 1=>2 2=>3 0=>4 aa=>5 2=>6 0=>7 bb=>8 4=>9 5=>10 4=>11 5=>12 */