PHP 数组浅析

来源:互联网 发布:python enumerate 编辑:程序博客网 时间:2024/05/19 22:55

一、概要

  • 简单介绍及基本使用
  • php数组 常用标准库函数使用
  • php数组模拟常见数据结构
  • php数组使用须知与注意点
  • FAQ

二、简单介绍及基本使用

  • PHP 中的数组实际上是一个有序映射。映射是一种把 values 关联到 keys 的类型。
  • 通过<?php $arr = array(1, 2, 3, 4); 便定义了一个普通数组
  • 我们还可以定义关联数组
<?php  $arr = array('a' => 1, 'z' => 100); >
  • 如果php版本 >= 5.4 , 我们可以使用更为简洁的方式定义数组
<?php $arr = [1, 2, 3, 4]; $arr = ['a' => 1, 'z' => 100]; 
  • php 数组很强大, 可以定义混合类型数组
<?php $arr = [1, 'hello' => '11', 'arr' => [1, 'a'=>'b']];
  • 关于数组的访问操作,可以通过[index] 访问:
<?php $arr = [1, 'hello' => '11', 'arr' => [1, 'a'=>'b']];var_dump($arr[0]); // 1var_dump($arr['arr']); // [1, 'a' => 'b']
  • 也可以通过[] 修改数组元素的值
<?php $arr = [1, 'hello' => '11', 'arr' => [1, 'a'=>'b']];$arr[0] = 'test';var_dump($arr);  
  • 你也可以在初始化数组继续添加数组元素
<?php$arr = [1, 2, 3, 4]; //追加元素$arr[] = 'a';// 添加 key, value$arr['test'] = 'b';
  • 当然,删除数组元素的操作必须支持
<?php$arr = [1, 'hello' => '11', 'arr' => [1, 'a'=>'b']];unset($arr['hello']);var_dump($arr['hello']);  // null
  • 开发中经常需要遍历数组, 可使用 foreach :
<?php $arr = [1, 'hello' => '11', 'arr' => [1, 'a'=>'b']];foreach($arr as $key => $value) {    var_dump($key . ' => ' . $value);}

更多数组遍历的方法请参考php-数组遍历
- 数组之间的比较,数组无法比较大小, 但可以根据一定条件判断是否相等

<?php // $a == $b 相等 如果 $a$b 具有相同的键/值对则为 TRUE。// $a === $b 全等 如果 $a$b 具有相同的键/值对并且顺序和类型都相同则为 TRUE。$a = [1, 2];$b = ['1' => 2, 0 => 1];var_dump($a == $b); // truevar_dump($a === $b); // false

三、实用的数组工具函数

掌握了数组的基本操作(定义使用,增删改查, 遍历) 之后,你就可以在开发中使用数组, 但只有这些操作是远远不够的,为了满足复杂多变的开发场景对数组操作的需求,php 提供了功能强大的一套 Array 操作的函数
- 获取数组长度

<?php $arr = [1, 2, 3];var_dump(count($arr)); // 3
  • 希望判断一个变量是否是数组,可以通过 is_array():
<?php $arr = [1, 2, 3];$notArr = '111';var_dump(is_array($arr)); // truevar_dump(is_array($notArr)); // false
  • 更具key 或者 value, 判断元素 是否在数组中
// 判断key 是否在数组中$arr = ['a' => 2, 4];var_dump(isset($arr['a'])); // true  var_dump(array_key_exists('a', $arr)); // true// 判断 value 是否在数组中in_array(5, $arr);  // false
  • 获取数组所有键(keys)
<?php $arr = ['a' => 2, 4];$keys = array_keys($arr); // ['a', 1]
  • 获取数组的所有值 (values)
<?php $arr = ['a' => 2, 4];$values = array_values($arr); // [2, 4]
  • 统计数组各个元素值出现的次数 可以使用 array_count_values:
<?php$arr = [1, 3, 2, 'a' => 1, 'b' => 2];var_dump(array_count_values($arr));/*array(3) {  [1]=>  int(2)  [3]=>  int(1)  [2]=>  int(2)}*/

