1. PHP_FUNCTION(array_diff)
  2. {
  3. zval *args;
  4. int argc, i;
  5. uint32_t num;
  6. HashTable exclude;
  7. zval *value;
  8. zend_string *str, *key;
  9. zend_long idx;
  10. zval dummy;
  11.  
  12. // 至少两个参数
  13. if (ZEND_NUM_ARGS() < ) {
  14. php_error_docref(NULL, E_WARNING, "at least 2 parameters are required, %d given", ZEND_NUM_ARGS());
  15. return;
  16. }
  17. // 类型描述符:* variable arguments list (0 or more)
  18. // 类型描述符:+ variable arguments list (1 or more)
  19. // 参数存放在数组args中,argc保存参数个数
  20. if (zend_parse_parameters(ZEND_NUM_ARGS(), "+", &args, &argc) == FAILURE) {
  21. return;
  22. }
  23.  
  24. // 第一个参数不是数组,则报错
  25. if (Z_TYPE(args[]) != IS_ARRAY) {
  26. php_error_docref(NULL, E_WARNING, "Argument #1 is not an array");
  27. RETURN_NULL(); // 返回NULL
  28. }
  29.  
  30. /* count number of elements */
  31. // 检查所有参数是否是数组
  32. num = ;
  33. for (i = ; i < argc; i++) {
  34. if (Z_TYPE(args[i]) != IS_ARRAY) {
  35. php_error_docref(NULL, E_WARNING, "Argument #%d is not an array", i + );
  36. RETURN_NULL(); // 返回NULL
  37. }
  38. num += zend_hash_num_elements(Z_ARRVAL(args[i])); // 所有参数数组的元素个数累加
  39. }
  40.  
  41. // 元素个数num为0,返回第一个参数数组(空数组)
  42. if (num == ) {
  43. ZVAL_COPY(return_value, &args[]);
  44. return;
  45. }
  46.  
  47. ZVAL_NULL(&dummy);
  48. /* create exclude map */
  49. zend_hash_init(&exclude, num, NULL, NULL, );
  50. // 从第二个数组开始,所以数组元素保存到exclude
  51. for (i = ; i < argc; i++) {
  52. // 遍历数组
  53. ZEND_HASH_FOREACH_VAL_IND(Z_ARRVAL(args[i]), value) {
  54. str = zval_get_string(value); // 数组元素值转为字符串
  55. zend_hash_add(&exclude, str, &dummy); //
  56. zend_string_release(str);
  57. } ZEND_HASH_FOREACH_END();
  58. }
  59.  
  60. /* copy all elements of first array that are not in exclude set */
  61. // 初始化返回值数组,大小和第一个数组一致。
  62. array_init_size(return_value, zend_hash_num_elements(Z_ARRVAL(args[])));
  63. // 循环遍历第一个数组
  64. ZEND_HASH_FOREACH_KEY_VAL_IND(Z_ARRVAL(args[]), idx, key, value) {
  65. str = zval_get_string(value); // 元素值转为字符串
  66. if (!zend_hash_exists(&exclude, str)) { // exclude中找不到该值
  67. // 插入到返回值数组中(差集),键名保留不变。
  68. if (key) {
  69. value = zend_hash_add_new(Z_ARRVAL_P(return_value), key, value);
  70. } else {
  71. value = zend_hash_index_add_new(Z_ARRVAL_P(return_value), idx, value);
  72. }
  73. zval_add_ref(value);
  74. }
  75. zend_string_release(str);
  76. } ZEND_HASH_FOREACH_END();
  77.  
  78. zend_hash_destroy(&exclude);
  79. }

