0

0

PHP的json_encode使用分析说明

php中文网

php中文网

发布时间:2016-05-25 16:40:11

|

1754人浏览过

|

来源于php中文网

原创

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格式究竟有什么不同?

实例代码如下:

立即学习PHP免费学习笔记(深入)”;

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函数获取,你用过吗?反正我没用过.

佳可商务购物程序 2004
佳可商务购物程序 2004

在原版的基础上做了一下修正评论没有提交正文的问题特价商品的调用连接问题去掉了一个后门补了SQL注入补了一个过滤漏洞浮动价不能删除的问题不能够搜索问题收藏时放入购物车时出错点放入购物车弹出2个窗口修正主题添加问题商家注册页导航连接问题销售排行不能显示更多问题热点商品不能显示更多问题增加了服务器探测 增加了空间使用查看 增加了在线文件编辑增加了后台管理里两处全选功能更新说明:后台的部分功能已经改过前台

下载

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, '"'); //输入 " 
//下面这一段代码就是将一些特殊字符转义如 双引号,反斜线等等 
while (pos < len) 
{ 
    us = utf16[pos++]; 
    switch (us) 
    { 
            case '"': 
                    if (options & PHP_JSON_HEX_QUOT) { 
                            smart_str_appendl(buf, "u0022", 6); 
                    } else { 
                            smart_str_appendl(buf, """, 2); 
                    } 
                    break; 
            case '': 
                    smart_str_appendl(buf, "\", 2); 
                    break; 
case '/': 
                    smart_str_appendl(buf, "/", 2); 
                    break; 
            case 'b': 
                    smart_str_appendl(buf, "b", 2); 
                    break; 
            case 'f': 
                    smart_str_appendl(buf, "f", 2); 
                    break; 
            case 'n': 
                    smart_str_appendl(buf, "n", 2); 
                    break; 
            case 'r': 
                    smart_str_appendl(buf, "r", 2); 
                    break; 
            case 't': 
                    smart_str_appendl(buf, "t", 2); 
                    break; 
            case '<': 
                    if (options & PHP_JSON_HEX_TAG) { 
                            smart_str_appendl(buf, "u003C", 6); 
                    } else { 
                            smart_str_appendc(buf, '<'); 
                    } 
                    break; 
            case '>': 
                    if (options & PHP_JSON_HEX_TAG) { 
                            smart_str_appendl(buf, "u003E", 6); 
                    } else { 
                            smart_str_appendc(buf, '>'); 
} 
                    break; 
            case '&': 
                    if (options & PHP_JSON_HEX_AMP) { 
                            smart_str_appendl(buf, "u0026", 6); 
                    } else { 
                            smart_str_appendc(buf, '&'); 
                    } 
                    break; 
            case ''': 
                    if (options & PHP_JSON_HEX_APOS) { 
                            smart_str_appendl(buf, "u0027", 6); 
                    } else { 
                            smart_str_appendc(buf, '''); 
                    } 
                    break; 
            default: //一直到这里,没有特殊字符就会把值append到buf中 
                    if (us >= ' ' && (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, '"'); //结束 双引号。 
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, '['); 
} else { 
    smart_str_appendc(buf, '{'); 
}   
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, ','); 
                        } 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] == '' && 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, ','); 
                                } else { 
                                        need_comma = 1; 
                                } 
                                json_escape_string(buf, key, key_len - 1, options & ~PHP_JSON_NUMERIC_CHECK TSRMLS_CC); 
                                smart_str_appendc(buf, ':'); 
                                php_json_encode(buf, *data, options TSRMLS_CC); 
                        } else { 
                                if (need_comma) { 
                                        smart_str_appendc(buf, ','); 
                                } else { 
                                        need_comma = 1; 
                                } 
                                smart_str_appendc(buf, '"'); 
                                smart_str_append_long(buf, (long) index); 
                                smart_str_appendc(buf, '"'); 
                                smart_str_appendc(buf, ':'); 
                                php_json_encode(buf, *data, options TSRMLS_CC); 
                        } 
                } 
                if (tmp_ht) { 
                        tmp_ht->nApplyCount--; 
                } 
        } 
    } 
} 
   //结束标签 
    if (r == PHP_JSON_OUTPUT_ARRAY) { 
            smart_str_appendc(buf, ']'); 
    } else { 
            smart_str_appendc(buf, '}'); 
    } 
}

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


本文地址:

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

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python 序列化
Python 序列化

本专题整合了python序列化、反序列化相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.02.02

AO3官网入口与中文阅读设置 AO3网页版使用与访问
AO3官网入口与中文阅读设置 AO3网页版使用与访问

本专题围绕 Archive of Our Own(AO3)官网入口展开,系统整理 AO3 最新可用官网地址、网页版访问方式、正确打开链接的方法,并详细讲解 AO3 中文界面设置、阅读语言切换及基础使用流程,帮助用户稳定访问 AO3 官网,高效完成中文阅读与作品浏览。

91

2026.02.02

主流快递单号查询入口 实时物流进度一站式追踪专题
主流快递单号查询入口 实时物流进度一站式追踪专题

本专题聚合极兔快递、京东快递、中通快递、圆通快递、韵达快递等主流物流平台的单号查询与运单追踪内容,重点解决单号查询、手机号查物流、官网入口直达、包裹进度实时追踪等高频问题,帮助用户快速获取最新物流状态,提升查件效率与使用体验。

27

2026.02.02

Golang WebAssembly(WASM)开发入门
Golang WebAssembly(WASM)开发入门

本专题系统讲解 Golang 在 WebAssembly(WASM)开发中的实践方法,涵盖 WASM 基础原理、Go 编译到 WASM 的流程、与 JavaScript 的交互方式、性能与体积优化,以及典型应用场景(如前端计算、跨平台模块)。帮助开发者掌握 Go 在新一代 Web 技术栈中的应用能力。

11

2026.02.02

PHP Swoole 高性能服务开发
PHP Swoole 高性能服务开发

本专题聚焦 PHP Swoole 扩展在高性能服务端开发中的应用,系统讲解协程模型、异步IO、TCP/HTTP/WebSocket服务器、进程与任务管理、常驻内存架构设计。通过实战案例,帮助开发者掌握 使用 PHP 构建高并发、低延迟服务端应用的工程化能力。

5

2026.02.02

Java JNI 与本地代码交互实战
Java JNI 与本地代码交互实战

本专题系统讲解 Java 通过 JNI 调用 C/C++ 本地代码的核心机制,涵盖 JNI 基本原理、数据类型映射、内存管理、异常处理、性能优化策略以及典型应用场景(如高性能计算、底层库封装)。通过实战示例,帮助开发者掌握 Java 与本地代码混合开发的完整流程。

5

2026.02.02

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

62

2026.01.31

go语言 math包
go语言 math包

本专题整合了go语言math包相关内容,阅读专题下面的文章了解更多详细内容。

55

2026.01.31

go语言输入函数
go语言输入函数

本专题整合了go语言输入相关教程内容,阅读专题下面的文章了解更多详细内容。

27

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
R 教程
R 教程

共45课时 | 6万人学习

Pandas 教程
Pandas 教程

共15课时 | 1万人学习

PHP课程
PHP课程

共137课时 | 10.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号