牛骨文教育服务平台(让学习变的简单)
博文笔记

PHP数组查找操作的函数及技巧

创建时间:2014-12-24 投稿人: 浏览次数:1022
PHP数组查找操作的函数: 1、bool in_array (mixed $needle , array $haystack [, bool $strict ] ) 该函数的作用是在数组array中搜索指定的value值, $strict是可选参数,如果设置该参数为 true ,则检查搜索的数据与数组的值的类型是否相同, 如果不使用第三个参数,比较一个数字和字符串或者比较涉及到数字内容的字符串,则字符串会被转换为数值并且比较按照数值来进行, 所以当in_array的needle与array中的值类型是否相同还不确定时,最好设置in_array函数的第三个参数 strict = true,这样在检查的时候会检查类型,数字与字符串就不会偷偷相等,也就避免类似这种问题。 如果比较一个整数和字符串,则字符串会被转换为整数。如果比较两个数字字符串,则作为整数比较。此规则也适用于 switch语句。 例1:
array("22",33,44,55,6)in_array("22",$array) //这样就会有一个把‘22’转换为22的操作,再进行比较var_dump(0 == "a"); // 0 == 0 -> truevar_dump("1" == "01"); // 1 == 1 -> truevar_dump("10" == "1e1"); // 10 == 10 -> truevar_dump(100 == "1e2"); // 100 == 100 -> true

例2:

$test = 0; $array = array("a", "b", "c");if (in_array($test, $array, true)) {    echo "in array";} else {    echo "no";}
php比较数字和字符串时,会尝试将字符串转换为数字再进行比较。 例子中的 "a", "b", "c" 转成数字都是0,所以和0相等,in_array就返回了true。

null 或 string string 将 NULL 转换为 "",进行数字或词汇比较
bool 或 null 任何其它类型 转换为 bool,FALSE < TRUE
object object 内置类可以定义自己的比较,不同类不能比较,相同类和数组同样方式比较属性(PHP 4 中),PHP 5 有其自己的说明
string,resource 或 number string,resource 或 number 将字符串和资源转换成数字,按普通数学比较
array array 具有较少成员的数组较小,如果运算数 1 中的键不存在于运算数 2 中则数组无法比较,否则挨个值比较(见下例)
array 任何其它类型 array 总是更大
object 任何其它类型 object 总是更大


2、bool array_key_exists ( mixed $key , array $search ) ?  为什么 array_key_exists 比 in_array快? in_array 是顺序查找 array_keys 是查哈希表
这里我了解到PHP Array的KEY是进行HASH组织的,查询很快。而VALUE是由KEY组织存放,本身没有索引,每次查找都是遍历。  array_key_exists 会调用get_defined_vars判断数组变量是否存在,而不是去遍历。
array_key_exists 和 in_array 查询的东西都不一样吧array_key_exists 判断是否有键值array_key_exists(a,arr)->if(isset(arr[a]))就是true而in_array 需要去遍历值 遍历到了才跳出循环

3、mixed array_search ( mixed $needle , array $haystack [, bool $strict ] ) array_search() 函数与 in_array() 一样,在数组中查找一个键值。如果找到了该值,则返回匹配该元素所对应的键名。如果没找到,则返回 false。同样如果第三个参数 strict 被指定为 true,则只有在数据类型和值都一致时才返回相应元素的键名。 源码中两者调用一个函数。
4、isset 和 empty 查找数组

三种方式的语法区别

  • empty: 参数为0或为NULL时(如上面列子),empty均返回TRUE,详细情况可以参见empty官方手册
  • isset: 参数为NULL时,返回FALSE,0与NULL在PHP中是有区别的,isset(0)返回TRUE
  • array_key_exists: 纯粹的判断数组键值对是否存在,无论值是多少

所以,从准确性的角度来看,array_key_exists是最准确的!

可以看到在大数据情况下,empty和isset的性能比array_key_exists快了2个数量级,差别还是很大。如果频繁判断,还是需要优化。产生这么大性能差别的原因,个人猜测,可能是isset和empty作为php语法结构不是函数,php解释器做了优化,而array_key_exists作为函数,没有相关优化。具体原因,有待通过源码考究。

三种方式的使用建议

(鉴于empty与isset性能类似,但是isset准确性较高,这里就只比较isset与array_key_exists)

  • 如果数组不可能出现值为NULL的情况,建议使用isset
  • 如果数组中经常出现值为NULL的情况,建议使用array_key_exists
  • 如果数组中可能出现值为NULL,但是较少的情况,建议结合isset与array_key_exists使用, 
  • 如“if (isset($arr[‘key’]) || array_key_exists(‘key’, $arr)){/*do somthing*/}”。此方法兼顾了性能和准确性,但是代码变长了。

