
例如一个接口,能够收到登录、更改密码两种请求,收到的 json 结构如下:
{ "t":"l", //登录 "username":"name1", "password":"pass1" } 或 { "t":"cp", // 更改密码 "username":"name1", "oldpass":"oldpass", "newpass":"newpass" } 在未反序列化之前后端无法知道数据属于哪个结构,只能直接反序列化到通用 map 或者先反序列化一次获得类型 t 的值,然后再次解析到具体结构。
但是这两个方案都有些小问题,静态语言下,map 通用结构使用不方便;两次反序列化方案浪费 cpu 资源。
原来是尽可能的使用不同的接口处理不同结构的数据,tcp 连接等二进制协议则自定义包头来区分,一直没什么大问题。
但是最近在处理 websocket 大量消息,就不太好用了。不可能每个结构使用一个独立的 websocket 连接,而常用的二进制协议包头方案也因为 web 前端不太方便不能使用(我不是专业前端,可能会有误解)。
那么对于这种情况,静态类型语言后端怎么处理比较好?
目前我这里的方案是,固定成为类似这样的结构:
{ "t":"?", "login":{ "username":"user1", "password":"pass1", }, "changepassword":{ "username":"name1", "oldpass":"oldpass", "newpass":"newpass" } } login 时 login 字段有值,changepassword 为 null 的方式工作。请教下,还有其他更好的方案吗?
1 wysnylc 2020-05-07 14:45:04 +08:00 加个字段声明请求内容是约定的什么结构 |
2 optional 2020-05-07 15:25:29 +08:00 Java 里 jackson subtype 可以直接吐出来 |
3 DGideas 2020-05-07 15:29:23 +08:00 |
4 teawithlife 2020-05-07 16:04:38 +08:00 我能想到的有两个办法: 1. 增加包头,不一定是二进制的包头,只要定长就可以了,前端应该很好实现。而且前端处理二进制也是没问题的,有对应的库 2. 固定一个字段做匹配,比如要求有个"__type__"字段,用来表示结构体格式,然后先用 substring 一类的函数,把这段内容提取出来解析,确定了格式之后,就按对应的格式解析。 |
5 HuHui 2020-05-07 16:08:57 +08:00 via Android JSON-RPC? |
6 dr1q65MfKFKHnJr6 2020-05-07 16:17:53 +08:00 貌似 现在用的这个 就是最优解了 |
7 gamexg OP @wysnylc #1 目前用 t 表示的类型,小缺陷是需要反序列化两次,或者一个结构包含所有的请求体。 @optional #2 这个没用过,看起来是 json 库本身的功能? @teawithlife #4 包头是一个方案。substring 方案需要约定请求内不能出现其他 __type__ 字段,并且要求必须是固定格式,不能多出空格等,限制大了些。 @HuHui #5 用现成的协议也是个选择。 |