服务器响应 json 字符串的两种格式,有点蒙圈 - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
LinJunzhu
V2EX    问与答

服务器响应 json 字符串的两种格式,有点蒙圈

  •  
  •   LinJunzhu 2016-05-10 17:53:19 +08:00 3615 次点击
    这是一个创建于 3524 天前的主题,其中的信息可能已经有所发展或是发生改变。

    1 、别家的服务器返回数据,在 Chrome 中的 response 查看为:

    {"message":"查询成功","data": "yo"} 

    2 、有人的服务器返回是:

    "{\"rateInfos\":[]}" 

    第一种就直接是 json 对象了,第二种是字符串,为什么呢?按道理 http 传输过程中,数据应该都是字符串,为何第一种可以直接为 json 对象了呢?

    查看了下,两者的 response header 都有: Content-Type: application/json

    有点蒙圈,求解答

    第 1 条附言    2016-05-11 11:53:50 +08:00

    通过不断 google + 实验,我觉得我得到了答案。

    首先, response header 内的 content_type 仅仅只是表明响应数据是怎样的数据,并不会自动去解析成 JSON 对象 (注: 有浏览器原生 JSON 对象,其他的都为语言的对象)。

    比如原生 AJAX ,拿到该响应,仅仅只是 string 。 而 jQuery 则会根据 content_type 去将响应解析成对应的格式

    至于浏览器开发者工具,你看到的 JSON 对象,是浏览器开发者工具为了方便我们开发,自己将它解析成了对应的格式。

    接着,当你在服务手动创建:

    # Rails render text: '{"a": "a"}', content_type: "application/json" 

    此时你在浏览器就会看到问题内的第一种情况,也即正常情况。 浏览器会根据 content_type 将他解析

    # Rails render json: { a: 'a' } 

    此时, Rails 会自动将他序列化成字符串(即跟手动创建字符串没两样),然后进行传输。

    那么,问题第二种情况是怎样的呢?如果服务器仅仅只是返回 JSON 字符串的话,那么肯定也是会自动解析的呀。答案在于服务器的开发人员傻了。

    # 正常情况 '{"a": "a"}' # 而开发人员由于某种原因,将 json 字符串又序列化了一遍 # 类似于这种效果 JSON.stringify(JSON.stringify(jsonObejct)) # 因此最终服务器返回的字符串其实是: "\"{\\\"a\\\": \\\"a\\\"\}\"" # 浏览器拿到的字符串解析后是: "{\"a\": \"a\"}" 
    13 条回复    2016-05-11 11:52:57 +08:00
    bdbai
        1
    bdbai  
       2016-05-10 18:19:15 +08:00 via Android
    第一种看起来正常一点,第二种是直接拿去 eval 了吧。
    wesley
        2
    wesley  
       2016-05-10 18:25:45 +08:00
    没啥好蒙的, 把第二个的开发人员拖出来弹 JJ 1000 遍
    learnshare
        3
    learnshare  
       2016-05-10 18:25:52 +08:00
    第二种错了,那只是字符串。问题出在后端, content-type 和内容不一致。
    just1
        4
    just1  
       2016-05-10 18:30:20 +08:00 via Android
    第二个代码写错了吧
    JiShuTui
        5
    JiShuTui  
       2016-05-10 18:33:21 +08:00
    下面这种情况是可以的
    {"message":"查询成功","data": "yo","data2":"{\"rateInfos\":[]}"}
    iyuki
        6
    miyuki  
       2016-05-10 19:36:12 +08:00 via Android
    正常人写的的都是第一种
    Pure88
        7
    Pure88  
       2016-05-10 22:11:57 +08:00
    @wesley 上下左右各弹 250 遍?
    murmur
        8
    murmur  
       2016-05-10 22:35:13 +08:00
    第二个如果没有 jsonp 的 callback 直接拉出去打
    sensui7
        9
    sensui7  
       2016-05-10 22:59:37 +08:00
    网络传输的都是字符串, 这是肯定的, 区别就是一种字符串包含的是 json 编码,楼猪第一个就是这种字符串,
    另一种是普通字符串, 你的第二个 , 只不过这是手动写成了 json 格式了. 相当于 手动版 json_encode
    LinJunzhu
        10
    LinJunzhu  
    OP
       2016-05-10 23:21:27 +08:00
    唔,我想了下:

    第一种,是因为服务器返回了 json 对象,并且服务器语言有相应的序列化 json 接口,在传输时会将 json 对象序列化,浏览器或语言 接收到该响应时,根据 content_type 去反序列解析响应,所以会直接拿到 json 对象。

    第二种,是因为浏览器返回的就是 json 字符串,所以客户端拿到的响应当然仅仅只是字符串了。

    不知道对不对呢?
    LinJunzhu
        11
    LinJunzhu  
    OP
       2016-05-10 23:24:46 +08:00
    @sensui7 感谢,不过我有个疑问,客户端是如何知道这个这个字符串是 json 编码并且进行反序列化呢?。 因为第二种 json 字符串,也符合 json 规范,也是可以进行 parse 操作的呀?
    congeec
        12
    congeec  
       2016-05-10 23:35:53 +08:00   1
    打死也不能 eval() !!!!
    用 JSON.parse()
    LinJunzhu
        13
    LinJnzhu  
    OP
       2016-05-11 11:52:57 +08:00
    通过不断 google + 实验,我觉得我得到了答案。

    首先, response header 内的 content_type 仅仅只是表明响应数据是怎样的数据,并不会自动去解析成 JSON 对象 (注: 有浏览器原生 JSON 对象,其他的都为语言的对象)。

    比如原生 AJAX ,拿到该响应,仅仅只是 string 。 而 jQuery 则会根据 content_type 去将响应解析成对应的格式

    至于浏览器开发者工具,你看到的 JSON 对象,是浏览器开发者工具为了方便我们开发,自己将它解析成了对应的格式。

    接着,当你在服务手动创建:
    ```ruby
    # Rails
    render text: '{"a": "a"}', content_type: "application/json"
    ```

    此时你在浏览器就会看到问题内的第一种情况,也即正常情况。 浏览器会根据 content_type 将他解析

    ```ruby
    # Rails
    render json: { a: 'a' }
    ```
    此时, Rails 会自动将他序列化成字符串(即跟手动创建字符串没两样),然后进行传输。

    那么,问题第二种情况是怎样的呢?如果服务器仅仅只是返回 JSON 字符串的话,那么肯定也是会自动解析的呀。答案在于服务器的开发人员傻了。

    ```ruby
    # 正常情况
    '{"a": "a"}'

    # 而开发人员由于某种原因,将 json 字符串又序列化了一遍
    # 类似于这种效果
    JSON.stringify(JSON.stringify(jsonObejct))

    # 因此最终服务器返回的字符串其实是:
    "\"{\\\"a\\\": \\\"a\\\"\}\""

    # 浏览器拿到的字符串解析后是:
    "{\"a\": \"a\"}"

    ```
    关于     帮助文档     自助推广系统     博客     API     FAQ     Solana     2376 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 24ms UTC 05:28 PVG 13:28 LAX 21:28 JFK 00:28
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86