5、in_array优化

1.数组key与value翻转,通过isset判断key是否存在于数组中

public static function inArray($item, $array) {     $flipArray = array_flip($array);     return isset($flipArray[$item]); }
2、用implode连接,直接用strpos判断

用implode函数+逗号连起来,直接用strpos判断。php里面字符串取位置速度非常快,尤其是在大数据量的情况下。不过需要注意的是首尾都要加"," ,这样比较严谨。如: ,user1,user2,user3, 查找的时候,查,user1,。还有strpos要用!== false,因为第一个会返回0。示例如下:

public static function inArray($item, $array) {     $str = implode(",", $array);     $str = "," . $str . ",";     $item = "," . $item . ",";     return false !== strpos($item, $str) ? true : false; }

3、in_array 加第三个参数。
经过实际性能对比,在数据量不大的时候,比如小于1000,查找用哪一种都行,都不会成为性能上的瓶颈。但当数据量比较大的时候,用 array_key_exists 比较合适。据测试 array_key_exist 要比 in_array 效率高十几甚至几十倍。
6、数组循环  foreach 是最快的遍历数组,其次是for 接下里是while

PHP数组添加和移除函数 1、mixed array_pop ( array &$array )       弹出并返回 array 数组的最后一个单元,并将数组 array 的长度减一。如果 array 为空(或者不是数组)将返回 NULL。
<?php    $stack = array("orange", "banana", "apple", "raspberry");    $fruit = array_pop($stack);    print_r($stack);?>

经过此操作后,$stack 将只有 3 个单元:

Array
(
    [0] => orange
    [1] => banana
    [2] => apple
)

并且 rasberry 将被赋给 $fruit。$fruit = "raspberry";

2、int array_push ( array &$array ,  mixed   $var [,  mixed   $... ] ) array_push() 将 array 当成一个栈,并将传入的变量压入 array 的末尾。array 的长度将根据入栈变量的数目增加。 相当于重复 $array[] = $var;
<?php    $stack = array("orange", "banana");    array_push($stack, "apple", "raspberry");    print_r($stack);?>
Note: 如果用 array_push() 来给数组增加一个单元,还不如用 $array[] = ,因为这样没有调用函数的额外负担。
3、 mixed   array_shift ( array &$array ) array_shift() 将 array 的第一个单元移出并作为结果返回,将 array 的长度减一并将所有其它单元向前移动一位。所有的数字键名将改为从零开始计数,文字键名将不变。如果 array 为空(或者不是数组),则返回 NULL 其他和array_pop一样
4、int array_unshift ( array &$array ,  mixed    $var [,  mixed    $... ] ) array_unshift() 将传入的单元插入到 array 数组的开头。注意单元是作为整体被插入的,因此传入单元将保持同样的顺序。所有的数值键名将修改为从零开始重新计数,所有的文字键名保持不变。    返回 array 数组新的单元数目。
<?php    $queue = array("orange", "banana");    array_unshift($queue, "apple", "raspberry");?>Array(    [0] => apple    [1] => raspberry    [2] => orange    [3] => banana)

