1. 程式人生 > >php中foreach遍歷類物件的總結

php中foreach遍歷類物件的總結

foreach 遍歷陣列很常見,同樣foreach也可以遍歷物件

做如下測試:

class my
{
	public $a = 'a';
	protected $b = 'b';
	private $c = 'c';
	private $data = array('fantasy','windows','linux');
	// 內部foreach遍歷class
	function traversable()
	{
		foreach($this as $key=>$val)
		{
			echo $key.'=>';
			print_r($val);
			echo '<br>';
		}
	}
}
$m = new my();
// 外部foreach遍歷class
foreach($m as $key=>$val)
{
	echo $key.'=>';
	print_r($val);
	echo '<br>';
}
echo '--------------------------<br>';
// 內部foreach遍歷class
$m->traversable();

輸出結果如下:
a=>a
--------------------------
a=>a
b=>b
c=>c
data=>Array ( [0] => fantasy [1] => windows [2] => linux ) 

由此可知,對於外部的foreach遍歷是沒有許可權訪問 protected private 這兩個修飾的屬性的,而在class內部是有許可權訪問,foreach可以遍歷所有的屬性。

今天在寫PDO的時候發現可以這樣寫: 

foreach($db->query('SELECT * FROM tab') as $row)
{
    print_r($row);
}

這樣快速的獲取了全部查詢結果,可奇怪的是$this->query() 返回的是 object型別 PDOStatement ,var_dump()打印出來的結果是這樣的:
object(PDOStatement)#2 (1) {
  ["queryString"]=>
  string(18) "SELECT * FROM user"
}

PDOStatement裡面就一個public屬性 queryString  並且foreach也沒有出現這個值,這樣的情況就不是簡單的對屬性進行遍歷了,而是class繼承了iterator迭代器,在foreach的時候會執行class裡面的迭代方式,遍歷迭代器指定的資料

關於迭代器看下面的例子:

class test implements Iterator
{
	public $a = 'a';
	private $data = array('apple','banlance','current');
	private $point = 0;
	public function __construct()
	{
		$this->point = 0;
	}
	public function current()
	{
		return $this->data[$this->point];
	}
	public function key()
	{
		return $this->point;
	}
	public function next()
	{
		++$this->point;
	}
	public function rewind()
	{
		$this->point=0;
	}
	public function valid()
	{
		return isset($this->data[$this->point]);
	}
}
$t = new test();
foreach($t as $val)
{
	print_r($val);
	echo '<br>';
}

輸出結果如下:
apple
banlance
current

test class 實現iterator的介面,foreach呼叫的時候會使用這個介面方法,呼叫過程大致如下面虛擬碼:
// 迭代過程虛擬碼
while(valid)
{
<span style="white-space:pre">	</span>current/key
<span style="white-space:pre">	</span>next
}
rewind
so,之前的foreach對class的處理過程是一種預設方法,如果是繼承iterator的class被foreach遍歷的時候是上面這種方式

由此情況去套用 PDO的寫法還是行不通,因為如果我們var_dump上面的哪個test類結果是這樣的:

test Object
(
    [a] => a
    [data:test:private] => Array
        (
            [0] => apple
            [1] => banlance
            [2] => current
        )

    [point:test:private] => 0
)

但是當我們var_dump $db->query返回的物件時並沒有見到point這個iterator介面中定義的屬性以及遍歷的資料 $data;

由此我們可以猜測PDOStatement繼承了一種迭代的介面但是並不是iterator

檢視手冊可以發現:

PDOStatement implements Traversable 
檢視Traversable的介紹如下圖:



由此明白了,PDOStatement的迭代實現都是在內部,繼承iterator是php指令碼的實現方式。

大致總結下:

foreach是可以遍歷陣列的,也可以遍歷物件。物件只能羅列出public的屬性,如果想要foreach羅列出保護的屬性可以讓class繼承iterator並實現其中的方法,這樣foreach遍歷一個class的時候是按照class內部實現的iterator進行處理的。
-------------------------------------------------------------
PDO的問題:

PDO::query() 返回的是物件 PDOStatement (繼承的Traversable這個空介面,必須由Iterator or iteratorAggregate 介面實現)。
PDOStatement 實現了Iterator介面的方法,其實現方法中操作的就是非public修飾的屬性,這個屬性裡面儲存的是查詢結果集。
至此,foreach($db->query('sql..') as $row) 的執行過程明白了