php内置函数分析array_diff()的更多相关文章

  1. map内置函数分析所得到的思路

    map:会根据提供的函数对指定序列做映射. map(func, *iterables) --> map object Make an iterator that computes the fun ...

  2. php内置函数分析之array_diff_assoc()

    static void php_array_diff_key(INTERNAL_FUNCTION_PARAMETERS, int data_compare_type) /* {{{ */ { uint ...

  3. php内置函数分析之array_combine()

    PHP_FUNCTION(array_combine) { HashTable *values, *keys; uint32_t pos_values = ; zval *entry_keys, *e ...

  4. php内置函数分析之ucwords()

    PHP_FUNCTION(ucwords) { zend_string *str; char *delims = " \t\r\n\f\v"; register char *r, ...

  5. php内置函数分析之strtoupper()、strtolower()

    strtoupper(): PHP_FUNCTION(strtoupper) { zend_string *str; ZEND_PARSE_PARAMETERS_START(, ) Z_PARAM_S ...

  6. php内置函数分析之ucfirst()、lcfirst()

    ucfirst($str) 将 str 的首字符(如果首字符是字母)转换为大写字母,并返回这个字符串. 源码位于 ext/standard/string.c /* {{{ php_ucfirst Up ...

  7. php内置函数分析之trim()

    官方手册中: 类似函数还有两个:ltrim() 和 rtrim().分别处理字符串的左侧.右侧. trim()的具体实现位于:ext/standard/string.c /* {{{ proto st ...

  8. php内置函数分析之str_pad()

    PHP_FUNCTION(str_pad) { /* Input arguments */ zend_string *input; /* Input string 输入字符串*/ zend_long ...

  9. php内置函数分析之array_fill_keys()

    PHP_FUNCTION(array_fill_keys) { zval *keys, *val, *entry; if (zend_parse_parameters(ZEND_NUM_ARGS(), ...

随机推荐

  1. What's the difference between HEAD^ and HEAD~ in Git?

    https://stackoverflow.com/questions/2221658/whats-the-difference-between-head-and-head-in-git Rules ...

  2. SecureCRT使用+堡垒机简单使用

    写在前面的话 自从升级为宝妈后,回来发现好多东西都遗忘了.熟话说:好记性不如烂笔头,我还是记录下来吧. 堡垒机使用的几个技巧 1.快捷操作         1) 输入 ID 直接登录.         ...

  3. Spring 缓存切面

    缓存切面:[通知+目标方法调用] 缓存操作执行过程: 1)如果是同步调用[sync=true],则首先尝试从缓存中读取数据,读取到则直接返回: 否则执行目标方法,将结果缓存后返回. 2)如果不是同步调 ...

  4. WPF DevExpress Chart控件 需要引用的类库

    DevExpress.Charts.v16.1.Core.dll DevExpress.Data.v16.1.dll DevExpress.Mvvm.v16.1.dll DevExpress.Xpf. ...

  5. 【漏洞学习】HOST 头攻击漏洞

    日期:2018-03-06 14:32:51 作者:Bay0net 0x01. 前言 在一般情况下,几个网站可能会放在同一个服务器上,或者几个 web 系统共享一个服务器,host 头来指定应该由哪个 ...

  6. vue启动流程

    继上一篇vue环境的搭建(在D盘新建文件夹vue_cli,把(我已经上传到了文件下)资料下tpls解压完后的所有文件都复制到D盘vue_cli下) 目录如图: 1.webstorm设置为了提高webS ...

  7. Java8---函数式编程-示例

    // Java8函数式编程示例—(Predicate.Stream.Optional) https://blog.csdn.net/weixin_41950473/article/details/84 ...

  8. 同一台电脑管理多个SSH KEY

    同一台电脑关于多个SSH KEY管理 笔者之前为电脑中的homestead虚拟机配置过id_rsa,但现在因为想在github上搭建基于hexo的博客,所以需要配置github的ssh key,因此产 ...

  9. Nginx 配置二级虚拟目录访问 Laravel 重写

    server { listen 80; server_name _; root /opt/sites; index index.php index.html index.htm; etag on; g ...

  10. python文件打包/导入 .so 文件

    打包so文件 见: http://www.cnblogs.com/ke10/p/py2so.html 导入so文件 import sys sys.path.append(r'/home/project ...