当前位置: 首页 > news >正文

php array_diff 比较两个数组bug避坑 深入了解

今天实用array_diff出现的异常问题,预想的结果应该是返回 "integral_initiate"=>"0",实际没有

先看测试代码:

$a = ["user_name"=>"测","see_num"=>0,"integral_initiate"=>"0"
];
$b = ["user_name"=>"测","see_num"=>0,"integral_initiate"=>10
];
$gf = array_diff($a,$b);
print_r($gf);

没有返回差异,纠结了好一阵子又查阅了文档看到这一句话才醒悟 

我们简化一下数组来看,通过简化数组发现只要两个数组中间都带有0的值就不会正常效验

解决方案就是换成 array_diff_assoc 对比键名与键值

------------序言--------------

虽然上面得到了想要的结果,但是本着刨根问底的思想继续研究了 一番,查了一些资料,也翻了下源码终于找到了一个合理的解释

/* {{{ proto array array_diff(array arr1, array arr2 [, array ...])Returns the entries of arr1 that have values which are not present in any of the others arguments. */
PHP_FUNCTION(array_diff)
{zval *args;int argc, i;uint32_t num;HashTable exclude;zval *value;zend_string *str, *tmp_str, *key;zend_long idx;zval dummy;if (ZEND_NUM_ARGS() < 2) {php_error_docref(NULL, E_WARNING, "at least 2 parameters are required, %d given", ZEND_NUM_ARGS());return;}ZEND_PARSE_PARAMETERS_START(1, -1)Z_PARAM_VARIADIC('+', args, argc)ZEND_PARSE_PARAMETERS_END();if (Z_TYPE(args[0]) != IS_ARRAY) {php_error_docref(NULL, E_WARNING, "Expected parameter 1 to be an array, %s given", zend_zval_type_name(&args[0]));RETURN_NULL();}num = zend_hash_num_elements(Z_ARRVAL(args[0]));if (num == 0) {for (i = 1; i < argc; i++) {if (Z_TYPE(args[i]) != IS_ARRAY) {php_error_docref(NULL, E_WARNING, "Expected parameter %d to be an array, %s given", i + 1, zend_zval_type_name(&args[i]));RETURN_NULL();}}RETURN_EMPTY_ARRAY();} else if (num == 1) {int found = 0;zend_string *search_str, *tmp_search_str;value = NULL;ZEND_HASH_FOREACH_VAL_IND(Z_ARRVAL(args[0]), value) {break;} ZEND_HASH_FOREACH_END();if (!value) {for (i = 1; i < argc; i++) {if (Z_TYPE(args[i]) != IS_ARRAY) {php_error_docref(NULL, E_WARNING, "Expected parameter %d to be an array, %s given", i + 1, zend_zval_type_name(&args[i]));RETURN_NULL();}}RETURN_EMPTY_ARRAY();}search_str = zval_get_tmp_string(value, &tmp_search_str);for (i = 1; i < argc; i++) {if (Z_TYPE(args[i]) != IS_ARRAY) {php_error_docref(NULL, E_WARNING, "Expected parameter %d to be an array, %s given", i + 1, zend_zval_type_name(&args[i]));RETURN_NULL();}if (!found) {ZEND_HASH_FOREACH_VAL_IND(Z_ARRVAL(args[i]), value) {str = zval_get_tmp_string(value, &tmp_str);if (zend_string_equals(search_str, str)) {zend_tmp_string_release(tmp_str);found = 1;break;}zend_tmp_string_release(tmp_str);} ZEND_HASH_FOREACH_END();}}zend_tmp_string_release(tmp_search_str);if (found) {RETVAL_EMPTY_ARRAY();} else {ZVAL_COPY(return_value, &args[0]);}return;}/* count number of elements */num = 0;for (i = 1; i < argc; i++) {if (Z_TYPE(args[i]) != IS_ARRAY) {php_error_docref(NULL, E_WARNING, "Expected parameter %d to be an array, %s given", i + 1, zend_zval_type_name(&args[i]));RETURN_NULL();}num += zend_hash_num_elements(Z_ARRVAL(args[i]));}if (num == 0) {ZVAL_COPY(return_value, &args[0]);return;}ZVAL_NULL(&dummy);/* create exclude map */zend_hash_init(&exclude, num, NULL, NULL, 0);for (i = 1; i < argc; i++) {ZEND_HASH_FOREACH_VAL_IND(Z_ARRVAL(args[i]), value) {str = zval_get_tmp_string(value, &tmp_str);zend_hash_add(&exclude, str, &dummy);zend_tmp_string_release(tmp_str);} ZEND_HASH_FOREACH_END();}/* copy all elements of first array that are not in exclude set */array_init_size(return_value, zend_hash_num_elements(Z_ARRVAL(args[0])));ZEND_HASH_FOREACH_KEY_VAL_IND(Z_ARRVAL(args[0]), idx, key, value) {str = zval_get_tmp_string(value, &tmp_str);if (!zend_hash_exists(&exclude, str)) {if (key) {value = zend_hash_add_new(Z_ARRVAL_P(return_value), key, value);} else {value = zend_hash_index_add_new(Z_ARRVAL_P(return_value), idx, value);}zval_add_ref(value);}zend_tmp_string_release(tmp_str);} ZEND_HASH_FOREACH_END();zend_hash_destroy(&exclude);
}
/* }}} */

