Mapl 结构

Jun 9, 2018 10:07:08 PM

作者:juqkai

为什么需要 Mapl 结构

一直以来都没有刻意的去思考说需要 Mapl 结构这样的东西. 所谓 Mapl 结构就是 Map-List 结构,我原来就叫 maplist 后来被灰太郎说太长,就改成 "mapl" 了

最初都是 wendal 在重写JSON的时候, 将JSON理解成了Mapl结构.然后突然在某一天发现, 咦, Mapl对象还可以这么用, 那么用, 慢慢的, 以Mapl 基础的小功能点越来越多, 已经不能够完全的与JSON概念协调, 所以才有了这样一个中间结构.

恩, 上面那段是给灰太狼, wendal看的. 我想你肯定看得头有点大, 要不我再说直白点, Mapl结构 就是为Json服务的, 为什么呢? MD, Json.fromJson()忒难用了, 要是只给它一个Reader, 而不给Type, 那它给我返回的就是 Mapl结构 , 苍天呀, 大地呀, 烦都烦死了.

我要取其中的某个值, 我得遍历N层的Map, List, 每次写这种东西的时候, 我都想哭, 所以干脆对它封装吧, 越封越多, 然后就有了这玩意…

当然, 它也与EL一样满足一小撮人的一小撮要求

提醒: Mapl这个名词,是MapList的缩写

什么是 Mapl 结构?

一种以 Map, List 接口对象所组织起来的结构体系. 类似于JSON结构便于JAVA在内存中处理的结构. 主要提供键值对, 与列表的有机组合, 因这种结构只由Map, List组成, 因些称其为Mapl结构.

  1. Map a = new HashMap();
  2. a.put("name","a");
  3. Map b = new HashMap();
  4. b.put("name","b");
  5. Map c = new HashMap();
  6. c.put("name","c");
  7. List list = new ArrayList();
  8. list.add(a);
  9. list.add(b);
  10. list.add(c);
  11. Map d = new HashMap();
  12. d.put("items", list);

通过上面的代码我们就组织了一个Mapl结构, 它等效于以下的JSON文档:

  1. {"items":[{"name":"a"},{"name":"b"},{"name":"c"}]}

当然, Mapl.仅可以用来表示JSON, 也可以用来表示JAVA对象的结构, 然后有了Mapl., 你会发现, 做转换, 合并, 都是非常轻松滴~~~

具体规则:

  • 对象以Map存储, key为属性名, value为属性值
  • 数组以List存储
  • Map直接存储为Map
  • List直接存储为List
  • 只要不是List, Map存储的, 都认为是可以直接写入对象的

Mapl.转 对象

也就是根据Mapl.及Type信息转换成一个Type的实体对象了啦, 直接看例子:

  1. class A{
  2. String name;
  3. Integer id;
  4. }
  5. class B{
  6. String name;
  7. List<A> as;
  8. }
  9. class C{
  10. public static void main(String args[]){
  11. String json = "{'name':'b', 'as':[{'name':'nutz','id':1},{'name':'jk','id':2}]}";
  12. //这样得到的就是Mapl结构的数据了.
  13. Object obj = Json.fromJson(json);
  14. B b = Mapl.maplistToObj(obj, B.class);
  15. }
  16. }

通过上面的 Mapl.maplistToObj() 方法就可以将一个Mapl.象转换成B类型的实体对象. 我偷偷的告诉你哦, JSON里面也是这样搞的哦 ~~~先将JSON字符串转换成Mapl结构后再调用Mapl.maplistToObj()方法转换成对应的类型.

对象转Maplist

除了通过JSON转换成Mapl结构以外, 还可以直接使用对象来转换成maplist结构

  1. A a = new A();
  2. a.name="a"
  3. B b = new B();
  4. b.name = "b";
  5. b.as = new ArrayList();
  6. b.as.add(a);
  7. Mapl.toMapl.b);

结果:

  1. {name:"b", as:[{name:"a", id:null}]}

通过toMapl.可以进行这种简单的转换

