使用jackson对java类中包含泛型属性的bean进行序列化和反序列化

最近在做项目时,需要对java的bean对象进行序列化转换为string字符串后,存入redis,然后再从redis中读取该string对象,反序列化为bean对象。正常的简单类型的对象使用jackson可以非常方便进行互为转换操作,但我们操作的对象有点复杂,造成存入redis的数据是对的,但反序列化时,一直没有正常将值反序列化成功,主要是由于类属性有一个Pair<double,double>的对象,该对象是一个abstract class对象,有相应的子类生成对应的实例对象。bean对象类似于下面的定义:

BoardcastRsp类中有一个Action的类,定义类似于如下:

action类中的geometry对象的定义如下所示:

对于Pair<Double,Double>类型,直接进行writeValueAsString和readValue则,从string中转换后的BoardcastRsp中的定义为Pair<Double,Double>类型的相关属性均无法正常反序列化。这种场景中,只能针对Pair类型的进行自定义的序列化和反序列化操作。相关的操作核心代码如下:

//自定义序列化方法
class PairSerializer extends JsonSerializer<Pair> {
    @Override
    public void serialize(Pair pair, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
        jsonGenerator.writeStartArray();
        jsonGenerator.writeNumber((Double) pair.getKey());
        jsonGenerator.writeNumber((Double) pair.getValue());
        jsonGenerator.writeEndArray();
    }
}

//自定义反序列化方法
class PairDeserializer extends JsonDeserializer<Pair> {
    @Override
    public Pair deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {
        ObjectCodec codec = jsonParser.getCodec();
        JsonNode node = codec.readTree(jsonParser);
        double key = node.get(0).asDouble();
        double value = node.get(1).asDouble();
        return Pair.of(key, value);
    }
}
public class TestSDClass {
    public static void main(String[] args) throws IOException {
        Action action = new Action();
        action.setGeometry(new Polyline("1", "#dddd", 1,
                MutablePair.of(0.0, 0.0), 222l,
                MutablePair.of(0.0, 0.0), 2, "L22323",
                MutablePair.of(0.0, 0.0),
                MutablePair.of(0.0, 0.0),
                "L1111"));
        action.setOp(Action.Operate.DRAW);

        BoardcastRsp boardcastRsp = new BoardcastRsp();
        boardcastRsp.setAction(action);
        boardcastRsp.setMediaId("1");
        boardcastRsp.setLevel(Level.FIVE);
        boardcastRsp.setCategory(BoardcastRsp.Category.ACTION);

        // 创建 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建自定义模块
        SimpleModule module = new SimpleModule();
        module.addSerializer(Pair.class, new PairSerializer());
        module.addDeserializer(Pair.class, new PairDeserializer());
        objectMapper.registerModule(module);

        try {
            // 序列化为 JSON 字符串
            String jsonString = objectMapper.writeValueAsString(boardcastRsp);
            System.out.println(jsonString);

            // 反序列化为 Java 对象
            BoardcastRsp myClass2 = objectMapper.readValue(jsonString, BoardcastRsp.class);
            System.out.println(myClass2.getLevel());
        } catch (JsonProcessingException e) {
            e.printStackTrace();
        }
    }
}

通过自定义的序列化和反序列化即可以将Pair<Double,Double>泛型相关的属性进行正常的操作了。上面的代码中,序列化时,将泛型对象设置为数组。反序列化时,从数组中取到两个double值,然后生成一个新的Pair对象,完成反序列化操作。

  • 19
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 中的 `Map` 属性泛型 `T` 都可以通过序列化转换为字节序列,以便在网络上传输或将其存储在磁盘上。 对于 `Map` 属性序列化,我们可以在包含属性类中实现 `Serializable` 接口,并将 `Map` 属性标记为 `transient`,以便在序列化对象时跳过该属性。然后,我们可以在 `writeObject()` 和 `readObject()` 方法中手动序列化反序列化属性。 以下是一个示例代码,展示了如何序列化一个包含 `Map` 属性的类: ```java import java.io.*; import java.util.*; public class MyClass implements Serializable { private transient Map<String, Integer> map; private String name; public MyClass() { map = new HashMap<>(); map.put("A", 1); map.put("B", 2); map.put("C", 3); name = "MyClass"; } private void writeObject(ObjectOutputStream out) throws IOException { out.defaultWriteObject(); out.writeInt(map.size()); for (Map.Entry<String, Integer> entry: map.entrySet()) { out.writeObject(entry.getKey()); out.writeObject(entry.getValue()); } } private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { in.defaultReadObject(); int size = in.readInt(); map = new HashMap<>(size); for (int i = 0; i < size; i++) { String key = (String) in.readObject(); Integer value = (Integer) in.readObject(); map.put(key, value); } } @Override public String toString() { return "MyClass{name='" + name + "', map=" + map + "}"; } public static void main(String[] args) { MyClass myClass = new MyClass(); System.out.println("Original object: " + myClass); try { // 序列化 FileOutputStream fos = new FileOutputStream("myclass.ser"); ObjectOutputStream oos = new ObjectOutputStream(fos); oos.writeObject(myClass); oos.close(); // 反序列化 FileInputStream fis = new FileInputStream("myclass.ser"); ObjectInputStream ois = new ObjectInputStream(fis); MyClass newMyClass = (MyClass) ois.readObject(); ois.close(); // 输出反序列化后的对象 System.out.println("Deserialized object: " + newMyClass); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } } } ``` 在上述代码中,`Map` 属性被标记为 `transient`,并在 `writeObject()` 和 `readObject()` 方法中手动序列化反序列化序列化时,我们首先调用 `defaultWriteObject()` 方法将对象的默认字段序列化,然后手动序列化 `Map` 属性反序列化时,我们首先调用 `defaultReadObject()` 方法读取对象的默认字段,然后手动反序列化 `Map` 属性。 对于泛型 `T` 的序列化,我们可以在序列化反序列化使用 Java 的类型擦除机制。具体来说,在序列化时,我们可以将 `T` 视为 `Object`,并将其强制转换为 `Object` 类型;在反序列化时,我们可以将反序列化后的 `Object` 对象强制转换为 `T` 类型。 以下是一个示例代码,展示了如何序列化一个包含泛型 `T` 属性的类: ```java import java.io.*; public class MyClass<T> implements Serializable { private T data; public MyClass(T data) { this.data = data; } public T getData() { return data; } @Override public String toString() { return "MyClass{data=" + data + "}"; } public static void main(String[] args) { MyClass<String> myClass = new MyClass<>("Hello, world!"); System.out.println("Original object: " + myClass); try { // 序列化 FileOutputStream fos = new FileOutputStream("myclass.ser"); ObjectOutputStream oos = new ObjectOutputStream(fos); oos.writeObject(myClass); oos.close(); // 反序列化 FileInputStream fis = new FileInputStream("myclass.ser"); ObjectInputStream ois = new ObjectInputStream(fis); MyClass<?> newMyClass = (MyClass<?>) ois.readObject(); ois.close(); // 输出反序列化后的对象 System.out.println("Deserialized object: " + newMyClass); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } } } ``` 在上述代码中,我们创建了一个泛型 `T` 类型为 `String` 的 `MyClass` 对象,并将其序列化反序列化。在序列化时,我们将泛型 `T` 视为 `Object`,并将其强制转换为 `Object` 类型;在反序列化时,我们将反序列化后的 `Object` 对象强制转换为 `T` 类型。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值