php中json 序列化为 [] 的弊端

作者:mb67ba89ce46f84 时间:2023-05-25 00:14:30 

在 PHP 中表示空的map或空数组都是以空数组形式,在转化为json数据时,会将空数组统一 json 序列化成 [],这样就存在一个类型问题。

以前我们在与前端交互时一般是与弱类型语言js交互,对于空数组转成 {} 还是 [] 区别不大。

但随着APP的流行,PHP很多时候不是跟浏览器端的JS交互,而是跟Java和ObjC这样的静态类型语言交互,返回值的类型定义,就很重要了,举个例子

$ret1 = [ 
    'choices' => ['鱼香肉丝', '宫保鸡丁'],
    'answers' => [
        '张三' => 0,
        '李四' => 1,
        '赵云' => 0,
    ],  
];

$ret2 = [ 
    'choices' => [], 
    'answers' => [], 
];

echo json_encode($ret1) . "\n";
echo json_encode($ret2) . "\n";

输出

{"choices":["\u9c7c\u9999\u8089\u4e1d","\u5bab\u4fdd\u9e21\u4e01"],"answers":{"\u5f20\u4e09":0,"\u674e\u56db":1,"\u8d75\u4e91":0}}

{"choices":[],"answers":[]}

客户端在定义这个model的时候,可能是这样定义的

class ResultDTO {
    lateinit var choices: List<String>
    lateinit var answers: Map<String, Int>
}

当返回ret1的时候,一切顺风顺水,皆大欢喜。如果返回ret2呢,客户端抗议了

com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of java.util.LinkedHashMap out of START_ARRAY token

原因是什么呢?PHP的json_encode面对一个空的array的时候,它很为难,它不知道应该当它是list还是map,所以它只能一刀切,认为它就是list,于是客户端就不高兴了。解决办法不是没有,依然是强制转换。

解决方法一:[推荐]

$arr = [
     'choices' => [],
    'answers' => new \ArrayObject([])
];

$jsonRet = json_encode($arr);

print_r($jsonRet);

使用 ArrayObject 还可以像数组一样操作数据,方便很多

解决方法二:

$ret2 = [
    'choices' => [],
    'answers' => (object) [],
];

但是这样就带来一个问题,如果answers不是写死的,而是某个API的返回值,你并不确定它是不是会返回空的,它也没有义务帮你cast成object,因为JSON序列化是跟前端交互的事情,不应该放到后端service层面解决。那么你只能自己动手了,手动把返回值中可能出现空map的地方,全部强制转换一遍。

PHP的关联数组的确很强大,算法设计的也不错,性能也很好,但是它不是没有代价的,上面的例子算是其中一个。如果PHP也像其它语言一样,区分map和list,可能会省事一些,毕竟区分{}和[],对程序员来说并不会增加很多学习成本。

来源:https://blog.51cto.com/sdwml/6131399

标签:json,序列化
0
投稿

猜你喜欢

  • 用JS开发页面动画效果时的一个设计思路

    2008-02-03 15:12:00
  • asp如何实现点击数的计算?

    2010-05-18 18:39:00
  • asp如何用CDONTS发送带附件的邮件?

    2010-06-11 19:57:00
  • asp更改Windows2000管理者密码?

    2010-06-26 11:03:00
  • 使用javascript+xml技术实现分页浏览

    2008-05-29 13:49:00
  • JS的编译和执行顺序

    2009-02-01 18:42:00
  • 运行asp.net程序 报错:磁盘空间不足

    2011-11-03 17:16:22
  • 3个常用的JS时间代码

    2009-03-22 15:29:00
  • SQL Server 2005 内置工具建审查系统

    2009-01-19 14:24:00
  • Pivot,信息组织的梦想之窗

    2009-12-25 14:32:00
  • 利用sort()和Math.random()实现元素的随机排列

    2010-10-19 12:42:00
  • 创建数据表/创建列的一些asp函数

    2008-06-24 12:21:00
  • 一些实用的sql语句

    2009-11-10 20:21:00
  • 使用Dreamweaver代码片断提高css开发效率

    2007-10-28 15:46:00
  • mysql 主从服务器的简单配置

    2009-09-06 12:06:00
  • Oracle数据库的安全策略

    2010-07-31 13:13:00
  • 详细了解 MySQL锁机制

    2010-08-08 09:04:00
  • 互联网一家之言(一):叫用户为你买单

    2009-06-09 11:32:00
  • 如何绝对获知浏览器类型?

    2009-12-16 18:58:00
  • 用MySQL做站点时如何记录未知错误的发生

    2010-09-30 14:11:00
  • asp之家 网络编程 m.aspxhome.com