数组与数组之间的操作:数组可以看做一个集合, 集合间的操作(交集,差集,并集, 补集, 比较等) php 也提供相应的方法实现

  • 数组的合并
<?php$arr1 = ['a' => 1, 2, 'b' => 3];$arr2 = ['b' => 5, 2];var_dump( array_merge($arr1, $arr2) ); /*array(4) {  ["a"]=>  int(1)  [0]=>  int(2)  ["b"]=>  int(5)  [1]=>  int(2)}// 你也可以使用 + 操作符, 请注意两种方法结果的差别var_dump($arr1 + $arr2); */
  • 如果是需要计算两个或跟多数组value的交集, 可使用 array_intersect
<?php$arr1 = [1, 2, 3];$arr2 = [5, 2];var_dump( array_intersect($arr1, $arr2) );  // [2]
  • 数组的差集 (按值value 和 按key)
<?php$a = [1, 2];$b = ['1' => 2, 0 => 1, 4];//array_diff 按照索引 和 值 比较差异var_dump(array_diff($a, $b));// array_diff_key() 函数用于比较两个(或更多个)数组的键名 ,并返回差集 var_dump(array_diff_key($a, $b)); 
  • 如果需要获取子数组, 可以通过 array_slice 产生类似 python 切片的效果
<?php$arr = [1, 2, 3, 4, 5, 6, 7, 8];// 从第3个元素开始, 直到结束var_dump(array_slice($arr, 2));// 从第3个元素开始, 长度为4var_dump(array_slice($arr, 2, 4));// 从第3个元素开始,到倒数第3个元素var_dump(array_slice($arr, 2, -2));// 注意 索引的差别var_dump(array_slice($arr, 2, -2, true));
  • 关于数组的排序操作,也是比较常见的开发需求,需要注意的是:php排序函数都是直接作用于数组本身, 而不是返回一个新的有序的数组。, 以下代码提供几种常见的场景, 更多请参考php对数组进行排序:
<?php// 按照值(value)升序排序, 索引更新$arr = [6,'a'=>2, 3, 4, 6, -1, 7, 8];sort($arr);var_dump($arr);// 按照值(value)升序排序, 索引保持$arr = [6,'a'=>2, 3, 4, 6, -1, 7, 8];asort($arr);var_dump($arr);// 按照值(value)降序排序, 索引保持$arr = [6,'a'=>2, 3, 4, 6, -1, 7, 8];arsort($arr);var_dump($arr);// 按照 键(key)进行升序排序 , 索引保持$arr = ['a' => 10, 'c' => 1, 'b' => 12];ksort($arr);var_dump($arr);// 按照 键(key)进行降序排序 , 索引保持$arr = ['a' => 10, 'c' => 1, 'b' => 12];krsort($arr);var_dump($arr);// 用户自定义排序, 根据值(value) , 索引更新// 请注意:对于自定义的比较函数,// 在第一个参数小于,等于或大于第二个参数时,// 该比较函数必须相应地返回一个小于,等于或大于 0 的整数。function cmp($val1, $val2){    if($val1 > $val2) {        return 1;    } elseif ($val1 == $val2) {        return 0;    } else {        return -1;    }}$arr = [    'a' => 1,    'A' => 3,    'B' => 2,];usort($arr, cmp);var_dump($arr);// 根据key 自定义排序规则,请使用 uksort(), 用法同usort()
  • 关于数组与字符串之间的操作一般有切割字符串,合并数组元素转为字符串两种操作,可以借助explode与implode实现
<?phpvar_dump(explode(',', "a,a,a,a,a,a")); // 以,为分割符将字符串"a,a,a,a,a,a" 切割成数组var_dump(implode('-', [1, 2, 3, 4, 5])); //以 - 为 拼接符 将 数组[1, 2, 3, 4, 5] 拼接成字符串

关于php 更多数组相关的函数, 可以参考 官方文档php数组函数列表


四、数组模拟常见数据结构

php 数组可以模拟常见的数据结构,最显而易见的便是 映射表 和 字典, 这里简单介绍php数组对栈和队列的模拟。

  • 模拟栈(FILO)