访问 Maplist

就如我最开始说的那样, Json.fromJson 很难用, 就是因为在读取Mapl结构的数据时非常的繁杂, 经常需要很多层的类型转换.

  1. String json = "{'name':'b', 'as':[{'name':'nutz','id':1},{'name':'jk','id':2}]}";
  2. //这样得到的就是Mapl结构的数据了.
  3. Object obj = Json.fromJson(json);

上面的obj, 如果我想取as索引为1的name的值, 怎么办? 只能这样:

  1. Map map = (Map) obj;
  2. List list = map.get("as");
  3. Map item = list.get(1);
  4. String name = item.get("name");

亲, 看到没, 看到没~妈哦, 还好这里只有几层, 要是再多几次这样的, 我一定会疯的, 你肯定也跟我一样吧. 所以咯, 让我们解脱吧~

  1. String name = (String) Mapl.cell(obj, "as[1].name");

完了? 这就样? 是的, 完了, 就这样, 一句话搞定. so easy~~~

最后说说关于里面path的规则:

  • map的值访问直接使用 '.', 如: abc.name
  • list的访问使用 "名称[索引]", 如: as[1]. 当然要是不想写[]也可以使用 as.1.name的形式.
  • 顶层为list时, 使用 "[索引].其它", 如: [1].name
  • 如果想得到一个List, 而不是它某个值, 则可以使用 "名称" 不加 "[索引]". 如: as
  • 如果List后加了"[]"中间却没有索引, 则默认访问第一个元素, 如: user[] 等效 user[0]

maplist 合并

哇咔咔, 一个神器来鸟. 为嘛我要说它是神器呢, 看名字就知道了噻, 当然, 这个只是一小撮的一小撮的一小撮人会觉得是神器…额…好吧, 它只是一个没啥大用的一个伪神器…

顾名思义, maplist 合并, 就是将多个maplist合并在一起, 组成一个新的 maplist .

  1. String json1 = "{'name':'nutz'}";
  2. String json2 = "{'age':12}"
  3. Object obj1 = Json.fromJson(json1);
  4. Object obj2 = Json.fromJson(json2);
  5. Object obj3 = Mapl.merge(obj1, obj2);

最终obj3的输出将是:

  1. {"name":"nutz", 'age':12}

规则:

  • 普通对象, 保存为List, 但是要去除重复.
  • 合并map, 如果key值相同, 那么后一个值覆盖前面的值, 注意, 对值将会进行递归合并
  • list不做递归合并, 只做简单的合并, 清除重复

maplist 过滤

这玩意有什么用呢, 用来剔除/筛选 maplist 中的值, 使maplist更加满足我们的需求. 还是用例子来说明吧.

  1. String json = "{name:'nutz', age:12, address:[{area:1,name:'abc'}, {area:2,name:'123'}]}";
  2. Object obj = Json.fromJson(json);
  3. List<String> list = new ArrayList<String>();
  4. list.add("age");
  5. list.add("address[].area");
  6. Object newobj = Mapl.excludeFilter(obj, list);

结果:

  1. {name:'nutz', address:[{name:"abc"}, {name:"123"}]}

可以发现, 通过给定的过滤列表, 可以将原始的maplist结构给过滤掉满足条件的内容, 当然, 除了排除, 还有包含.

  1. String json = "{name:'nutz', age:12, address:[{area:1,name:'abc'}, {area:2,name:'123'}]}";
  2. Object obj = Json.fromJson(json);
  3. List<String> list = new ArrayList<String>();
  4. list.add("age");
  5. list.add("address[].area");
  6. Object newobj = Mapl.includeFilter(obj, list);

结果:

  1. {age:12, address:[{area:1},{area:2}]}

excludeFilter与includeFilter是一组完全相反的功能.

path规则:

  • map以 "key." 间隔
  • list以"key[]."间隔, 即多一个[], 注意其中没有索引哦.

maplist 结构转换

好吧, 我觉得这个才是神器~啦~啦~啦~~完全是神一样的存在.

