>백엔드 개발 >PHP 튜토리얼 >다차원 배열을 탐색하기 위해 PHP foreach를 구현하는 방법

다차원 배열을 탐색하기 위해 PHP foreach를 구현하는 방법

高洛峰
高洛峰원래의
2016-12-12 10:34:121242검색

소개
일반적으로 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
*/
성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.