你真的了解 Array 吗?

@TOC

前言

知其然,也要知其所以然

PHP数组是什么?

关于这个问题,官方文档给出了这样的解释:

PHP 中的数组实际上是一个有序映射。映射是一种把 values 关联到 keys
的类型。此类型在很多方面做了优化,因此可以把它当成真正的数组,或列表(向量),散列表(是映射的一种实现),字典,集合,栈,队列以及更多可能性。由于数组元素的值也可以是另一个数组,树形结构和多维数组也是允许的。

值得注意的是,和JAVA、C 等静态语言不同。在PHP中,初始化数组的时候不必指定数组的大小和存储数据的类型,这既是PHP的优点,也是它的缺点。

PHP数组如何分类?

通常来讲,PHP数组一般分为 索引数组关联数组 两类。
当然,你也可以从其他方向分类数组。如:可以按照数组维度,分为 一维数组多维数组 等。
从数组的定义中我们可以知道,数组其实就是键值对的有序映射。这也就意味着,数组中的每一个元素都是以 key => value 的形式存在。当一个数组中元素的每一个 key 都是数字的时候,那么这个数组就称为 索引数组 ;反之,如果存在 key 不是数字的数组就称为 关联数组

遍历数组的方式有哪些?

1. for

for 循环是 PHP 中最复杂的循环结构。它的行为和 C 语言的相似。 for 循环的语法是:

for (expr1; expr2; expr3)
statement 第一个表达式(expr1)在循环开始前无条件求值(并执行)一次。

expr2 在每次循环开始前求值。如果值为 TRUE,则继续循环,执行嵌套的循环语句。如果值为 FALSE,则终止循环。

expr3 在每次循环之后被求值(并执行)。

每个表达式都可以为空或包括逗号分隔的多个表达式。表达式 expr2 中,所有用逗号分隔的表达式都会计算,但只取最后一个结果。

值得注意的是:expr2 为空意味着将无限循环下去(和 C 一样,PHP 暗中认为其值为 TRUE)。这可能不像想象中那样没有用,因为经常会希望用有条件的 break 语句来结束循环而不是用 for 的表达式真值判断。如:

$arr = ['a','b','c'];

for ($i = 0; ; $i++) {
    echo $arr[$i];
    if ($arr[$i] === 'c')
        break;
}

for循环是我们常用的一种遍历数组的方式,它可以能方便的对数组进行遍历,并操作数组的每一个元素,一个简单的示例:

$arr = ['a','b','c'];

for ($i = 0; $i < count($arr); $i++) {
    echo $arr[$i];
}

这种写法或许是刚入行的coder最常用的写法,但是它可能是糟糕的(可能会导致执行很慢)。存在一个问题,每一次循环都要使用 count() 函数计算一次数组 $arr 的长度,由于数组的长度始终不变,一种更优的方法是先将数组长度用中间变量保存起来,而不是多次调用 count() ,如下:

$arr = ['a','b','c'];

for ($i = 0, $len = count($arr); $i < $len; $i++) {
    echo $arr[$i];
}

另外,PHP 也支持用冒号的 for 循环的替代语法。

$arr = ['a','b','c'];

for ($i = 0, $len = count($arr); $i < $len; $i++):
    echo $arr[$i];
endfor;

for 循环的适用范围很广,可以用于索引数组、关联数组,也可用于循环字符串等……

$arr = [
    'a' => 'a',
    'b' => 'b',
    'c' => 'c'
];

//循环打印关联数组值
for ($i = 'a', $j = 0; $j <count($arr); $j++, $i = chr(ord($i) + 1) ) {
    echo $arr[$i];
}

$str = 'abcdefg';
//依次打印字符串元素
for ($i = 0, $len = strlen($str); $i < $len; $i++):
    echo $str[$i];
    echo "<hr/>";
endfor;
2. foreach

foreach 语法结构提供了遍历数组的简单方式。
foreach 仅能够应用于数组和对象,如果尝试应用于其他数据类型的变量,或者未初始化的变量将发出错误信息。有两种语法:
foreach (array_expression as $value)
statement

foreach (array_expression as $key => $value)
statement
第一种格式遍历给定的 array_expression 数组。每次循环中,当前单元的值被赋给 $value 并且数组内部的指针向前移一步(因此下一次循环中将会得到下一个单元)。

第二种格式做同样的事,只不过当前单元的键名也会在每次循环中被赋给变量 $key。

一个简单的示例:

$arr = [
    'a' => 'a',
    'b' => 'b',
    'c' => 'c'
];

foreach ($arr as $value) {
    echo $value;
}  //只获取值

foreach ($arr as $key => $value) {
    echo $key;
    echo $value;
}  //获取键 和 值

如果想在 foreach 内部修改数组的元素也是可以的,可以很容易地通过在 $value 之前加上 & 来修改数组的元素。此方法将以引用赋值而不是拷贝一个值。例如:

$arr = [
    'a' => 'a',
    'b' => 'b',
    'c' => 'c'
];

foreach ($arr as &$value) {
    $value = $value.'zzz';
}  //只获取值

var_dump($arr); //array(3) { ["a"]=> string(4) "azzz" ["b"]=> string(4) "bzzz" ["c"]=> &string(4) "czzz" }

此处需要注意的是,$value 的引用仅在被遍历的数组可以被引用时才可用(例如是个变量)。以下代码则无法运行:

foreach (array(1, 2, 3, 4) as &$value) {
    $value = $value * 2;
}

*Warning
数组最后一个元素的 $value 引用在 foreach 循环之后仍会保留,建议使用 unset() 来将其销毁。因为这及其容易造成一些问题,如:foreach 造成的奇怪问题

Note:
当 foreach 开始执行时,数组内部的指针会自动指向第一个单元。这意味着不需要在 foreach 循环之前调用 reset()。
由于 foreach 依赖内部数组指针,在循环中修改其值将可能导致意外的行为。
foreach 不支持用“@”来抑制错误信息的能力。

foreach 遍历对象
值得一提的是,foreach只能遍历可见的属性,这意味着在对象外部使用 foreach ,只能访问 public 类型的属性。如果需要遍历 protected类型的属性,就需要在本类或其子类内部定义方法使用 foreach。而遍历 private 类型的属性,则必须在其本类中进行。

3. 使用 list()、each()遍历数组

所涉及的函数有:
list() :把数组中的值赋给一组变量【像 array() 一样,这不是真正的函数,而是语言结构】
each() :返回数组中当前的键/值对并将数组指针向前移动一步
reset :将数组的内部指针指向第一个单元,并返回数组第一个单元的值,如果数组为空则返回 FALSE。

示例如下:

$arr = [
    'ak' => 'av',
    'bk' => 'bv',
    'ck' => 'cv'
];
reset($arr);  //开始循环之前,需调用此函数将数组内部指针指向第一个单元。避免指针为指向第一个单元,造成数组循环不全或循环失败的情况
while (list($k, $v) = each($arr)) {
    echo $k.'=>'.$v;
    echo "<hr/>";
}

值得注意的事:

1.自PHP7.2.0开始,each() 函数已被弃用。非常不推荐使用此函数。【如在PHP 7.3.4版本中调用,会返回错误(Deprecated: The each() function is deprecated. This message will be suppressed on further calls in …….)】
2.PHP 5 里,list() 从最右边的参数开始赋值; PHP 7 里,list() 从最左边的参数开始赋值。如果你用单纯的变量,不用担心这一点。 但是如果你用了具有索引的数组,通常你期望得到的结果和在 list()
中写的一样是从左到右的,但在 PHP 5 里实际上不是, 它是以相反顺序赋值的。
通常而言,不建议依赖于操作的顺序,在未来可能会再次发生修改。

4. 使用数组指针遍历数组

所涉及的函数有:
reset :将数组的内部指针指向第一个单元,并返回数组第一个单元的值,如果数组为空则返回 FALSE。

key():函数返回数组中内部指针指向的当前单元的键名。 但它不会移动指针。如果内部指针超过了元素列表尾部,或者数组是空的,此函数会返回 NULL。

current():current() 函数返回当前被内部指针指向的数组单元的值,并不移动指针。如果内部指针指向超出了单元列表的末端,current() 返回 FALSE。【值得注意的是,如果数组某一个元素的值就等于 (bool) false,此函数也会返回 FALSE,所以不能使用此函数判断是否到数组末尾】

next():next() 和 current() 的行为类似,只有一点区别,在返回值之前将内部指针向前移动一位。这意味着它返回的是下一个数组单元的值并将数组指针向前移动了一位。【同样,如果数组某一个元素的值就等于 (bool) false,此函数也会返回 FALSE,所以不能使用此函数判断是否到数组末尾】

$arr = [
    'ak' => 'av',
    'false' => false,
    false => 'cv'
];

while (true) {
    $k  = key($arr);
    if (!isset($k)) {
        reset($arr); //结束之前应将数组内部指针指向第一个单元
        unset($k);
        break;
    }
    echo $k.'=>'.current($arr);
    echo '<hr/>';
    next($arr);
}

一些值得注意的事:

1$arr[true] 等价于 $arr[1]$arr[false] 等价于 $arr[0]2、使null做为键名,相当于创建或覆盖一个$arr[null],可以使用$arr[null]$arr[""]来访问。

3、使用带小数点的数字作为键名时,键名会自动截取整数部分作为键名。如$arr[123.45]=5,你使用$arr[123.45]$arr[123]均可以取得键值;用foreach遍历时,使用的是$arr[123]4$arr[]=5,会在数组$arr后面添加上该元素。
本作品采用《CC 协议》,转载必须注明作者和本文链接
讨论数量: 0
(= ̄ω ̄=)··· 暂无内容!

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!