这就可以解释为什么使用array_diff 不能得到想要的结果,因为只要第一个数组中的值在第二个数组中出现过就不算差异,所以并不是bug,反过来再去看一下文档上面英文注释是这样写的

我只能说这个中文版的翻译有待提升,至此问题就到这里了

相关文档:

1、问题解答: https://stackoverflow.com/questions/4742405/array-diff-to-compare-two-associative-arrays/4742438#4742438 2、PHP 如何查看php函数源码-CSDN博客

3、源码地址 :GitCode - 开发者的代码家园 

相关文章:

  • 【书生·浦语大模型实战营06】《OpenCompass 大模型评测》学习笔记
  • java并发面试题
  • openssl3.2/test/certs - 006 - trust variants: +anyEKU -anyEKU
  • C语言练习day8
  • 【Linux】:线程池(逐行解析代码)
  • 配置redis挂载
  • 使用docker以容器方式安装redis
  • 【论文+视频控制】23.08DragNUWA1.5:通过集成文本、图像和轨迹来进行视频生成中的细粒度控制 (24.01.08开源最新模型)
  • linux下vsc的自动切换输入法解决方案
  • 【数据库】第三章 MySQL库表操作
  • 【AI接口】语音版、文心一言大模型和AI绘图、图片检测API
  • php基础学习之变量
  • Python项目——计算器(PySide6+Pyinstaller)
  • 盖子的c++小课堂:第二十六讲:双向链表
  • JavaEE-微服务-Vuex
  • php的引用
  • [rust! #004] [译] Rust 的内置 Traits, 使用场景, 方式, 和原因
  • emacs初体验
  • Git 使用集
  • iOS 颜色设置看我就够了
  • maya建模与骨骼动画快速实现人工鱼
  • puppeteer stop redirect 的正确姿势及 net::ERR_FAILED 的解决
  • PyCharm搭建GO开发环境(GO语言学习第1课)
  • vue2.0开发聊天程序(四) 完整体验一次Vue开发(下)
  • 阿里研究院入选中国企业智库系统影响力榜
  • 构造函数(constructor)与原型链(prototype)关系
  • 开发了一款写作软件(OSX,Windows),附带Electron开发指南
  • 码农张的Bug人生 - 见面之礼
  • 手机端车牌号码键盘的vue组件
  • 双管齐下,VMware的容器新战略
  • 听说你叫Java(二)–Servlet请求
  • 通过获取异步加载JS文件进度实现一个canvas环形loading图
  • 用jQuery怎么做到前后端分离
  • 支付宝花15年解决的这个问题,顶得上做出十个支付宝 ...
  • ​力扣解法汇总1802. 有界数组中指定下标处的最大值
  • # 数据结构
  • $.ajax中的eval及dataType
  • (02)Hive SQL编译成MapReduce任务的过程
  • (附源码)spring boot基于Java的电影院售票与管理系统毕业设计 011449
  • (附源码)springboot美食分享系统 毕业设计 612231
  • (学习日记)2024.01.09
  • (转)EXC_BREAKPOINT僵尸错误
  • (转)Oracle存储过程编写经验和优化措施
  • .gitattributes 文件
  • .NET Core 控制台程序读 appsettings.json 、注依赖、配日志、设 IOptions
  • .Net Core/.Net6/.Net8 ,启动配置/Program.cs 配置
  • .NET(C#) Internals: as a developer, .net framework in my eyes
  • .net流程开发平台的一些难点(1)
  • .net实现头像缩放截取功能 -----转载自accp教程网
  • :中兴通讯为何成功
  • @ 代码随想录算法训练营第8周(C语言)|Day57(动态规划)
  • @Mapper作用
  • [C#]手把手教你打造Socket的TCP通讯连接(一)
  • [C++]打开新世界的大门之C++入门
  • [codevs 1515]跳 【解题报告】