要使用數組中的元素,就需要數組的定位。定位的實現需要藉助數組指標來完成。PHP中有很多函數可以移動數組指標。下面將給大家介紹幾個。
將指標移動到下一個數組位置next()
next()函數返回緊接著放在當前數組指標下一個位置的數組值。其形式如下:
以下是一個例子:
$fruits = array("apple", "banana", "orange", "pear");echo next($fruits);echo next($fruits);// banana// orange
還可以將指標前移,也可以直接移動到數組的開始和結尾位置。
將指標移動到前一個數組位置prev()
prev()函數返回位於當前指標前一個位置的數組值,如果指標本來就位於數組的第一個位置,則返回false。其形式如下:
prev()的用法與next()相同,例子就省略了。
將指標移到第一個數組位置reset()
reset()函數用於將數組指標設定回數組的開始位置。其形式如下:
如果需要在指令碼中多次查看或處理一個數組,就經常使用這個函數,另外這個函數還經常在排序結束時使用。
將指標移動到最後一個數組位置end()
end()函數將指標移動到數組的最後一個位置,將返回最後一個元素。其形式如下:
下面的例子展示了如果擷取第一個和最後一個數組值:
$fruits = array("apple", "banana", "orange", "pear");echo current($fruits);echo end($fruits);// apple// pear
對於數組作為參數在函數間傳遞時又存在這樣一個規律:我們知道,函數調用時,系統會將實參copy一份賦值給形參(引用調用除外),而對於數組,不僅僅 copy了實參的值,而且還copy了實參數組當前內部指標的位置。如果實參內部指標的位置指向了數組末尾,那麼系統會將形參的內部指標重設,指向形參數 組的第一個單元;如果實參內部指標的位置不在數組末尾,即指向了有效單元,那麼系統會將形參的數組指標位置與實參的數組指標指向值相同的數組單元。
如果不做 $arr['var6'] = 6 這一步操作,6個變數($var1-$var6)都將有值,因為在each之後,數組指標已經指向了數組的末尾,那麼在調用函數 func()時,系統會重設 $arrtmp的數組指標,將其指向第一個元素。但是在進行 $arr['var6'] = 6這步操作之後,一切就改變了,這一個操作讓$arr的數組指標由原來指向一個NULL變成了一個有效值(說明一下,賦值前後,數組指標指向的地址單元一 直是沒有變化的,只不過是賦值前,那個地址單元什麼也沒有,而賦值之後變成了6)。這就使得$arr的數組指標指向了一個有效單元,那麼在調用函數 func()時,系統是不會重設$arrtmp的數組指標的,$arrtmp的數組指標將會跟$arr的數組指標一樣,指向他自己的最後一個單元。而 each函數又是從當前數組指標的位置開始工作的。因此each函數操作的第一個結果的傳回值就是數組$arrtmp的最後一個元素了,它將數組指標再向下移動一位,while迴圈到此結束,因此$arrtmp['var1']-$arrtmp['var5']都沒有被遍曆到,最終導致$var1-$var6為NULL。
數組在賦值的過程中,賦值數組和被賦值數組各自數組指標的變化情況。 先給出一條結論,然後我們在用代碼來證明這個結論吧。$arrtmp=$arr;在這個賦值運算式中我把$arr叫做賦值數組,把$arrtmp叫做被賦值數組。數 組在賦值時,如果賦值數組的數組指標已經指向了數組末尾,則賦值之後賦值數組的數組指標會被重設,指向數組第一個元素;如果在賦值時,賦值數組的數組指標 沒有指向數組末尾,而是指向了任何一個有效數組元素,那麼在賦值之後賦值數組的數組指標是不會被重設的,而是保留其原來指向的元素。在賦值之後,被賦值 數組不僅有了賦值數組的值,而且賦值數組的數組指標指向了那個元素,被賦值的數組也會指向自己中值相同的那個元素。
demo1:
<?php $arr = array('var1'=>1,'var2'=>2,'var3'=>3,'var4'=>4,'var5'=>5); while( list($key,$value) = each($arr) ) { if($value == 4) break; } var_dump(current($arr)); $arr1 = $arr; var_dump(current($arr)); var_dump(current($arr1));?>
demo1 的執行結果是:int(5) int(5) int(5) 。從這個結果可以看出,賦值前後$arr的數組指標位置沒有發生任何變化,$arr1不僅值跟$arr相同,而且數組指標所指向的元素值也是相同的。現在 用上述結論來解釋這個結果,在while迴圈中,有一個if判斷語句,目的是不讓$arr的數組指標指向數組末尾,而是保留在一個有效位置。 在$value=4時會跳出迴圈,而each這個函數會將數組指標向前移動一位,這就導致了$arr的數組指標指向了第5個元素,所以在賦值之 前,current($arr)的結果是5,賦值之後,由於在賦值之前$arr的當前指標並沒有指向末尾,因此在賦值之後不會將$arr的數組指標進行重 置,而是保留了其原有的位置,因此在賦值之後使用current($arr)的結果仍然是5。賦值時$arr1不僅獲得了$arr的值,而且數組指標指向 的元素和$arr的相同,二者都是5。
<?php$arr = array('var1'=>1,'var2'=>2,'var3'=>3,'var4'=>4,'var5'=>5);while( list($key,$value) = each($arr) ){ //if($value == 4) break;}var_dump(current($arr));$arr1 = $arr;var_dump(current($arr));var_dump(current($arr1));?>
demo2中我們將 if($value == 4) break; 這一句注釋掉了,目的很簡單,就是通過each將$arr的數組指標位置指向數組末尾。
demo2 的執行結果:bool(false) int(1) bool(false) 。如果數組指標對應的元素為0,"",或者不是一個有效值時,current函數會返回false,$arr的值中沒有為0或者""的情況,因此可以斷 定是因為數組指標指向了一個無效的元素而導致current返回了一個false。換句話說就是可以確定在while迴圈完成之後,$arr的數組指標已 經指向了數組的末尾。所以我們看到在賦值之前current($arr)的值是false,而賦值之後current($arr)的值變成了1,說明賦值 之後$arr的數組指標被重設了,指向了數組的第一個元素。current($arr1)的值為false,說明賦值之後$arr1讓然保留了賦值之 前$arr的數組指標指向的元素。
通過demo1和demo2就可以證明上述結論了。
因此為了在遍曆數組時不受數組指標的影響,最好在使用each()函數之前或者之後調用函數reset()將數組指標重設。這樣就可以避免上述問題的發生了。另外還有一個運算元組指標的函數prev(),它的作用是將數組指標當前的位置後退一位,它也需要注意一點,就是如果數組指標已經指向數組末尾,那麼使它就得不到想要的結果了。
順便說一下foreach這個函數,使用foreach函數來遍曆數組時,它會重設數組指標,將其指向數組的第一個元素。必須注意的是foreach操作的對象是對你要遍曆的數組的copy值,而不是遍曆數組本身。