Snack3 一个新的微型JSON框架
Snack3 一个新的微型JSON框架
一个作品,一般表达作者的一个想法。因为大家想法不同,所有作品会有区别。就做技术而言,因为有很多有区别的框架,所以大家可以选择的框架很丰富。
snack3。基于jdk8
,60kb
,无其它依赖,非常小巧。
- 强调文档树的链式操控和构建能力
- 强调中间媒体,方便不同格式互转
- 支持
序列化、反序列化
- 支持
Json path
查询
ONode 即 One node
之意;借签了 Javascript
所有变量由 var
申明,及 Xml dom
一切都是 Node
的设计。这应该也是与别的框架不同之处。
设计思路
ONode
是一个接口界面,内部由nodeType
和nodeData
组成。通过nodeType
进行类型识别,通过nodeData
容纳所有的类型数据。采用
Fromer
->ONode
->Toer
的架构设计。这是重要的一个设计,可使转换具有很强的扩展能力,Fromer
和Toer
各做一半工作,也可各自切换(有好处也会有坏处)。强调非null操作,通过o.isNull()进行判断(留意后面
-get(key | idx)
接口的注释;可通过getOrNull(key | idx)
获取null
)
项目源码
Meven配置
<dependency>
<groupId>org.noear</groupId>
<artifactId>snack3</artifactId>
<version>3.1.5.12</version>
</dependency>
一、简单的代码演示
- 简单操控
//1.加载json
// name支持没有引号
// 字符串支持单引号(在java里写代码方便点;输出是标准的引号)
ONode n = ONode.load("{code:1,msg:'Hello world',data:{list:[1,2,3,4,5]}}");
//2.1.取一个属性的值
String msg = n.get("msg").getString();
//2.2.取列表里的一项
int li2 = n.get("data").get("list").get(2).getInt();
int li2 = n.select("data.list[2]").getInt(); //使用josn path
//2.3.获取一个数组
List<Integer> ary = n.get("data").get("list").toObject(ArrayList.class);
List<Integer> ary = n.select("data.list").toObject(ArrayList.class); //使用josn path
//3.1.设置值
n.set("msg","Hello world2");
n.get("msg").val("Hello world2"); //另一种设置方式
//3.2.设置列表里的一项值为12
n.get("data").get("list").get(2).val(12);
n.select("data.list[2]").val(12); //使用josn path
//3.3.清掉一个数组
n.get("data").get("list").clear();
n.select("data.list").clear(); //使用josn path
二、更多代码演示
- 1.字符串化
//将 java object 转为 json
String json = ONode.stringify(user);
- 2.序列化
//1.序列化(通过@type申明类型)
String json = ONode.serialize(user);
//2.反序列化
UserModel user = ONode.deserialize(json, UserModel.class);
- 3.转换
//1.1.将json string 转为 ONode
ONode o = ONode.load(json);
//1.2.将java bean 转为 ONode
ONode o = ONode.load(user);
//2.1.将 ONode 转为 string(由转换器决定,默认转换器为json)
o.toString();
//2.2.将 ONode 转为 json
o.toJson();
//2.3.将 ONode 转为XxxModel (例:UserModel)
o.toObject(UserModel.class);
//>将 ONode 转为 Map 或 List 或 常规值
o.toObject(null);
//>将 ONode 转为 自动类型的 Java Object
o.toObject(Object.class);
- 4.填充
ONode o = ONode.load("{code:1,msg:'Hello world!'}");
//填充java object
List<UserModel> list = new ArrayList<>();
//...
o.get("data").get("list").fill(list);
//填充string json
o.get("data").get("list").fill("[{a:1,b:2},{a:2,c:3}]");
- 5.一个应用示例(极光推送的rest api调用)
public static void push(Collection<String> alias_ary, String text) {
ONode data = new ONode().build((d)->{
d.get("platform").val("all");
d.get("audience").get("alias").addAll(alias_ary);
d.get("options")
.set("apns_production",false);
d.get("notification").build(n->{
n.get("ios")
.set("alert",text)
.set("badge",0)
.set("sound","happy");
});
});
String message = data.toJson();
String author = Base64Util.encode(appKey+":"+masterSecret);
Map<String,String> headers = new HashMap<>();
headers.put("Content-Type","application/json");
headers.put("Authorization","Basic "+author);
HttpUtil.postString(apiUrl, message, headers);
}
- 5.数据遍历
//遍历Object数据 方案1
o.forEach((k,v)->{
//...
});
//遍历Object数据 方案2
for(Map.Entry<String,ONode> kv : n.obj().entrySet()){
//...
}
//遍历Array数据 方案1
o.forEach((v)->{
//...
});
//遍历Array数据 方案2
for(ONode v : o.ary()){
//..
}
三、混合加载与转换代码演示
List<UserModel> list = new ArrayList<>();
//..
ONode o = ONode.load("{code:1,msg:'succeed'}");
o.get("data").get("list").fill(list);
关于三组类型接口的设计(Json object,array,value)
- 关于
json object
的操作
-obj() -> Map<String,ONode> //获取节点对象数据结构体(如果不是对象类型,会自动转换)
-contains(key:String) -> bool //是否存在对象子节点?
-get(key:String) -> child:ONode //获取对象子节点(不存在,生成新的子节点并返回)
-getOrNull(key:String) -> child:ONode //获取对象子节点(不存在,返回null)
-getNew(key:String) -> child:ONode //生成新的对象子节点,会清除之前的数据
-set(key:String,val:Object) -> self:ONode //设置对象的子节点(会自动处理类型)//val:为常规类型或ONode
-setNode(key:String,val:ONode) -> self:ONode //设置对象的子节点,值为ONode类型
-setAll(obj:ONode) -> self:ONode //设置对象的子节点,将obj的子节点搬过来
-setAll(map:Map<String,T>) ->self:ONode //设置对象的子节点,将map的成员搬过来
-setAll(map:Map<String,T>, (n,t)->..) ->self:ONode //设置对象的子节点,将map的成员搬过来,并交由代理处置
-remove(key:String) //移除对象的子节点
-forEach((k,v)->..) //遍历对象的子节点
- 关于
json array
的操作
-ary() -> List<ONode> //获取节点数组数据结构体(如果不是数组,会自动转换)
-get(index:int) -> child:ONode //获取数组子节点(超界,返回空节点)
-getOrNull(index:int) -> child:ONode //获取数组子节点(超界,返回null)
-addNew() -> child:ONode //生成新的数组子节点
-add(val) -> self:ONode //添加数组子节点 //val:为常规类型或ONode
-addNode(val:ONode) -> self:ONode //添加数组子节点,值为ONode类型
-addAll(ary:ONode) -> self:ONode //添加数组子节点,将ary的子节点搬过来
-addAll(ary:Collection<T>) -> self:ONode //添加数组子节点,将ary的成员点搬过来
-addAll(ary:Collection<T>,(n,t)->..) -> self:ONode //添加数组子节点,将ary的成员点搬过来,并交由代理处置
-removeAt(index:int) //移除数组的子节点
-forEach(v->..) //遍历数组的子节点
- 关于
json value
的操作
-val() -> OValue //获取节点值数据结构体(如果不是值类型,会自动转换)
-val(val:Object) -> self:ONode //设置节点值 //val:为常规类型或ONode
-getString() //获取值并以string输出 //如果节点为对象或数组,则输出json
-getShort() //获取值并以short输出...(以下同...)
-getInt()
-getBoolean()
-getLong()
-getDate()
-getFloat()
-getDouble()
-getDouble(scale:int)
-getChar()
关于序列化的特点
- 对象
{"@type":"...","a":1,"b":"2"}
- 数组(可以精准反序列化列表类型;需要特性开启)
[{"@type":"..."},[1,2,3]]
关于Json path的支持
- 支持完整的
Json path
的选择操作
支持操作 | 说明 |
---|---|
$ | 查询根元素(或当前元素)。 这将启动所有路径表达式(可以不输)。 |
* | 通配符,必要时可用任何地方的名称或数字。 |
.. | 深层扫描。 必要时在任何地方可以使用名称。 |
. | 点,表示子节点 |
['' (, '')] | 括号表示子项 |
[ (, )] | 数组索引或索引 |
[start:end] | 数组切片操作 |
例:n.select("$.store.book[0].title")
或 n.select("$['store']['book'][0]['title']")
附:完整的接口字典
//初始化操作
//
-asObject() -> self:ONode //将节点切换为对象
-asArray() -> self:ONode //将节点切换为数组
-asValue() -> self:ONode //将节点切换为值
-asNull() -> self:ONode //将节点切换为null
//检测操作
//
-isObject() -> bool //检查节点是否为对象
-isArray() -> bool //检查节点是否为数组
-isValue() -> bool //检查节点是否为值
-isNull() -> bool //检查节点是否为null
//公共
//
-nodeData() -> ONodeData //获取节点数据
-nodeType() -> ONodeType //获取节点类型
-cfg(cfg:Constants) -> self:ONode //切换配置
-cfg() -> Constants //获取配置
-build(n->..) -> self:ONode //节点构建表达式
-select(jpath:String) -> new:ONode //使用JsonPath表达式选择节点(默认缓存路径编译)
-select(jpath:String, useStandard:boolean)-> new:ONode //useStandard:使用标准模式,默认非标准
-select(jpath:String, useStandard:boolean, cacheJpath:boolean)-> new:ONode //cacheJpath:是否缓存javaPath编译成果,默认缓存
-clear() //清除子节点,对象或数组有效
-count() -> int //子节点数量,对象或数组有效
//值操作
//
-val() -> OValue //获取节点值数据结构体(如果不是值类型,会自动转换)
-val(val:Object) -> self:ONode //设置节点值 //val:为常规类型或ONode
-getString() //获取值并以string输出 //如果节点为对象或数组,则输出json
-getShort() //获取值并以short输出...(以下同...)
-getInt()
-getBoolean()
-getLong()
-getDate()
-getFloat()
-getDouble()
-getDouble(scale:int)
-getChar()
//对象操作
//
-obj() -> Map<String,ONode> //获取节点对象数据结构体(如果不是对象类型,会自动转换)
-readonly() -> self:ONode //只读形态(get时,不添加子节点)
-contains(key:String) -> bool //是否存在对象子节点?
-rename(key:String,newKey:String) -> self:ONode //重命名子节点并返回自己
-get(key:String) -> child:ONode //获取对象子节点(不存在,生成新的子节点并返回)
-getOrNull(key:String) -> child:ONode //获取对象子节点(不存在,返回null)
-getNew(key:String) -> child:ONode //生成新的对象子节点,会清除之前的数据
-set(key:String,val:Object) -> self:ONode //设置对象的子节点(会自动处理类型)
-setNode(key:String,val:ONode) -> self:ONode //设置对象的子节点,值为ONode类型
-setAll(obj:ONode) -> self:ONode //设置对象的子节点,将obj的子节点搬过来
-setAll(map:Map<String,T>) ->self:ONode //设置对象的子节点,将map的成员搬过来
-setAll(map:Map<String,T>, (n,t)->..) ->self:ONode //设置对象的子节点,将map的成员搬过来,并交由代理处置
-remove(key:String) //移除对象的子节点
-forEach((k,v)->..) -> self:ONode //遍历对象的子节点
//数组操作
//
-ary() -> List<ONode> //获取节点数组数据结构体(如果不是数组,会自动转换)
-get(index:int) -> child:ONode //获取数组子节点(超界,返回空节点)
-getOrNull(index:int) -> child:ONode //获取数组子节点(超界,返回null)
-addNew() -> child:ONode //生成新的数组子节点
-add(val) -> self:ONode //添加数组子节点 //val:为常规类型或ONode
-addNode(val:ONode) -> self:ONode //添加数组子节点,值为ONode类型
-addAll(ary:ONode) -> self:ONode //添加数组子节点,将ary的子节点搬过来
-addAll(ary:Collection<T>) -> self:ONode //添加数组子节点,将ary的成员点搬过来
-addAll(ary:Collection<T>,(n,t)->..) -> self:ONode //添加数组子节点,将ary的成员点搬过来,并交由代理处置
-removeAt(index:int) //移除数组的子节点
-forEach(v->..) -> self:ONode //遍历数组的子节点
//特性操作(不破坏数据的情况上,添加数据;或用于构建xml dom)
//
-attrGet(key:String) -> String //获取特性
-attrSet(key:String,val:String) -> self:ONode //设置特性
-attrForeach((k,v)->..) -> self:ONode //遍历特性
//转换操作
//
-toString() -> String //转为string (由字符串转换器决定,默认为json)
-toJson() -> String //转为json string
-toData() -> Object //转为数据结构体(Map,List,Value)
-toObject(clz:Class<T>) -> T //转为java object(clz=Object.class:自动输出类型)
-to(toer:Toer, clz:Class<T>) -> T //将当前节点通过toer进行转换
-to(toer:Toer) -> T //将当前节点通过toer进行转换
//填充操作(为当前节点填充数据;source 为 String 或 java object)
-fill(source:Object) -> self:ONode //填充数据
-fill(source:Object, fromer:Fromer) -> self:ONode //填充数据,由fromer决定处理
/**
* 以下为静态操作
**/
//加载操作(source 为 String 或 java object)
//
+load(source:Object) -> new:ONode //加载数据
+load(source:Object, cfg:Constants) -> new:ONode
+load(source:Object, cfg:Constants, fromer:Fromer) -> new:ONode
//加载 string
+loadStr(source:String) -> new:ONode //仅String
//加载 java object
+loadObj(source:Object) -> new:ONode //仅java object
//字符串化操作
//
+stringify(source:Object) -> String //字符串化;
//序列化操作
//
+serialize(source:Object) -> String //序列化(带@type属性)
+deserialize(source:String) -> T //反序列化
+deserialize(source:String, clz:Class<?>) -> T //反序列化
Snack3 一个新的微型JSON框架的更多相关文章
- Weed3 for java 新的微型ORM框架
Weed3,微型ORM框架(支持:java sql,xml sql,annotation sql:存储过程:事务:缓存:监听:等...) 05年时开发了第一代: 08年时开发了第二代,那时候进入互联网 ...
- 高性能JSON框架之FastJson的简单使用
1.前言 1.1.FastJson的介绍: JSON协议使用方便,越来越流行,JSON的处理器有很多,这里我介绍一下FastJson,FastJson是阿里的开源框架,被不少企业使用,是一个极其优秀的 ...
- dotweb——go语言的一个微型web框架(一)
dotweb是16年正式托管到github的一个开源项目,go语言的web框架目前也有很多,出名的有bee和echo.它们都是很优秀的框架,但是我们喜欢更轻.更小的东西,经历一些之后我们更青睐微服务这 ...
- dotweb——go语言的一个微型web框架(二)启动dotweb
以上的代码截图表示启动一个dotweb服务,在浏览器里输入127.0.0.1:8080,将会得到一个"index"的页面. app := dotweb.New() dotweb.N ...
- Linux下利用json-c从一个json数组中提取每一个元素中的部分字段组成一个新json数组
先把代码贴上来,有时间整理一下 首先说一下要实现的功能: 假定现在有一个json格式的字符串,而且他是一个josn中的数组,比如: [ { "id": "NEW20170 ...
- 如何着手学习一个新的PHP框架
如今的PHP框架层出不穷,名气也各不相同.如何快速掌握一种框架?看看本文吧~ 如今的PHP框架层出不穷,名气也各不相同.我不是这方面的专家,甚至不能熟练地使用其中的一种,所以就不作推荐了.这里我要讨论 ...
- JS解析Json 数据并跳转到一个新页面,取消A 标签跳转
JS解析Json 数据并跳转到一个新页面,代码如下 $.getJSON("http://api.cn.abb.com/common/api/staff/employee/" + o ...
- Google C++测试框架系列入门篇:第二章 开始一个新项目
上一篇:Google C++测试框架系列入门篇:第一章 介绍:为什么使用GTest? 原始链接:Setting up a New Test Project 词汇表 版本号:v_0.1 开始一个新项目 ...
- 【Flask】微型web框架flask大概介绍
Flask Flask是一个基于python的,微型web框架.之所以被称为微型是因为其核心非常简单,同时具有很强的扩展能力.它几乎不给使用者做任何技术决定. 安装flask时应该注意其必须的几个支持 ...
随机推荐
- vue.config.js常用配置
使用vue-cli3.0搭建项目比之前更简洁,没有了build和config文件夹. vue-cli3的一些服务配置都迁移到CLI Service里面了,对于一些基础配置和一些扩展配置需要在根目录新建 ...
- 《计算机网络 自顶向下方法》 第2章 应用层 Part2
域名.主机名? 从范围上看: 域名的范围比主机名大 一个域名下通常有多个主机名 从组成上看: 主机名 = 服务器名(或计算机名) + 域名 举例说明: baidu.com 是百度的域名 www.b ...
- awk 实用案例介绍
awk 简介 • awk是 3 个姓氏的首字母,代表该语言的 3 个作者 • awk的版本有很多,包括: 旧版 awk,新版 awk(nawk), GNUawk(gawk)等 • awk程序有 awk ...
- 使用 Casbin 作为 ThinkPHP 的权限控制中间件
PHP-Casbin 是一个强大的.高效的开源访问控制框架,它支持基于各种访问控制模型的权限管理. Think-Casbin 是一个专为 ThinkPHP5.1 定制的 Casbin 的扩展包,使开发 ...
- nyoj 52-无聊的小明 (模拟, SET)
52-无聊的小明 内存限制:64MB 时间限制:3000ms Special Judge: No accepted:1 submit:3 题目描述: 这天小明十分无聊,没有事做,但不甘于无 ...
- .NET高级特性-Emit(1)
在这个大数据/云计算/人工智能研发普及的时代,Python的崛起以及Javascript的前后端的侵略,程序员与企业似乎越来越青睐动态语言所带来的便捷性与高效性,即使静态语言在性能,错误检查等方面的优 ...
- windows 10 上源码编译OpenCV并支持CUDA | compile opencv with CUDA support on windows 10
本文首发于个人博客https://kezunlin.me/post/6580691f/,欢迎阅读! compile opencv with CUDA support on windows 10 Ser ...
- 对js中局部变量、全局变量和闭包的理解
对js中局部变量.全局变量和闭包的理解 局部变量 对于局部变量,js给出的定义是这样的:在 JavaScript函数内部声明的变量(使用 var)是局部变量,所以只能在函数内部访问它.(该变量的作用域 ...
- 在UEFI+GPT下使用rEFind实现Win10 + Kali2.0 双引导
转载自:在UEFI+GPT下使用rEFind实现Win10 + Kali2.0 双引导 https://www.linuxidc.com/Linux/2016-07/133717.htm
- 联想Y7000,I5-9300H+Nvidia GTX 1050, kali linux的nvidia显卡驱动安装
转载自,Linux安装NVIDIA显卡驱动的正确姿势 https://blog.csdn.net/wf19930209/article/details/81877822#NVIDIA_173 ,主要用 ...