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

PHP的json_encode使用分析说明

json的优点就不说了,有个习惯,我在输出json的时候,喜欢用sprintf拼成json格式,前两天被朋友说不标准,必须要用json_encode生成的才是标准的json格式,我当然很郁闷...
json的优点就不说了,有个习惯,我在输出json的时候,喜欢用 sprintf 拼成json格式,前两天被朋友说不标准,必须要用json_encode生成的才是标准的json格式,我当然很郁闷啦,用了这么多年了,刚知道这样做不标准,既然说我不标准,那什么才是标准的json格式?代码如下:

{a : 'abc'} {'a' : 'abc'} {a : "abc"} {"a" : "abc"}

谁都知道,只有第四种才是标准的json格式,我这么做,代码如下:

$ret_json='{"%s":"%s"}';echo json_encode($ret_json,"a","abc");

必然也符合标准,既然如此,那我就要刨根问底,json_encode生成的json格式究竟有什么不同?

实例代码如下:

static PHP_FUNCTION(json_encode) 
{ 
    zval *parameter; 
    smart_str buf = {0}; 
    long optiOns= 0; 
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z|l", ¶meter, &options) == FAILURE) { 
            return;//开源代码phprm.com 
    }   
    JSON_G(error_code) = PHP_JSON_ERROR_NONE; 
    php_json_encode(&buf, parameter, options TSRMLS_CC); 
    ZVAL_STRINGL(return_value, buf.c, buf.len, 1); 
    smart_str_free(&buf); 
} 
JSON_G(error_code) = PHP_JSON_ERROR_NONE;

是定义的json错误,该错误可以通过json_last_error函数获取,你用过吗?反正我没用过.

php_json_encode是主要的操作,代码如下:

PHP_JSON_API void php_json_encode(smart_str *buf, zval *val, int options TSRMLS_DC) /* {{{ */ 
{ 
    switch (Z_TYPE_P(val)) 
    { 
            case IS_NULL: 
                    smart_str_appendl(buf, "null", 4); //输出NULL 
                    break; 
            case IS_BOOL: 
                    if (Z_BVAL_P(val)) { 
                            smart_str_appendl(buf, "true", 4);//输出true 
                    } else { 
                            smart_str_appendl(buf, "false", 5);//输出false 
                    } 
                    break; 
            case IS_LONG: 
                    smart_str_append_long(buf, Z_LVAL_P(val));//输出长整形的值 
                    break; 
            case IS_DOUBLE: 
                    { 
                            char *d = NULL; 
                            int len; 
                            double dbl = Z_DVAL_P(val); 
                            if (!zend_isinf(dbl) && !zend_isnan(dbl)) {//非无穷尽 
                                    len = spprintf(&d, 0, "%.*k", (int) EG(precision), dbl); 
                                    smart_str_appendl(buf, d, len); 
                                    efree(d); 
                            } else { 
                                    php_error_docref(NULL TSRMLS_CC, E_WARNING, "double %.9g does not conform to the JSON spec, encoded as 0", dbl); 
                                    smart_str_appendc(buf, '0'); 
                            } 
                   } 
                    break; 
            case IS_STRING://字符串 
                    json_escape_string(buf, Z_STRVAL_P(val), Z_STRLEN_P(val), options TSRMLS_CC); 
                    break; 
            case IS_ARRAY://数组和对象 
            case IS_OBJECT: 
                    json_encode_array(buf, &val, options TSRMLS_CC); 
                    break; 
            default: 
                    php_error_docref(NULL TSRMLS_CC, E_WARNING, "type is unsupported, encoded as null"); 
                    smart_str_appendl(buf, "null", 4); 
                    break; 
    } 
    return; 
} 
很明显,根据不同的类型,会有相应的case,最复杂的是字符串 、数组 、对象这三种类型,数组和对象是同一种操作,先看看字符串吧,很长,注释直接写在代码里,代码如下:
/options应该是5.3版本之后才支持的,由以下常量组成的二进制掩码:JSON_HEX_QUOT, JSON_HEX_TAG, JSON_HEX_AMP, JSON_HEX_APOS, JSON_NUMERIC_CHECK, JSON_PRETTY_PRINT, JSON_UNESCAPED_SLASHES, JSON_FORCE_OBJECT, JSON_UNESCAPED_UNICODE.虽然我没用过。。。 
static void json_escape_string(smart_str *buf, char *s, int len, int options TSRMLS_DC) /* {{{ */ 
{ 
    int pos = 0; 
    unsigned short us; 
    unsigned short *utf16; 
    if (len == 0) {//如果长度为0,则直接返回 双引号 "" 
            smart_str_appendl(buf, """", 2); 
            return; 
    } 
    if (options & PHP_JSON_NUMERIC_CHECK) {//检测是否为0-9的数字,如果是数字,那么就会直接把数据作为long或double类型返回。 
            double d; 
            int type; 
            long p; 
            if ((type = is_numeric_string(s, len, &p, &d, 0)) != 0) { 
                    if (type == IS_LONG) { 
                            smart_str_append_long(buf, p); 
                    } else if (type == IS_DOUBLE) { 
                            if (!zend_isinf(d) && !zend_isnan(d)) { 
                                    char *tmp; 
                                    int l = spprintf(&tmp, 0, "%.*k", (int) EG(precision), d); 
                                    smart_str_appendl(buf, tmp, l); 
                                    efree(tmp); 
                            } else { 
                                    php_error_docref(NULL TSRMLS_CC, E_WARNING, "double %.9g does not conform to the JSON spec, encoded as 0", d); 
                                    smart_str_appendc(buf, '0'); 
                            } 
                    } 
                    return; 
            } 
    } 
    utf16 = (unsigned short *) safe_emalloc(len, sizeof(unsigned short), 0); 
    len = utf8_to_utf16(utf16, s, len); //这里会对你输入的值一次处理转成对应的Dec码,比如1是49,a是97这样的,保存到utf16中。 
if (len <= 0) {//如果len小于0 说明出错。如果用json_encode处理GBK的编码,就会在这里挂掉。 
    if (utf16) { 
            efree(utf16); 
    } 
    if (len <0) { 
            JSON_G(error_code) = PHP_JSON_ERROR_UTF8; 
            if (!PG(display_errors)) { 
                    php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid UTF-8 sequence in argument"); 
            } 
            smart_str_appendl(buf, "null", 4); 
    } else { 
            smart_str_appendl(buf, """", 2); 
    } 
    return; 
} 
smart_str_appendc(buf, &#39;"&#39;); //输入 " 
//下面这一段代码就是将一些特殊字符转义如 双引号,反斜线等等 
while (pos &#39;: 
                    if (options & PHP_JSON_HEX_TAG) { 
                            smart_str_appendl(buf, "u003E", 6); 
                    } else { 
                            smart_str_appendc(buf, &#39;>&#39;); 
} 
                    break; 
            case &#39;&&#39;: 
                    if (options & PHP_JSON_HEX_AMP) { 
                            smart_str_appendl(buf, "u0026", 6); 
                    } else { 
                            smart_str_appendc(buf, &#39;&&#39;); 
                    } 
                    break; 
            case &#39;&#39;&#39;: 
                    if (options & PHP_JSON_HEX_APOS) { 
                            smart_str_appendl(buf, "u0027", 6); 
                    } else { 
                            smart_str_appendc(buf, &#39;&#39;&#39;); 
                    } 
                    break; 
            default: //一直到这里,没有特殊字符就会把值append到buf中 
                    if (us >= &#39; &#39; && (us & 127) == us) { 
                            smart_str_appendc(buf, (unsigned char) us); 
                    } else { 
                            smart_str_appendl(buf, "u", 2); 
                            us = REVERSE16(us); 
                            smart_str_appendc(buf, digits[us & ((1 <<4) - 1)]); 
                            us >>= 4; 
                            smart_str_appendc(buf, digits[us & ((1 <<4) - 1)]); 
                            us >>= 4; 
                            smart_str_appendc(buf, digits[us & ((1 <<4) - 1)]); 
                            us >>= 4; 
                            smart_str_appendc(buf, digits[us & ((1 <<4) - 1)]); 
                    } 
                    break; 
    } 
} 
smart_str_appendc(buf, &#39;"&#39;); //结束 双引号。 
efree(utf16); 
}  
再来看看数组和对象,也很简单,代码如下:
static void json_encode_array(smart_str *buf, zval **val, int options TSRMLS_DC) /* {{{ */ 
{ 
int i, r; 
HashTable *myht; 
if (Z_TYPE_PP(val) == IS_ARRAY) { 
    myht = HASH_OF(*val); 
    r = (options & PHP_JSON_FORCE_OBJECT) ? PHP_JSON_OUTPUT_OBJECT : json_determine_array_type(val TSRMLS_CC); 
} else { 
    myht = Z_OBJPROP_PP(val); 
    r = PHP_JSON_OUTPUT_OBJECT; 
}   
if (myht && myht->nApplyCount > 1) { 
    php_error_docref(NULL TSRMLS_CC, E_WARNING, "recursion detected"); 
    smart_str_appendl(buf, "null", 4); 
    return; 
} 
//开始标签 
if (r == PHP_JSON_OUTPUT_ARRAY) { 
    smart_str_appendc(buf, &#39;[&#39;); 
} else { 
    smart_str_appendc(buf, &#39;{&#39;); 
}   
i = myht ? zend_hash_num_elements(myht) : 0; 
if (i > 0) 
{ 
    char *key; 
    zval **data; 
    ulong index; 
    uint key_len; 
    HashPosition pos; 
    HashTable *tmp_ht; 
    int need_comma = 0; 
    zend_hash_internal_pointer_reset_ex(myht, &pos); 
//便利哈希表 
    for (;; zend_hash_move_forward_ex(myht, &pos)) { 
        i = zend_hash_get_current_key_ex(myht, &key, &key_len, &index, 0, &pos); 
        if (i == HASH_KEY_NON_EXISTANT) 
                break; 
        if (zend_hash_get_current_data_ex(myht, (void **) &data, &pos) == SUCCESS) { 
                tmp_ht = HASH_OF(*data); 
                if (tmp_ht) { 
                        tmp_ht->nApplyCount++; 
                } 
                if (r == PHP_JSON_OUTPUT_ARRAY) { 
                        if (need_comma) { 
                                smart_str_appendc(buf, &#39;,&#39;); 
                        } else { 
                                need_comma = 1; 
                        } 
    //将值append到 buf中 
                        php_json_encode(buf, *data, options TSRMLS_CC); 
                } else if (r == PHP_JSON_OUTPUT_OBJECT) { 
                        if (i == HASH_KEY_IS_STRING) { 
                                if (key[0] == &#39;&#39; && Z_TYPE_PP(val) == IS_OBJECT) { 
                                        /* Skip protected and private members. */ 
                                        if (tmp_ht) { 
                                                tmp_ht->nApplyCount--; 
                                        } 
                                        continue; 
                                } 
                                if (need_comma) { 
                                        smart_str_appendc(buf, &#39;,&#39;); 
                                } else { 
                                        need_comma = 1; 
                                } 
                                json_escape_string(buf, key, key_len - 1, options & ~PHP_JSON_NUMERIC_CHECK TSRMLS_CC); 
                                smart_str_appendc(buf, &#39;:&#39;); 
                                php_json_encode(buf, *data, options TSRMLS_CC); 
                        } else { 
                                if (need_comma) { 
                                        smart_str_appendc(buf, &#39;,&#39;); 
                                } else { 
                                        need_comma = 1; 
                                } 
                                smart_str_appendc(buf, &#39;"&#39;); 
                                smart_str_append_long(buf, (long) index); 
                                smart_str_appendc(buf, &#39;"&#39;); 
                                smart_str_appendc(buf, &#39;:&#39;); 
                                php_json_encode(buf, *data, options TSRMLS_CC); 
                        } 
                } 
                if (tmp_ht) { 
                        tmp_ht->nApplyCount--; 
                } 
        } 
    } 
} 
   //结束标签 
    if (r == PHP_JSON_OUTPUT_ARRAY) { 
            smart_str_appendc(buf, &#39;]&#39;); 
    } else { 
            smart_str_appendc(buf, &#39;}&#39;); 
    } 
}

通过简单分析,证明了一个问题,跟我上面用sprintf的方法其实是一样的,都是拼接字符串,而且 为了性能,更应该鼓励用sprintf来拼接json格式,因为 json_encode会进行很多循环操作,而且所消耗的性能是线性的 O(n^2).


本文地址:

转载随意,但请附上文章地址:-)

推荐阅读
  • YOLOv7基于自己的数据集从零构建模型完整训练、推理计算超详细教程
    本文介绍了关于人工智能、神经网络和深度学习的知识点,并提供了YOLOv7基于自己的数据集从零构建模型完整训练、推理计算的详细教程。文章还提到了郑州最低生活保障的话题。对于从事目标检测任务的人来说,YOLO是一个熟悉的模型。文章还提到了yolov4和yolov6的相关内容,以及选择模型的优化思路。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 使用在线工具jsonschema2pojo根据json生成java对象
    本文介绍了使用在线工具jsonschema2pojo根据json生成java对象的方法。通过该工具,用户只需将json字符串复制到输入框中,即可自动将其转换成java对象。该工具还能解析列表式的json数据,并将嵌套在内层的对象也解析出来。本文以请求github的api为例,展示了使用该工具的步骤和效果。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • 本文介绍了前端人员必须知道的三个问题,即前端都做哪些事、前端都需要哪些技术,以及前端的发展阶段。初级阶段包括HTML、CSS、JavaScript和jQuery的基础知识。进阶阶段涵盖了面向对象编程、响应式设计、Ajax、HTML5等新兴技术。高级阶段包括架构基础、模块化开发、预编译和前沿规范等内容。此外,还介绍了一些后端服务,如Node.js。 ... [详细]
  • 如何查询zone下的表的信息
    本文介绍了如何通过TcaplusDB知识库查询zone下的表的信息。包括请求地址、GET请求参数说明、返回参数说明等内容。通过curl方法发起请求,并提供了请求示例。 ... [详细]
  • 在Android中解析Gson解析json数据是很方便快捷的,可以直接将json数据解析成java对象或者集合。使用Gson解析json成对象时,默认将json里对应字段的值解析到java对象里对应字段的属性里面。然而,当我们自己定义的java对象里的属性名与json里的字段名不一样时,我们可以使用@SerializedName注解来将对象里的属性跟json里字段对应值匹配起来。本文介绍了使用@SerializedName注解解析json数据的方法,并给出了具体的使用示例。 ... [详细]
  • 电话号码的字母组合解题思路和代码示例
    本文介绍了力扣题目《电话号码的字母组合》的解题思路和代码示例。通过使用哈希表和递归求解的方法,可以将给定的电话号码转换为对应的字母组合。详细的解题思路和代码示例可以帮助读者更好地理解和实现该题目。 ... [详细]
  • 在Docker中,将主机目录挂载到容器中作为volume使用时,常常会遇到文件权限问题。这是因为容器内外的UID不同所导致的。本文介绍了解决这个问题的方法,包括使用gosu和suexec工具以及在Dockerfile中配置volume的权限。通过这些方法,可以避免在使用Docker时出现无写权限的情况。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • faceu激萌变老特效的使用方法详解
    本文介绍了faceu激萌变老特效的使用方法,包括打开faceu激萌app、点击贴纸、选择热门贴纸中的变老特效,然后对准人脸进行拍摄,即可给照片添加变老特效。操作简单,适合新用户使用。 ... [详细]
  • 大连微软技术社区举办《.net core始于足下》活动,获得微软赛百味和易迪斯的赞助
    九月十五日,大连微软技术社区举办了《.net core始于足下》活动,共有51人报名参加,实际到场人数为43人,还有一位专程从北京赶来的同学。活动得到了微软赛百味和易迪斯的赞助,场地也由易迪斯提供。活动中大家积极交流,取得了非常成功的效果。 ... [详细]
  • 给定一个二叉树,要求随机选择树上的一个节点。解法:遍历树的过程中,随机选择一个节点即可。具体做法参看:从输入 ... [详细]
  • 本文介绍了在微店中如何修改分销产品的价格以及设置价格的方法。客户在拍下商品后,在1小时内可以进行修改价格的操作,通过进入订单管理,点击未付款子项,可以找到订单信息并进行改价操作。修改价格后,买家会收到改价后的短信通知,在微店订单中进行付款即可。 ... [详细]
author-avatar
小苗的秘密
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有