<?php$stack = [1, 2, 3, 4];//入栈array_push($stack, -1);var_dump($stack); // [1, 2, 3, 4, -1]//出栈$e = array_pop($stack);var_dump($e); // -1var_dump($stack);  // [1, 2, 3, 4]
  • 模拟队列 (FIFO)
<?php$queue = [];//入队列array_unshift($queue, 1);array_unshift($queue, 2);array_unshift($queue, 3);array_unshift($queue, 4);//出队列$e = array_pop($queue);var_dump($e); // 1$e = array_pop($queue);var_dump($e); // 2$e = array_pop($queue);var_dump($e); // 3$e = array_pop($queue);var_dump($e); // 4

五、php数组使用须知与注意点

  • php 数组 key值 会存在以下强制转换
    • 包含有合法整型值的字符串会被转换为整型。例如键名 “8” 实际会被储存为 8。但是 “08” 则不会强制转换,因为其不是一个合法的十进制数值。
    • 浮点数也会被转换为整型,意味着其小数部分会被舍去。例如键名 8.7 实际会被储存为 8。
    • 布尔值也会被转换成整型。即键名 true 实际会被储存为 1 而键名 false 会被储存为 0。
    • Null 会被转换为空字符串,即键名 null 实际会被储存为 “”。
    • 数组和对象不能被用为键名。坚持这么做会导致警告:Illegal offset type。

因此以下代码可能导致意外的结果,请注意以下代码的输出

<?php$arr = [1, 2, '8' => 3];$arr[false] = -20;var_dump($arr); // [-20, 2, '8' => 3]$arr[8] = 20;var_dump($arr); // [-20, 2, 8 => 20]$arr[8.7] = 15;var_dump($arr); // [-20, 2, 8 => 15]$arr["8.7"] = 10;var_dump($arr); // [-20, 2, 8 => 10]$arr[$val]  = 5; // 注意$val之前为声明,因此默认值为null, 数组key为null时会被转为""空串var_dump($arr); // [-20, 2, 8 => 10, "" => 5]$arr[bar] = 6; // 标识符被转化为 'bar'var_dump($arr); // [-20, 2, 8 => 10, "" => 5, 'bar' => 6]
  • 关于php数组的类型转换
    php数组可以将其他一切类型转为数组,转化的效果请参考一下代码,重点观察对 null 和 object对象的转化:
<?php$var = true;var_dump((array)$var);/* array(1) {  [0]=>  bool(true)}*/$var = 1;var_dump((array)$var);/* array(1) {  [0]=>  int(1)}*/$var = 1.1;var_dump((array)$var);/* array(1) {  [0]=>  float(1.1)}*/$var = "111";var_dump((array)$var);/* array(1) {  [0]=>  string(3) "111"}*/$var = null;var_dump((array)$var);  // 返回空数组/* array(0) {} */class Cls { public $a = 1; protected $b = 2; private $c = 3; }var_dump((array)(new Cls)); // 可见性不同 key值格式有所不同/* array(3) {  ["a"]=>  int(1)  ["*b"]=>  int(2)  ["Clsc"]=>  int(3)} */

关于PHP类型转换的了解,请参考PHP-类型转换的判别


六、FAQ

  • 如何添加数组元素更为高效? array_push($arr, key, value) or $arr[key] = value ? 答: 后者更为高效, 更多细节请参考官方资料
  • isset or array_key_exists() ? 答:
    • 对于存在key的数组,如果 对应的value = null , isset($arr[key]) 会返回 false;而对于array_key_exists 只要对应key存在就会返回true;
    • 然而在效率方面,isset 效率 高于array_key_eixsts
      在判断数组元素是否存在的最佳实践如下:
<?php if (isset($arr[$key]) or array_key_exists($key, $arr)) { ...}
  • 数组合并 +array_merge 的区别? 答:可以参考该资料
  • array_diff== 的异同? 答:语义有所差别, 数组的相等比较 推荐只使用==
  • 遍历方式那种更高效? 答:foreach 方式 遍历 最为高效
3 0
原创粉丝点击