Jackson 框架,轻易转换JSON

Posted on

Jackson 框架,轻易转换JSON

hoojo 学习在于积累:滴水可以石穿! 学而不思则罔,思而不学则殆! 博客园 首页 博问 闪存 新随笔 联系 订阅订阅 管理

随笔-139 评论-674 文章-3 trackbacks-0

Jackson 框架,轻易转换JSON

Jackson可以轻松的将Java对象转换成json对象和xml文档,同样也可以将json、xml转换成Java对象。

前面有介绍过json-lib这个框架,在线博文:http://www.cnblogs.com/hoojo/archive/2011/04/21/2023805.html

相比json-lib框架,Jackson所依赖的jar包较少,简单易用并且性能也要相对高些。而且Jackson社区相对比较活跃,更新速度也比较快。

一、**准备工作**

1、 下载依赖库jar包

Jackson的jar all下载地址:http://jackson.codehaus.org/1.7.6/jackson-all-1.7.6.jar

然后在工程中导入这个jar包即可开始工作

官方示例:http://wiki.fasterxml.com/JacksonInFiveMinutes

因为下面的程序是用junit测试用例运行的,所以还得添加junit的jar包。版本是junit-4.2.8

如果你需要转换xml,那么还需要stax2-api.jar

2、 测试类基本代码如下 package com.hoo.test; import java.io.IOException;import java.io.StringWriter;import java.util.ArrayList;import java.util.HashMap;import java.util.Iterator;import java.util.LinkedHashMap;import java.util.List;import java.util.Map;import java.util.Set;import org.codehaus.jackson.JsonEncoding;import org.codehaus.jackson.JsonGenerationException;import org.codehaus.jackson.JsonGenerator;import org.codehaus.jackson.JsonParseException;import org.codehaus.jackson.map.JsonMappingException;import org.codehaus.jackson.map.ObjectMapper;import org.codehaus.jackson.node.JsonNodeFactory;import org.codehaus.jackson.xml.XmlMapper;import org.junit.After;import org.junit.Before;import org.junit.Test;import com.hoo.entity.AccountBean; //// function:Jackson 将java对象转换成JSON字符串,也可以将JSON字符串转换成java对象/ jar-lib-version: jackson-all-1.6.2/ jettison-1.0.1/ @author hoojo/ @createDate 2010-11-23 下午04:54:53/ @file JacksonTest.java/ @package com.hoo.test/ @project Spring3/ @blog http://blog.csdn.net/IBM_hoojo/ @email hoojo@126.com/ @version 1.0//@SuppressWarnings("unchecked")public class JacksonTest {private JsonGenerator jsonGenerator = null;private ObjectMapper objectMapper = null;private AccountBean bean = null;@Beforepublic void init() {bean = new AccountBean();bean.setAddress("china-Guangzhou");bean.setEmail("hoojo@126.com");bean.setId(1);bean.setName("hoojo");objectMapper = new ObjectMapper();try {jsonGenerator = objectMapper.getJsonFactory().createJsonGenerator(System.out, JsonEncoding.UTF8);} catch (IOException e) {e.printStackTrace();}}@Afterpublic void destory() {try {if (jsonGenerator != null) {jsonGenerator.flush();}if (!jsonGenerator.isClosed()) {jsonGenerator.close();}jsonGenerator = null;objectMapper = null;bean = null;System.gc();} catch (IOException e) {e.printStackTrace();}}}

3、 所需要的JavaEntity

