简介:

在使用 Gson 对象json字符串进行烦序列化的时候,发现了一个问题,有一个json字符串对应的枚举字段是小写,但是对象的定义却是全大写,这导致,有一个字段无法反序列化,映射不了字段,为空,在网上一段找发现都没有好的解决方法接下来看看我给出的解决方案吧。
问题:
需要反序列化的json 字符串:

`{\"id\":\"1\",\"name\":\"测试\",\"method\":\"WeChat\"}`

反序列化的对象实体定义:

public class UserDemo {

    private String id;

    private String name;

    private Method method;

}

Method :

public enum Method {

    _JSAPI_("JSAPI"),
    _WECHAT_("WeChat");

    private final String value;
}

转换的代码:

`String data = "{\"id\":\"1\",\"name\":\"测试\",\"method\":\"WeChat\"}";
UserDemo userDemo = new Gson().fromJson(data, UserDemo.class);
System._out_.println(userDemo);`

转换对象之后发现method属性为空,无法映射
``UserDemo(id=1, name=测试, method=null)

解决方案1:

在转换的时候,我们定义一个mode实体类,这个实体类定义与需要转换的json字符串大小写保持一致,然后在进行对象转换。
Method2:`

public enum Method2 {

    _JSAPI_("JSAPI"),
    _WeChat_("WeChat");

    private final String value;

}`

这样子转换就可以直接映射字段数据进去。因为我们在开发中,都会有一层用来接受参数然后对象转换的,所以这个问题可以友好解决。

解决方案2

有可能来说我们不想定义一个对象,并且定义的modl对象属性是有定义的不一致的,这就很麻烦,从某种意义来说,这是不允许的,你可以想想,你接口定义的字段和model定义的字段属性是一样的,这样子从长远来说是不友好的,所以我的解决方案是,我先转对象,然后在针对某写字段在进去单独的处理。

`public static  UserDemo toUserDemo(String body){
    UserDemo userDemo = new Gson().fromJson(body, UserDemo.class);
    if(Objects._isNull_(userDemo.getMethod())){
        Map<String,Object> map = new Gson().fromJson(body, Map.class);
        String method = map.get("method").toString();
        userDemo.setMethod(Method._getMethod_(method));
    }
    return  userDemo;
}`

枚举类中需要定义一个,根据value获取的方法:

`public static Method getMethod(String value) {
    for (Method confMethod : Method._values_()) {
        if (confMethod.value.equals(value)) {
            return confMethod;
        }
    }
    throw new IllegalArgumentException("Unexpected value '" + value + "'");
}`

解决方案3

使用 SerializedName 注解 属性重命名来实现字段转换实现数据映射

`@Documented
@Retention(RetentionPolicy._RUNTIME_)
@Target({ElementType._FIELD_, ElementType._METHOD_})
public @interface SerializedName {

  _/**
   * _**_@return _**_the desired name of the field when it is serialized or deserialized
   */
  _String value();
  _/**
   * _**_@return _**_the alternative names of the field when it is deserialized
   */
  _String[] alternate() default {};
}`

value 需要重命名的字段
alternate : 那些字段是对应重命名的字段

具体详细详情请看注释。
接下来我们看看这么去使用他:

`@AllArgsConstructor
@ToString
public enum Method {

    _JSAPI_("JSAPI"),
    _/**
     * value 需要重命名的字段
     * alternate 是一个数组,那些字段重命名
     */
    _@SerializedName(value = "WECHAT", alternate = {"WeChat","wechat"})
    _WECHAT_("WeChat");

    private final String value;
    
}`

这样子使用后,直接通过gson转对象就可以映射数据了:

`UserDemo userDemo = new Gson().fromJson(data, UserDemo.class);`

结尾

到这里问题就算完美解决了,不过,最好的还是定义一直,不要出现大小写的情况,要么全大写,要么全小写
这样子对于的维护更加安全,简单。