有没有使用过其它公司的API? 有吧, 其它公司都返回些什么格式? 它的格式与你程序的格式一样吗? 或许有, 但大部分是不一样的, 对吧. 既然这样, 那结构转换是肯定的了.

  1. String json = "[{'name':'jk', 'age':12},{'name':'nutz', 'age':5}]";
  2. String model = "[{'name':['user[].姓名', 'people[].name'], 'age':['user[].年龄', 'people[].age']}]";
  3. String dest = "{\"people\":[{\"age\":12,\"name\":\"jk\"}, "
  4. + "{\"age\":5,\"name\":\"nutz\"}],"
  5. + "\"user\":[{\"姓名\":\"jk\",\"年龄\":12}, "
  6. + "{\"姓名\":\"nutz\",\"年龄\":5}]}";
  7. Object obj = Mapl.convert(Json.fromJson(new StringReader(json)), new StringReader(model));
  8. assertEquals(dest, Json.toJson(obj, new JsonFormat()));

结果:

  1. {
  2. "people":[
  3. {"age":12,"name":"jk"},
  4. {"age":5,"name":"nutz"}
  5. ],
  6. "user":[
  7. {"姓名":"jk","年龄":12},
  8. {"姓名":"nutz","年龄":5}
  9. ]
  10. }";

通过一个简单的操作, 我们就将一个maplist结构转换成了一个完全不一样的结构, 是不是很神奇?

什么是 maplist 结构转换呢? 就是将一种MapList结构转换成另外一种MapList结构.例:

  1. {
  2. "age":"123",
  3. "name":"juqkai"
  4. }

转换成:

  1. {
  2. "年龄":"123",
  3. "姓名":"juqkai"
  4. }

要进行这样的转换需要预先配置一个对应关系的配置, 具体的配置关系说明如下:

  • 使用原MapList一样的结构
  • 有数组的, 只写第一个元素的结构
  • 原结构中的值, 以字符串或字符串数组做为目标结构的对应关系
  • 对应关系可以为数组
  • 有数组的, 目标结构以key[].abc来代替数组
  • 原结构数组层次强制限定一致, 目标结构中'[]'的索引按原结构中出现先后顺序进行匹配.
  • 如果原结果不存在, 那默认为0
  • 未在模板中申明的不做转换
    例:

    例1:

  1. {
  2. "age":"user.年龄",
  3. "name":["user.name", "user.姓名"]
  4. }

例2

  1. (原json:[{"name":"nutz"},{"name":"juqkai"}]):
  2. [{
  3. "name":"[].姓名"
  4. }]

例3:

  1. {
  2. users:[
  3. {
  4. "name":["people[].name", "users[].name"],
  5. "age":"users[].name"
  6. }
  7. ]
  8. }

MapList的增删改

只有访问, 肯定是不够的, 难免会添加, 删除, 修改某个结点. 所以, 特意的为您添加了这些功能.很简单的, 其实就三个接口而已.添加: Mapl.put, Mapl.del, Mapl.update. 具体的使用方法, 你看看注释咯, 简单得很.

除了上面的几个一次性的接口外. MapList还包含一个 MaplRebuild 类, 从名字就可以很容易知道它是干嘛的. 没错, 就是Maplist重建. 你可以根据已有的Maplist来构建它, 也可以全新的构建它, 然后你就可以对它进行添加新的结点, 修改某个结点, 或者删除某个结点. 如此反复.

下面看看Mapl.put的实现你就知道怎么用了:

  1. /**
  2. * 添加新的结点
  3. * @param obj 原始的MapList
  4. * @param path 路径
  5. * @param val 值
  6. */
  7. public static void put(Object obj, String path, Object val) {
  8. Object mapList = Mapl.toMaplist(val);
  9. MaplRebuild rebuild = new MaplRebuild(obj);
  10. rebuild.put(path, mapList);
  11. }

本页面的文字允许在知识共享 署名-相同方式共享 3.0协议GNU自由文档许可证下修改和再使用。

原文: http://nutzam.com/core/maplist/overview.html