androidgson解析json
⑴ Android Gson 使用详解
Json 是一种文本形式的数据交换格式,比 xml 更为轻量。Json 的解析和生成的方式很多,在 Android 平台上最常用的类库有 Gson 和 FastJson 两种,这里要介绍的是 Gson
Gson 的 GitHub 主页点击这里: Gson
在进行序列化与反序列操作前,需要先实例化一个 com .google.gson.Gson 对象,获取 Gson 对象的方法有两种
利用 Gson 可以很方便地生成 Json 字符串,通过使用 addProperty 的四个重载方法
addProperty 方法底层调用的是 add(String property, JsonElement value) 方法,即将基本数据类型转化为了 JsonElement 对象,JsonElement 是一个抽象类,而 JsonObject 继承了 JsonElement ,因此我们可以通过 JsonObject 自己来构建一个 JsonElement
Json数组 与 字符串数组
Json数组 与 List
Gson 也提供了 toJson() 和 fromJson() 两个方法用于转化 Model 与 Json,前者实现了序列化,后者实现了反序列化
首先,声明一个 User 类
序列化的方法很简单,调用 gson 对象的 toJson 方法,传入要序列化的对象
反序化的方式也类似
继续使用上一节声明的 User 类,根据 User 类声明的各个属性名,移动端的开发者希望接口返回的数据格式即是如下这样的
如果没有和服务器端沟通好或者是 API 改版了,接口返回的数据格式可能是这样的
如果继续使用上一节介绍的方法,那无疑会解析出错
例如
name 属性值解析不到,所以为 null
此时为了兼顾多种格式的数据,就需要使用 SerializedName 注解
根据 SerializedName 的声明来看,SerializedName 包含两个属性值,一个是字符串,一个是字符串数组,而字符串数组含有默认值
SerializedName 的作用是为了在序列化或反序列化时,指导 Gson 如果将原有的属性名和其它特殊情况下的属性名联系起来
例如,修改 User 类,为 name 声明 SerializedName 注解,注解值为 userName
在序列时,Json 格式就会相应改变
在反序列化时也一样,能够解析到正确的属性值
还有个问题没解决,为了应对多种属性名不一致的情况,难道我们要声明多个 User 类吗?这显然是不现实的,所以还需要为 User 类设置多个备选属性名,这就需要用到 SerializedName 注解的另一个属性值 alternate 了。
以下几种情况都能够被正确的反序列化
有时候并不是所有的字段都需要进行系列化和反序列化,因此需要对某些字段进行排除,有四种方法可以来实现这种需求。
Expose 注解包含两个属性值,且均声明了默认值。Expose 的含义即为“暴露”,即用于对外暴露字段,serialize 用于指定是否进行序列化,deserialize 用于指定是否进行反序列化。如果字段不声明 Expose 注解,则意味着不进行序列化和反序列化操作,相当于两个属性值均为 false 。此外,Expose 注解需要和 GsonBuilder 构建的 Gson 对象一起使用才能生效。
Expose 注解的注解值声明情况有四种
现在来看个例子,修改 User 类
按照如上的注解值,只有声明了 Expose 注解且 serialize 值为 true 的字段才能被序列化,只有声明了 Expose 注解且 deserialize 值为 true 的字段才能被反序列化
Gson 提供了 @Since 和 @Until 两个注解基于版本对字段进行过滤,@Since 和 @Until 都包含一个 Double 属性值,用于设置版本号。Since 的意思是“自……开始”,Until 的意思是“到……为止”,一样要和 GsonBuilder 配合使用。
当版本( GsonBuilder 设置的版本) 大于或等于 Since 属性值或小于 Until 属性值时字段会进行序列化和反序列化操作,而没有声明注解的字段都会加入序列化和反序列操作
现在来看个例子,修改 User 类
访问修饰符由 java.lang.reflect.Modifier 提供 int 类型的定义,而 GsonBuilder 对象的 excludeFieldsWithModifiers 方法接收一个 int 类型可变参数,指定不进行序列化和反序列化操作的访问修饰符字段
看个例子
GsonBuilder 类包含 setExclusionStrategies(ExclusionStrategy... strategies) 方法用于传入不定长参数的策略方法,用于直接排除指定字段名或者指定字段类型
看个例子
字段名为 "intField" 和字段类型为 double 的字段都会被排除掉
setExclusionStrategies 方法在序列化和反序列化时都会生效,如果只是想指定其中一种情况下的排除策略或分别指定排除策略,可以改为使用以下两个方法
对于 Gson 而言,在序列化时如果某个属性值为 null 的话,那么在序列化时该字段不会参与进来,如果想要显示输出该字段的话,可以通过 GsonBuilder 进行配置
默认的序列化后的 Josn 字符串并不太直观,可以选择格式化输出
Gson 也可以对时间值进行格式化
TypeAdapter 是一个泛型抽象类,用于接管某种类型的序列化和反序列化过程,包含两个抽象方法,分别用于自定义序列化和反序列化过程
下面看个简单的例子
定义 TypeAdapter 的子类 UserTypeAdapter 来接管 User 类的序列化和反序列化过程
这里设定当 User 类序列化时 Json 中的Key值都是大写字母开头,反序列化时支持“name”和“Name”两种不同的 Json 风格
可以看到 User 类按照预定义的策略来完成序列化和反序列化了
TypeAdapter 将序列化和反序列操作都接管了过来,其实 Gson 还提供了只接管序列化过程的接口,即 JsonSerializer
看个例子
相对应的,JsonDeserializer 接口提供了反序列化的接口
这里有个比较麻烦的地方,那就是在使用 TypeAdapter 、JsonSerializer 和 JsonDeserializer 时,总需要调用 registerTypeAdapter 方法进行注册,那有没有更简单的注册方法呢?
有的,Gosn 还提供了另一个注解 @JsonAdapter 用于进行简单的声明
类似于这样,声明了 User 类的序列化或反序列化操作由 UserTypeAdapter 完成,注解的优先级高于 registerTypeAdapter 方法
TypeAdapterFactory 是用于创建 TypeAdapter 的工厂类,通过参数 TypeToken 来查找确定对应的 TypeAdapter,如果没有就返回 null 并由 Gson 默认的处理方法来进行序列化和反序列化操作,否则就由用户预定义的 TypeAdapter 来进行处理
这一篇文章好像写得太长了一点?Gson 的知识点介绍到这里也差不多了,以后如果还发现新内容的话我会继续补充,现在就先这样啦
⑵ android中json解析后的数据怎样显示在手机上
android中json解析有很多种方式,解析完,直接给TextView赋值即可
以下为android中常见的JSON解析类库
1. android SDK:androidSDK中自带的JSONObject
2. gson google专门为androd开发的json解析框架
3.jackson java EE Spring中的json转换工具
4.json-lib: JSON-lib框架,转换JSON、XML
在Java EE中使用最广泛的是Jackson Json,使用非常方便而且效率极高;另外还有两个比较有名气的是Gson和JSON-lib,某些大侠已经对它们做了对比实验,结论如下执行效率:Jackson>Gson>JSON-lib
⑶ android使用gson解析嵌套复杂的json数据,数据怎么显示到布局上,布局怎么写
首先先讲一个比较简单点的例子(最简单的我就不讲啦,网上很多),帮助新手理解Gson的使用方法:
比如我们要解析一个下面这种的Json:
String json = {"a":"100","b":[{"b1":"b_value1","b2":"b_value2"},{"b1":"b_value1","b2":"b_value2"}],"c":{"c1":"c_value1","c2":"c_value2"}}
首先我们需要定义一个序列化的Bean,这里采用内部类的形式,看起来会比较清晰一些:
public class JsonBean {
public String a;
public List<B> b;
public C c;
public static class B {
public String b1;
public String b2;
}
public static class C {
public String c1;
public String c2;
}
}
很多时候大家都是不知道这个Bean是该怎么定义,这里面需要注意几点:
1、内部嵌套的类必须是static的,要不然解析会出错;
2、类里面的属性名必须跟Json字段里面的Key是一模一样的;
3、内部嵌套的用[]括起来的部分是一个List,所以定义为 public List<B> b,而只用{}嵌套的就定义为 public C c,
具体的大家对照Json字符串看看就明白了,不明白的我们可以互相交流,本人也是开发新手!
Gson gson = new Gson();
java.lang.reflect.Type type = new TypeToken<JsonBean>() {}.getType();
JsonBean jsonBean = gson.fromJson(json, type);
然后想拿数据就很简单啦,直接在jsonBean里面取就可以了!
如果需要解析的Json嵌套了很多层,同样可以可以定义一个嵌套很多层内部类的Bean,需要细心的对照Json字段来定义哦。
⑷ android开发用gson解析json字符串,如果键不是固定值怎么办
这样没试过,不过我认为可以将json中键值不固定的存在JSONArray中访问时奇数位保存键,偶数保存值;也可以只传值;还有就是多层嵌套(例如:假设字母表示键,数字表示值:a:2,b:5替换成i:a,ii:2,iii:b,iiii:5这意思就是将键与值都作为值传过来)。要么每一次变化都写一个类来与之对应,不同的时候用不同的类的实例接收
⑸ Android studio使用Retrofit框架,Get发送请求,Gson解析返回的json数据时报错怎么办
数据库一直以来给我的感觉就是——麻烦!!!
接触了Realm之后才终于可以开开心心的使用数据库了。
本文总结一些Realm数据库的常用知识点,包括多线程访问,以及如何与Retrofit2.0一起使用等...
看懂这些知识点之后,个人认为就可以在一般的项目中使用Realm了。
1. model类必须extends RealmObject,所有属性必须用private修饰
2. model中支持基本数据结构:boolean, byte, short, ìnt, long, float, double, String, Dateand byte[]
3.若要使用List必须用RealmList<T>,或者继承RealmList
4.与Retrofit2.*一起使用,通过Gson来解析Json数据并直接生成RealmObject,可参考如下写法:
[java] view plain
Gson gson = new GsonBuilder()
.setExclusionStrategies(new ExclusionStrategy() {
@Override
public boolean shouldSkipField(FieldAttributes f) {
return f.getDeclaringClass().equals(RealmObject.class);
}
@Override
public boolean shouldSkipClass(Class<?> clazz) {
return false;
}
⑹ android gson可以只解析json中想要的属性吗
当然可以,按照格式只解析你需要的;其它可以不管。
⑺ android中gson怎么解析json数组
首先要导入Gson包;
例如有个penson类:
Gson gson = new Gson();
String json ;
person[] pers = gson. fromJson (json , person[].class);
⑻ Android 如何引用本地json文件
有时候遇到要一些模拟数据的时候,要么写个List<T>,要么写死.
然后数据比较多的时候可以引用本地资源文件,记录下如何使用本地json文件
<1>在java同级目录下创建"assets"资源文件夹
<2>编辑json文本内容
至于json文件夹内容 就自定义了,想要什么内容写什么,对于一个做项目多了的开发者而言 json数据是熟悉到不能再熟悉的了
以下是我在淘宝一个个复制过来自己拼装的一些商品数据
不需要加任何标识,只需要标准的json数据格式就可以了
<3>引用和json解析的工具类,解析用的 Gson也是经常用到的
<4>最后在需要的地方引用,"ExchangeBean"是根据json内容写的一个模型,用于接收存放数据
有时间再补上demo,下面是效果图:
⑼ gson怎么解析jsonarray
Json是一种类似于XML的通用数据交换格式,具有比XML更高的传输效率.
从结构上看,所有的数据(data)最终都可以分解成三种类型:
第一种类型是标量(scalar),也就是一个单独的字符串(string)或数字(numbers),比如"北京"这个单独的词。
第二种类型是序列(sequence),也就是若干个相关的数据按照一定顺序并列在一起,又叫做数组(array)或列表(List),比如"北京,上海"。
第三种类型是映射(mapping),也就是一个名/值对(Name/value),即数据有一个名称,还有一个与之相对应的值,这又称作散列(hash)或字典(dictionary),比如"首都:北京"。
Json的规格非常简单,只用一个页面几百个字就能说清楚,而且Douglas Crockford声称这个规格永远不必升级,因为该规定的都规定了。
1) 并列的数据之间用逗号(",")分隔。
2) 映射用冒号(":")表示。
3) 并列数据的集合(数组)用方括号("[]")表示。
4) 映射的集合(对象)用大括号("{}")表示。
在Android中可以使用Gson解析JSON数据
首先,从 code.google.com/p/google-gson/downloads/list下载GsonAPI:
google-gson-1.7.1-release.zip
把gson-1.7.jar 到libs(项目根目录新建一个libs文件夹)中。
可以使用以下两种方法解析JSON数据:
通过获取JsonReader对象解析JSON数据:
⑽ android中gson解析怎样防止参数确实
最近在做天气预报的项目,需要用到多重JSON的解析,这里我选用了目前流行的,也公认是最简单易用的框架GSON来解析这个复杂的JSON,虽然在最后结果很简单,但并不意味着整个过程很简单,没有一点问题,恰恰是相反的,我是第一次学习这个GSON,虽然在网上看博客一看即知怎么操作,但是一动手,却发生问题还是有的。
首先,先上我的JSON结构:
{"desc":"OK","status":1000,"data":{"wen":"15","ganmao":"昼夜温差很大,易发生感冒,请注意适当增减衣服,加强自我防护避免感冒。","forecast":[{"fengxiang":"南风","fengli":"微风级","high":"高温16℃","type":"晴","low":"低温2℃","date":"14日星期二"},{"fengxiang":"南风","fengli":"微风级","high":"高温16℃","type":"多云","low":"低温4℃","date":"15日星期三"},{"fengxiang":"南风","fengli":"微风级","high":"高温17℃","type":"多云","low":"低温7℃","date":"16日星期四"},{"fengxiang":"南风","fengli":"微风级","high":"高温16℃","type":"阴","low":"低温5℃","date":"17日星期五"},{"fengxiang":"南风","fengli":"微风级","high":"高温19℃","type":"晴","low":"低温5℃","date":"18日星期六"}],"yesterday":{"fl":"微风","fx":"南风","high":"高温14℃","type":"晴","low":"低温2℃","date":"13日星期一"},"aqi":"38","city":"北京"}}
在实现时,我是有进行过分层的,这里,我就忽略了……
首先,照着网上一个简单的教程,我把JAVABEAN,设置成这样:
{CSDN:CODE:public class TestBean {
public String desc;
public int status;
public List<Data> data;
public class Data {
public String wen;
public String ganmao;
public List<Forecast> forecast;
public List<Yesterday> yesterday;
public String aqi;
public String city;
public class Forecast {
public String fengxiang;
public String fengli;
public String high;
public String type;
public String low;
public String date;}
public class Yesterday {
public String fl;
public String fx;
public String high;
public String type;
public String low;
public String date;}
}
}}
这里有两个点想说的,正确的是,一是成员变量名一定是跟JSON的字段完全一样的,那个没有“”的数字数据的话要用int,我一开始照着网上模仿这个,把内层的数据都用一个List<内部类名称>来表示。结果是失败的。
接着,我把代码又拆成了网上分多个类出来,每个类都加多一个getter和setter,我本着大脑想,或许上个想得过于简单,可能系统还需要重写toString方法,而且这些都是有规矩的。
如下图:
做到这里,我还是想验证所有我不太清楚的点,第一,我改了类名,比如把内部类的名称改了,这里证明是不会有影响的,但是字段名是一定要一样的。第二,我又建了一个和开始一样的,只有极简的结构类,尝试如果没有 setter和getter有没有影响,结果是没影响的,toString也是可有可无的。
总之,只要保证对JSON结果的正确解构,还有分析对象,是单个类,还是一个集合类。
最后,学习的过程中,要尽量地去尝试,不断地思考,不要嫌简单,遇到问题,不要想着去依赖别人帮自己解决,问题千变万化,自己还得停下来想想。
感谢阅读,如果不妥的地方,请指出。