package com.hoo.entity; public class AccountBean {private int id;private String name;private String email;private String address;private Birthday birthday;//getter、setter@Overridepublic String toString() {return this.name + "/#" + this.id + "/#" + this.address + "/#" + this.birthday + "/#" + this.email;}}

Birthday

package com.hoo.entity; public class Birthday {private String birthday;public Birthday(String birthday) {super();this.birthday = birthday;} //getter、setter public Birthday() {}@Overridepublic String toString() {return this.birthday;}}

二、**Java对象转换成JSON**

1、 JavaBean(Entity/Model)转换成JSON //// function:将java对象转换成json字符串/ @author hoojo/ @createDate 2010-11-23 下午06:01:10//@Testpublic void writeEntityJSON() {try {System.out.println("jsonGenerator");//writeObject可以转换java对象,eg:JavaBean/Map/List/Array等jsonGenerator.writeObject(bean);System.out.println();System.out.println("ObjectMapper");//writeValue具有和writeObject相同的功能objectMapper.writeValue(System.out, bean);} catch (IOException e) {e.printStackTrace();}}

运行后结果如下:

jsonGenerator{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo@126.com"}ObjectMapper{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo@126.com"}

上面分别利用JsonGenerator的writeObject方法和ObjectMapper的writeValue方法完成对Java对象的转换,二者传递的参数及构造的方式不同;JsonGenerator的创建依赖于ObjectMapper对象。也就是说如果你要使用JsonGenerator来转换JSON,那么你必须创建一个ObjectMapper。但是你用ObjectMapper来转换JSON,则不需要JSONGenerator。

objectMapper的writeValue方法可以将一个Java对象转换成JSON。这个方法的参数一,需要提供一个输出流,转换后可以通过这个流来输出转换后的内容。或是提供一个File,将转换后的内容写入到File中。当然,这个参数也可以接收一个JSONGenerator,然后通过JSONGenerator来输出转换后的信息。第二个参数是将要被转换的Java对象。如果用三个参数的方法,那么是一个Config。这个config可以提供一些转换时的规则,过指定的Java对象的某些属性进行过滤或转换等。

2、 将Map集合转换成Json字符串 //// function:将map转换成json字符串/ @author hoojo/ @createDate 2010-11-23 下午06:05:26//@Testpublic void writeMapJSON() {try {Map map = new HashMap();map.put("name", bean.getName());map.put("account", bean);bean = new AccountBean();bean.setAddress("china-Beijin");bean.setEmail("hoojo@qq.com");map.put("account2", bean);System.out.println("jsonGenerator");jsonGenerator.writeObject(map);System.out.println("");System.out.println("objectMapper");objectMapper.writeValue(System.out, map);} catch (IOException e) {e.printStackTrace();}}

转换后结果如下:

jsonGenerator{"account2":{"address":"china-Beijin","name":null,"id":0,"birthday":null,"email":"hoojo@qq.com"},"name":"hoojo","account":{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo@126.com"}}objectMapper{"account2":{"address":"china-Beijin","name":null,"id":0,"birthday":null,"email":"hoojo@qq.com"},"name":"hoojo","account":{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo@126.com"}}

3、 将List集合转换成json

//// function:将list集合转换成json字符串/ @author hoojo/ @createDate 2010-11-23 下午06:05:59//@Testpublic void writeListJSON() {try {List list = new ArrayList();list.add(bean);bean = new AccountBean();bean.setId(2);bean.setAddress("address2");bean.setEmail("email2");bean.setName("haha2");list.add(bean);System.out.println("jsonGenerator");//list转换成JSON字符串jsonGenerator.writeObject(list);System.out.println();System.out.println("ObjectMapper");//用objectMapper直接返回list转换成的JSON字符串System.out.println("1/#/#/#" + objectMapper.writeValueAsString(list));System.out.print("2/#/#/#");//objectMapper list转换成JSON字符串objectMapper.writeValue(System.out, list);} catch (IOException e) {e.printStackTrace();}}

结果如下:

jsonGenerator[{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo@126.com"},{"address":"address2","name":"haha2","id":2,"birthday":null,"email":"email2"}]ObjectMapper1/#/#/#[{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo@126.com"},{"address":"address2","name":"haha2","id":2,"birthday":null,"email":"email2"}]2/#/#/#[{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"},{"address":"address2","name":"haha2","id":2,"birthday":null,"email":"email2"}]

外面就是多了个[]中括号;同样Array也可以转换,转换的JSON和上面的结果是一样的,这里就不再转换了。~.~

4、下面来看看jackson提供的一些类型,用这些类型完成json转换;如果你使用这些类型转换JSON的话,那么你即使没有JavaBean(Entity)也可以完成复杂的Java类型的JSON转换。下面用到这些类型构建一个复杂的Java对象,并完成JSON转换。 @Testpublic void writeOthersJSON() {try {String[] arr = { "a", "b", "c" };System.out.println("jsonGenerator");String str = "hello world jackson!";//bytejsonGenerator.writeBinary(str.getBytes());//booleanjsonGenerator.writeBoolean(true);//nulljsonGenerator.writeNull();//floatjsonGenerator.writeNumber(2.2f);//charjsonGenerator.writeRaw("c");//StringjsonGenerator.writeRaw(str, 5, 10);//StringjsonGenerator.writeRawValue(str, 5, 5);//StringjsonGenerator.writeString(str);jsonGenerator.writeTree(JsonNodeFactory.instance.POJONode(str));System.out.println();//ObjectjsonGenerator.writeStartObject();//{jsonGenerator.writeObjectFieldStart("user");//user:{jsonGenerator.writeStringField("name", "jackson");//name:jacksonjsonGenerator.writeBooleanField("sex", true);//sex:truejsonGenerator.writeNumberField("age", 22);//age:22jsonGenerator.writeEndObject();//}jsonGenerator.writeArrayFieldStart("infos");//infos:[jsonGenerator.writeNumber(22);//22jsonGenerator.writeString("this is array");//this is arrayjsonGenerator.writeEndArray();//]jsonGenerator.writeEndObject();//}AccountBean bean = new AccountBean();bean.setAddress("address");bean.setEmail("email");bean.setId(1);bean.setName("haha");//complex ObjectjsonGenerator.writeStartObject();//{jsonGenerator.writeObjectField("user", bean);//user:{bean}jsonGenerator.writeObjectField("infos", arr);//infos:[array]jsonGenerator.writeEndObject();//}} catch (Exception e) {e.printStackTrace();}}

运行后,结果如下:

jsonGenerator"aGVsbG8gd29ybGQgamFja3NvbiE=" true null 2.2c world jac worl "hello world jackson!" "hello world jackson!"{"user":{"name":"jackson","sex":true,"age":22},"infos":[22,"this is array"]}{"user":{"address":"address","name":"haha","id":1,"birthday":null,"email":"email"},"infos":["a","b","c"]}

怎么样?构造的json字符串和输出的结果是一致的吧。关键看懂用JSONGenerator提供的方法,完成一个Object的构建。

三、**JSON转换成Java对象**

1、 将json字符串转换成JavaBean对象 @Testpublic void readJson2Entity() {String json = "{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}";try {AccountBean acc = objectMapper.readValue(json, AccountBean.class);System.out.println(acc.getName());System.out.println(acc);} catch (JsonParseException e) {e.printStackTrace();} catch (JsonMappingException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

很简单,用到了ObjectMapper这个对象的readValue这个方法,这个方法需要提供2个参数。第一个参数就是解析的JSON字符串,第二个参数是即将将这个JSON解析吃什么Java对象,Java对象的类型。当然,还有其他相同签名方法,如果你有兴趣可以一一尝试使用方法,当然使用的方法和当前使用的方法大同小异。运行后,结果如下:

hahahaha/#1/#address/#null/#email

2、 将json字符串转换成List集合

//// function:json字符串转换成list/ @author hoojo/ @createDate 2010-11-23 下午06:12:01//@Testpublic void readJson2List() {String json = "[{\"address\": \"address2\",\"name\":\"haha2\",\"id\":2,\"email\":\"email2\"},"+"{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}]";try {List> list = objectMapper.readValue(json, List.class);System.out.println(list.size());for (int i = 0; i < list.size(); i++) {Map map = list.get(i);Set set = map.keySet();for (Iterator it = set.iterator();it.hasNext();) {String key = it.next();System.out.println(key + ":" + map.get(key));}}} catch (JsonParseException e) {e.printStackTrace();} catch (JsonMappingException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

尝试过将上面的JSON转换成List,然后List中存放AccountBean,但结果失败了。但是支持Map集合。因为你转成List.class,但是不知道List存放何种类型。只好默然Map类型。因为所有的对象都可以转换成Map结合,运行后结果如下:

2address:address2name:haha2id:2email:email2address:addressname:hahaid:1email:email

3、 Json字符串转换成Array数组,由于上面的泛型转换不能识别到集合中的对象类型。所有这里用对象数组,可以解决这个问题。只不过它不再是集合,而是一个数组。当然这个不重要,你可以用Arrays.asList将其转换成List即可。

//// function:json字符串转换成Array/ @author hoojo/ @createDate 2010-11-23 下午06:14:01//@Testpublic void readJson2Array() {String json = "[{\"address\": \"address2\",\"name\":\"haha2\",\"id\":2,\"email\":\"email2\"},"+"{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}]";try {AccountBean[] arr = objectMapper.readValue(json, AccountBean[].class);System.out.println(arr.length);for (int i = 0; i < arr.length; i++) {System.out.println(arr[i]);}} catch (JsonParseException e) {e.printStackTrace();} catch (JsonMappingException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

运行后的结果:

2haha2/#2/#address2/#null/#email2haha/#1/#address/#null/#email

4、 Json字符串转换成Map集合

//// function:json字符串转换Map集合/ @author hoojo/ @createDate Nov 27, 2010 3:00:06 PM//@Testpublic void readJson2Map() {String json = "{\"success\":true,\"A\":{\"address\": \"address2\",\"name\":\"haha2\",\"id\":2,\"email\":\"email2\"},"+"\"B\":{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}}";try {Map> maps = objectMapper.readValue(json, Map.class);System.out.println(maps.size());Set key = maps.keySet();Iterator iter = key.iterator();while (iter.hasNext()) {String field = iter.next();System.out.println(field + ":" + maps.get(field));}} catch (JsonParseException e) {e.printStackTrace();} catch (JsonMappingException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

运行后结果如下:

3success:trueA:{address=address2, name=haha2, id=2, email=email2}B:{address=address, name=haha, id=1, email=email}

四、**JacksonXML的支持**

Jackson也可以完成java对象到xml的转换,转换后的结果要比json-lib更直观,不过它依赖于stax2-api.jar这个jar包。 //// function:java对象转换成xml文档/ 需要额外的jar包 stax2-api.jar/ @author hoojo/ @createDate 2010-11-23 下午06:11:21/*/@Testpublic void writeObject2Xml() {//stax2-api-3.0.2.jarSystem.out.println("XmlMapper");XmlMapper xml = new XmlMapper();try {//javaBean转换成xml//xml.writeValue(System.out, bean);StringWriter sw = new StringWriter();xml.writeValue(sw, bean);System.out.println(sw.toString());//List转换成xmlList list = new ArrayList();list.add(bean);list.add(bean);System.out.println(xml.writeValueAsString(list));//Map转换xml文档Map map = new HashMap();map.put("A", bean);map.put("B", bean);System.out.println(xml.writeValueAsString(map));} catch (JsonGenerationException e) {e.printStackTrace();} catch (JsonMappingException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

运行上面的方法,结果如下:

XmlMapper

china-Guangzhou
hoojo1hoojo@126.com
china-Guangzhou
hoojo1hoojo
@126.com
china-Guangzhou
hoojo1hoojo@126.com
china-Guangzhou
hoojo1hoojo
@126.com
china-Guangzhou
hoojo1hoojo_@126.com

看结果,根节点都是unknown 这个问题还没有解决,由于根节点没有转换出来,所有导致解析xml到Java对象,也无法完成。

Top

收藏 关注

评论 分类: JavaEE, JavaSE, Others, JSON&XML

标签: JavaEE, JavaSE, JSON, XML转换, JSON转换 绿色通道: 好文要顶 关注我 收藏该文与我联系

hoojo 关注 - 1 粉丝 - 475

+加关注

7

0 (请您对文章做出评价)

« 上一篇:JSON-lib框架,转换JSON、XML不再困难 » 下一篇:xStream完美转换XML、JSON

posted on 2011-04-22 10:45 hoojo 阅读(37975) 评论(16) 编辑 收藏

评论:

/#1楼 2011-04-22 13:29 | Dawnxu 这个很不错,可惜android下用不上..

支持(0)反对(0)

/#2楼[楼主] 2011-04-22 13:50 | hoojo @Dawnxu 怎么用不上,可以将转换后的字符串当做参数传递过去,这样在接收的地方再进行反序列化。应该是可以的!

支持(0)反对(0) http://pic.cnitblog.com/face/u151517.jpg

/#3楼 2011-12-27 09:31 | dearxiaofan 很不错的文章,我们android项目数据获取的框架用的就是这个。学习了。~

支持(0)反对(0)

/#4楼 2012-02-28 15:25 | 7-up 文章写得很详细,但我还是不太理解java对象要怎么设计 比如下面的一段json: {"error":0,"data":{"name":"ABC","age":20,"phone":{"home":"abc","mobile":"def"},"friends":[{"name":"DEF","phone":{"home":"hij","mobile":"klm"}},{"name":"GHI","phone":{"home":"nop","mobile":"qrs"}}]},"other":{"nickname":[]}} 我要怎样才能将其反序列? 再问一个问题对于一个未知的json有没有通用方法将其反序列,还是说一定要自己构造相应的java对象,才能实现反序列化呢?

支持(0)反对(0)

/#5楼[楼主] 2012-03-01 09:20 | hoojo @7-up String json = {"error":0,"data":{"name":"ABC","age":20,"phone":{"home":"abc","mobile":"def"},"friends":[{"name":"DEF","phone":{"home":"hij","mobile":"klm"}},{"name":"GHI","phone":{"home":"nop","mobile":"qrs"}}]},"other":{"nickname":[]}} Map> maps = objectMapper.readValue(json, Map.class); 这样就可以反序列化了,是一个map对象,map对象中存放的是LinkHashMap、LinkList对象,不一定要封装成Java对象。

支持(0)反对(0) http://pic.cnitblog.com/face/u151517.jpg
/#6楼 2012-03-03 15:21 | 7-up @hoojo 谢谢楼主帮忙,问题已解决

支持(0)反对(0)

/#7楼 2012-03-16 13:33 | fanjifeng 你好!楼主! 我看你的博文,但是对于Jackson对XML的支持其中需要使用stax2-api-3.0.2.jar包,我下载该包后,发现没有org.codehaus.jackson.xml.XmlMapper;,能否请楼主给个下载地址,或者发下包,我邮箱fjfzsm@sina.com

支持(0)反对(0)

/#8楼[楼主] 2012-03-20 09:52 | hoojo @fanjifeng 是否在jackson-all-1.7.6.jar这个包中

支持(0)反对(0) http://pic.cnitblog.com/face/u151517.jpg

/#9楼 2012-03-20 22:37 | fanjifeng 我查看了jackson-all-1.7.6.jar, jackson-all-1.9.5.jar 都没有org.codehaus.jackson.xml.XmlMapper类,而且没有org.codehaus.jackson.xml包路径。

支持(0)反对(0)

/#10楼[楼主] 2012-03-21 09:12 | hoojo @fanjifeng jackson-all-1.6.2.jar有这个类,新版本的可能去掉了

支持(0)反对(0) http://pic.cnitblog.com/face/u151517.jpg

/#11楼 2012-07-19 11:38 | 梦里六月的飞雪 这篇博客很不错,而且在很多地方也看到相同文章,我找到这篇文章,下载了,jackson-all- 1.7.6.jar 实现了,实现了,除转换为xml之外的其他所有功能,但是转换为xml一直不成功,很是 纠结,一直找不懂XmlMapper这个类,第一天下午没搞定,然后第二天上午还是在google上查资料, 终于 在 http://www.cowtowncoder.com/blog/archives/2012/03/entry_467.html 上面找到了些头 绪 ,jackson-dataformat-xml https://github.com/FasterXML/jackson-dataformat-xml ,在这 个页面上下载了 jackson-dataformat-xml-2.0.2.jar, 终于找到 XmlMapper 了,但是 放到 classpath下怎么也不能导入,后来才知道,他依赖的jar是 jackson-core-2.0.4.jar、jackson- databind-2.0.4.jar、jackson-annotations-2.0.4.jar ,然后 在加上 stax2-api-3.1.1.jar , 终于测试成功了,原来的 jackson-all-1.7.6.jar ,从classpath移除,加入新的包后,全部测试通 过,博主的案例将的很好,很详细,如果还解决不了 ,fmx35699@163.com ,我可以测试通过的源码 给大家,

支持(1)反对(0)

/#12楼 2012-07-19 12:36 | 梦里六月的飞雪 @fanjifeng 照我的解决方案试下,就可以了,jackson一直更新,而且更新很快,现在的最新2.04了,换下jar试试

支持(0)反对(0)

/#13楼 2012-08-26 22:38 | 余波可 @梦里六月的飞雪 我遇到和你一样的问题,XmlMapper这个类没有找到。其他的测试都通过了 。我按你的解决方案 “原来的 jackson-all-1.7.6.jar ,从classpath移除,加入新的包jackson-core-2.0.4.jar、jackson-databind-2.0.4.jar、jackson-annotations-2.0.4.jar后,stax2-api-3.1.1.jar 也加入了, 之前测试通过的反而测试失败了。特求一份源码。 我的邮箱:timyuheng@163.com 谢谢先

支持(0)反对(0)

/#14楼 2013-02-16 01:46 | kingdelee 博主你好哈~ 我自己尝试了一下用jackson2和1都没有成功,能否在您的博客中提供一份工程源码下载参考一下或者电邮给我呢,感激不尽啊!!! kingdelee@gmail.com

支持(0)反对(0)

/#15楼 2013-04-12 15:14 | 苏城 好文,学习了。。

支持(0)反对(0)

/#16楼27053592013/6/15 11:55:09 2013-06-15 11:55 | 章小慢 没有发现umknow 情况

支持(0)反对(0)

刷新评论刷新页面返回顶部

注册用户登录后才能发表评论,请 登录注册访问网站首页。 博客园首页博问新闻闪存程序员招聘知识库

最新IT新闻: · 消息称金庸将联合完美和畅游对大陆市场侵权游戏展开维权 · Google 主义 vs 苹果主义 · 移动开发者爱金钱,更爱成就感 · “厕所休闲网站”Backlabel · 蒂法竟然不是第一 细数玩家心中20位红粉知己 » 更多新闻...

最新知识库文章: · 我现在是这样编程的 · 如何做到 jQuery-free? · 如何成为一位优秀的创业CEO · 十年前的Java企业应用开发世界 · 专注做好一件事 » 更多知识库文章...

About Me

网名:hoojo E-mail:hoojo_@126.com 专注于Java,现从事电警卡口、智能交通、电子警察、数字城市等应用开发,擅长JavaEE、Flex、ActionScript及Web前端HTML、CSS、JavaScript、ExtJS、jQuery、Mootools等开发。对常用开源框架有一定的认识和见解。

版权声明

hoojo 所有文章遵循创作共用版权协议,要求署名、非商业、保持一致。在满足创作共用版权协议的基础上可以转载,但请以超链接形式注明出处。

访客统计

Powered By Clicki.cn

订阅统计

订阅我的博客

订阅文章

抓虾 google reader netvibes my yahoo bloglines 鲜果 哪吒 有道 QQ邮箱 订阅内容到手机

订阅历史

IT 新闻

+加关注

<2011年4月>日一二三四五六27282930311234567891011121314151617181920212223242526272829301234567

搜索

常用链接

最新随笔

我的标签

随笔分类(387)

随笔档案(139)

文章档案(3)

相册

关注博客

我的链接

积分与排名

  • 积分 - 189629
  • 排名 - 523

最新评论

阅读排行榜

评论排行榜

推荐排行榜

Powered by: 博客园 模板提供:沪江博客 Copyright ©2013 hoojo

希望本站内容对您有点用处,有什么疑问或建议请在后面留言评论
转载请注明作者(RobinChia)和出处 It so life ,请勿用于任何商业用途