热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

你真的了解Array吗?

@TOC前言知其然,也要知其所以然PHP数组是什么?关于这个问题,官方文档给出了这样的解释:PHP中的数组实际上是一个有序映射。映射是一种把values关联到keys的类型。此类型






@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 "


";
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 "


";
}

值得注意的事:


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 '


';
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后面添加上该元素。



推荐阅读
  • javascript  – 概述在Firefox上无法正常工作
    我试图提出一些自定义大纲,以达到一些Web可访问性建议.但我不能用Firefox制作.这就是它在Chrome上的外观:而那个图标实际上是一个锚点.在Firefox上,它只概述了整个 ... [详细]
  • Python正则表达式学习记录及常用方法
    本文记录了学习Python正则表达式的过程,介绍了re模块的常用方法re.search,并解释了rawstring的作用。正则表达式是一种方便检查字符串匹配模式的工具,通过本文的学习可以掌握Python中使用正则表达式的基本方法。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • 集合的遍历方式及其局限性
    本文介绍了Java中集合的遍历方式,重点介绍了for-each语句的用法和优势。同时指出了for-each语句无法引用数组或集合的索引的局限性。通过示例代码展示了for-each语句的使用方法,并提供了改写为for语句版本的方法。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • JVM 学习总结(三)——对象存活判定算法的两种实现
    本文介绍了垃圾收集器在回收堆内存前确定对象存活的两种算法:引用计数算法和可达性分析算法。引用计数算法通过计数器判定对象是否存活,虽然简单高效,但无法解决循环引用的问题;可达性分析算法通过判断对象是否可达来确定存活对象,是主流的Java虚拟机内存管理算法。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
  • 本文介绍了在Android开发中使用软引用和弱引用的应用。如果一个对象只具有软引用,那么只有在内存不够的情况下才会被回收,可以用来实现内存敏感的高速缓存;而如果一个对象只具有弱引用,不管内存是否足够,都会被垃圾回收器回收。软引用和弱引用还可以与引用队列联合使用,当被引用的对象被回收时,会将引用加入到关联的引用队列中。软引用和弱引用的根本区别在于生命周期的长短,弱引用的对象可能随时被回收,而软引用的对象只有在内存不够时才会被回收。 ... [详细]
  • 本文总结和分析了JDK核心源码(2)中lang包下的基础知识,包括常用的对象类型包和异常类型包。在对象类型包中,介绍了Object类、String类、StringBuilder类、StringBuffer类和基本元素的包装类。在异常类型包中,介绍了Throwable类、Error类型和Exception类型。这些基础知识对于理解和使用JDK核心源码具有重要意义。 ... [详细]
  • php缓存ri,浅析ThinkPHP缓存之快速缓存(F方法)和动态缓存(S方法)(日常整理)
    thinkPHP的F方法只能用于缓存简单数据类型,不支持有效期和缓存对象。S()缓存方法支持有效期,又称动态缓存方法。本文是小编日常整理有关thinkp ... [详细]
  • 本文介绍了在Go语言中可见性与scope的规则,包括在函数内外声明的可见性、命名规范和命名风格,以及变量声明和短变量声明的语法。同时,还介绍了变量的生命周期,包括包级别变量和局部变量的生命周期,以及变量在堆和栈上分配的规则和逃逸分析的概念。 ... [详细]
author-avatar
低调pasta_730
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有