PHP数组回调方法 将用户自定义函数 funcname 应用到 array 数组中的每个单元。典型情况下 funcname 接受两个参数。array 参数的值作为第一个,键名作为第二个。如果提供了可选参数 userdata,将被作为第三个参数传递给 callback funcname
$fruits = array("d" => "lemon", "a" => "orange", "b" => "banana", "c" => "apple"); function test_alter(&$item1, $key, $prefix){    $item1 = "$prefix: $item1";} function test_print($item2, $key){    echo "$key. $item2<br />
";} echo "Before ...:
";array_walk($fruits, "test_print"); array_walk($fruits, "test_alter", "fruit");echo "... and after:
"; array_walk($fruits, "test_print");

2、array_walk_recursive — 对数组中的每个成员递归地应用用户函数 ,同上一个函数,只是这个函数会递归调用。
<?php$sweet = array("a" => "apple", "b" => "banana");$fruits = array("sweet" => $sweet, "sour" => "lemon"); function test_print($item, $key){    echo "$key holds $item
";} array_walk_recursive($fruits, "test_print");?>以上例程会输出: a holds apple b holds bananasour holds lemon

注意上例中的键 "sweet" 并没有显示出来。任何其值为数组的键都不会被传递到回调函数中去。

3、array array_map ( callback $callback , array $arr1 [, array $... ] ) array_map() 返回一个数组,该数组包含了 arr1 中的所有单元经过 callback 作用过之后的单元。callback 接受的参数数目应该和传递给 array_map() 函数的数组数目一致。
Example #1 array_map() 例子<?phpfunction cube($n){    return($n * $n * $n);} $a = array(1, 2, 3, 4, 5);$b = array_map("cube", $a);print_r($b);?>这使得 $b 成为: Array(    [0] => 1    [1] => 8    [2] => 27     [3] => 64    [4] => 125)

Example #2 array_map() - 使用更多的数组 <?phpfunction show_Spanish($n, $m){    return("The number $n is called $m in Spanish");} function map_Spanish($n, $m){    return(array($n => $m));} $a = array(1, 2, 3, 4, 5);$b = array("uno", "dos", "tres", "cuatro", "cinco"); $c = array_map("show_Spanish", $a, $b);print_r($c); $d = array_map("map_Spanish", $a , $b);print_r($d);?>以上例程会输出:// printout of $cArray(    [0] => The number 1 is called uno in Spanish    [1] => The number 2 is called dos in Spanish    [2] => The number 3 is called tres in Spanish    [3] => The number 4 is called cuatro in Spanish    [4] => The number 5 is called cinco in Spanish)









PHP数组其他方法
1、bool shuffle ( array &$array ) — 将数组打乱


2、array range ( mixed $low , mixed $high [, number $step ]
  )  建立一个包含指定范围单元的数组
range(1,20) 产生一个20个元素的数组
range(1,20,2) 产生一个 20/2 个元素的数组,每个间隔是2


3、array compact ( mixed $varname [, mixed $... ]
 ) 把变量转换为数组。


$name = "caida";$sax = 20;$phone = "139..."; $arr = array("name", "sax", "phone");$newarr = compact($arr);print_r($newarr);print_r(compact("name"));

4、int extract ( array $var_array [, int $extract_type [, string $prefix ]] ) 把数组转换成变了。
$var_array = array("color" => "blue",                   "size"  => "medium",                   "shape" => "sphere"); extract($var_array);echo $size; #medium
5、array array_chunk ( array $input , int $size [, bool $preserve_keys ] ) 将一个数组分割成多个
$input_array = array("a", "b", "c", "d", "e");print_r(array_chunk($input_array, 2));print_r(array_chunk($input_array, 2, true));

6、array array_count_values ( array $input  )  统 计数组中所有的值出现的次数
7、array_flip — 交换数组中的键和值
8、array array_filter ( array $input [, callback $callback ] ) array_filter() 依次将 input 数组中的每个值传递到 callback 函数。如果 callback 函数返回 TRUE,则 input 数组的当前值会被包含在返回的结果数组中。数组的键名保留不变。
<?phpfunction odd($var){    return($var % 2 == 1);} function even($var){    return($var % 2 == 0);} $array1 = array("a"=>1, "b"=>2, "c"=>3, "d"=>4, "e"=>5);$array2 = array(6, 7, 8, 9, 10, 11, 12); echo "Odd :
";print_r(array_filter($array1, "odd"));echo "Even:
";print_r(array_filter($array2, "even"));?>Odd : Array (     [a] => 1    [c] => 3    [e] => 5) Even: Array(    [0] => 6    [2] => 8    [4] => 10     [6] => 12)
9、array array_merge ( array $array1 [, array $array2 [, array $... ]] )  一维数组       array array_merge_recursive ( array $array1 [, array $... ] )          多维数组

array_merge() 将一个或多个数组的单元合并起来,一个数组中的值附加在前一个数组的后面。返回作为结果的数组。如果输入的数组中有相同的字符串键名,则该键名后面的值将覆盖前一个值。然而,如果数组包含数字键名,后面的值将不会覆盖原来的值,而是附加到后面。如果只给了一个数组并且该数组是数字索引的,则键名会以连续方式重新索引。


10、mixed array_rand ( array $input [, int $num_req ] ) 从数组中随机取出一个或多个单元
11、array array_slice ( array $array , int $offset [, int $length [, bool $preserve_keys ]] ) array_slice() 返回根据 offset 和 length 参数所指定的 array 数组中的一段序列。 第三个参数为TRUE是不重建数组键。
$input = array("a", "b", "c", "d", "e"); $output = array_slice($input, 2);      // returns "c", "d", and "e"$output = array_slice($input, -2, 1);  // returns "d"$output = array_slice($input, 0, 3);   // returns "a", "b", and "c" // note the differences in the array keysprint_r(array_slice($input, 2, -1));print_r(array_slice($input, 2, -1, true));

12、array array_unique ( array $array ) 注意键名保留不变。array_unique() 先将值作为字符串排序,然后对每个值只保留第一个遇到的键名,接着忽略所有后面的键名。这并不意味着在未排序的 array 中同一个值的第一个出现的键名会被保留。
PHP数组常用方法 1、一个值,对应其他的值。(删除指定值得数组)
            
声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。