结合业务进行学习Gson排除策略,业务如下:
一个类有6个属性 ,用Gson进行序列化和反序列化,其中有1个属性需要排除。
一个类有6个属性 ,用Gson进行序列化和反序列化,其中有1个属性需要排除。
使用自定义之前,有必要了解一下@Expose注解
源码
package com.google.gson.annotations;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface Expose {
/**
* 是否序列化
*/
public boolean serialize() default true;
/**
* 是否反序列化
*/
public boolean deserialize() default true;
}
Expose注解有2个属性serialize
序列化(默认为true)deserialize
反序列化(默认为true)
业务实体类
public class ExposeUser {
public static int expStatic;//Gson默认情况会排除带有static关键字属性
@Expose
public String name;//姓名
@Expose
public int age;//年龄
@Expose
public String sex;//性别
@Expose
public int height;//身高
@Expose
public float weight;//体重
public boolean isLogin;//是否登录
@Override
public String toString() {
return "{\"expStatic\":\"" + expStatic
+ "\",\"name\":\"" + name
+ "\",\"age\":\"" + age
+ "\",\"sex\":\"" + sex
+ "\",\"height\":\"" + height
+ "\",\"weight\":\"" + weight
+ "\",\"isLogin\":\"" + isLogin + "\"}";
}
}
使用该注解时必须在构建Gson对象时调用excludeFieldsWithoutExposeAnnotation此方法生效,否则不会生效。
针对业务需求使用@Expose注解也能达到需求,但有个问题是,如果实体类包含多个属性(比如15个需要序列化的属性),那在代码上就会不优雅。
public static void main(String[] args) {
Gson gson = new GsonBuilder()
.excludeFieldsWithoutExposeAnnotation()//只序列化和反序列化带Expose注解属性
.create();
ExposeUser eu = new ExposeUser();
eu.name = "Gson";
eu.age = 22;
eu.sex = "男";
eu.height = 175;
eu.weight = 66.5f;
eu.isLogin = true;
String json = gson.toJson(eu);
System.out.println(json);//输出结果:{"name":"Gson","age":22,"sex":"男","height":175,"weight":66.5}
json = "{\"expStatic\":\"5\",\"name\":\"Gson\",\"age\":22,\"sex\":\"男\",\"height\":175,\"weight\":66.5,\"isLogin\":true}";
ExposeUser deserEu = gson.fromJson(json, ExposeUser.class);
System.out.println(deserEu.toString());//输出结果:{"expStatic":"0","name":"Gson","age":"22","sex":"男","height":"175","weight":"66.5","isLogin":"false"}
}
在上段代码中输出结果可以看出,带有static关键字的属性会自动排除,不进行序列化和反序列化。
自定义排除策略
基于相同业务,但是实体类会在注解上有变化
public class MExposeUser {
public static int expStatic;//Gson默认情况会排除带有static关键字属性
public String name;//姓名
public int age;//年龄
public String sex;//性别
public int height;//身高
public float weight;//体重
@MyExclus
public boolean isLogin;//是否登录
@Override
public String toString() {
return "{\"expStatic\":\"" + expStatic
+ "\",\"name\":\"" + name
+ "\",\"age\":\"" + age
+ "\",\"sex\":\"" + sex
+ "\",\"height\":\"" + height
+ "\",\"weight\":\"" + weight
+ "\",\"isLogin\":\"" + isLogin + "\"}";
}
}
Gson提供ExclusionStrategy接口,通过类名就知道作用
public interface ExclusionStrategy {
/**
* @param f the field object that is under test
* @return true if the field should be ignored; otherwise false
*/
public boolean shouldSkipField(FieldAttributes f);
/**
* @param clazz the class object that is under test
* @return true if the class should be ignored; otherwise false
*/
public boolean shouldSkipClass(Class<?> clazz);
}
先来看看如何实现的
/**
* 自定义排除策略
*/
public class MyExclusionStrategy implements ExclusionStrategy {
/**
* 需要跳过的属性
*
* @param f
* @return
*/
@Override
public boolean shouldSkipField(FieldAttributes f) {
//如果属性带有MyExclus 注解,则排除
return f.getAnnotation(MyExclus.class) != null;
}
/**
* 需要跳过的类
*
* @param clazz
* @return
*/
@Override
public boolean shouldSkipClass(Class<?> clazz) {
return false;
}
}
从业务逻辑来讲需要排除的是某个属性属性,所以实现shouldSkipField方法,方法中判断如果当前属性带有MyExclus注解则排除。如何需要排除类,则可以实现shouldSkipClass方法。
MyExclus注解是自定义的,在需要排除的属性上添加此注解
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface MyExclus {
}
万事具备,只欠东风,在构建Gson对象的时候,将自定义的排除策略调用setExclusionStrategies方法设置一下就可以了。
public static void main(String[] args) {
Gson gson = new GsonBuilder()
.setExclusionStrategies(new MyExclusionStrategy())
.create();
MExposeUser eu = new MExposeUser();
eu.name = "Gson";
eu.age = 22;
eu.sex = "男";
eu.height = 175;
eu.weight = 66.5f;
eu.isLogin = true;
String json = gson.toJson(eu);
System.out.println(json);
json = "{\"expStatic\":\"5\",\"name\":\"Gson\",\"age\":22,\"sex\":\"男\",\"height\":175,\"weight\":66.5,\"isLogin\":true}";
MExposeUser deserEu = gson.fromJson(json, MExposeUser.class);
System.out.println(deserEu.toString());
}
最后了解下shouldSkipClass方法如何实现
Gson gson = new GsonBuilder()
.addSerializationExclusionStrategy(new ExclusionStrategy() {
@Override
public boolean shouldSkipField(FieldAttributes f) {
return false;
}
@Override
public boolean shouldSkipClass(Class<?> clazz) {
// 排除指定的类
return Integer.class == clazz;
}
})
.create();
如果将身高声明为Integer的话,则会被排除在外
总结一下
希望通过写博客来提升自己的技术与语言组织水平,有什么写错的地方,还请大家多多反馈交流
以上2种解决方案可以根据业务的实际情况选择适合的方案
下载说明:
1、本站所有资源均从互联网上收集整理而来,仅供学习交流之用,因此不包含技术服务请大家谅解!
2、本站不提供任何实质性的付费和支付资源,所有需要积分下载的资源均为网站运营赞助费用或者线下劳务费用!
3、本站所有资源仅用于学习及研究使用,您必须在下载后的24小时内删除所下载资源,切勿用于商业用途,否则由此引发的法律纠纷及连带责任本站和发布者概不承担!
4、本站站内提供的所有可下载资源,本站保证未做任何负面改动(不包含修复bug和完善功能等正面优化或二次开发),但本站不保证资源的准确性、安全性和完整性,用户下载后自行斟酌,我们以交流学习为目的,并不是所有的源码都100%无错或无bug!如有链接无法下载、失效或广告,请联系客服处理!
5、本站资源除标明原创外均来自网络整理,版权归原作者或本站特约原创作者所有,如侵犯到您的合法权益,请立即告知本站,本站将及时予与删除并致以最深的歉意!
6、如果您也有好的资源或教程,您可以投稿发布,成功分享后有站币奖励和额外收入!
7、如果您喜欢该资源,请支持官方正版资源,以得到更好的正版服务!
8、请您认真阅读上述内容,注册本站用户或下载本站资源即您同意上述内容!
原文链接:https://www.dandroid.cn/6178,转载请注明出处。
1、本站所有资源均从互联网上收集整理而来,仅供学习交流之用,因此不包含技术服务请大家谅解!
2、本站不提供任何实质性的付费和支付资源,所有需要积分下载的资源均为网站运营赞助费用或者线下劳务费用!
3、本站所有资源仅用于学习及研究使用,您必须在下载后的24小时内删除所下载资源,切勿用于商业用途,否则由此引发的法律纠纷及连带责任本站和发布者概不承担!
4、本站站内提供的所有可下载资源,本站保证未做任何负面改动(不包含修复bug和完善功能等正面优化或二次开发),但本站不保证资源的准确性、安全性和完整性,用户下载后自行斟酌,我们以交流学习为目的,并不是所有的源码都100%无错或无bug!如有链接无法下载、失效或广告,请联系客服处理!
5、本站资源除标明原创外均来自网络整理,版权归原作者或本站特约原创作者所有,如侵犯到您的合法权益,请立即告知本站,本站将及时予与删除并致以最深的歉意!
6、如果您也有好的资源或教程,您可以投稿发布,成功分享后有站币奖励和额外收入!
7、如果您喜欢该资源,请支持官方正版资源,以得到更好的正版服务!
8、请您认真阅读上述内容,注册本站用户或下载本站资源即您同意上述内容!
原文链接:https://www.dandroid.cn/6178,转载请注明出处。
评论0