专栏首页10km的专栏fastjson:javabean按字段(field)序列化存储为Map并反序列化

fastjson:javabean按字段(field)序列化存储为Map并反序列化

大部分json工具对java对象整体序列化都提供了简单的调用方式,以fastjson为例:

Model model = new Model();
String json = JSON.toJSONString(model); // 序列化
Model model2 = JSON.parseObject(json, Model.class); // 反序列化

需求说明

最近的项目应用到redis数据库,需要将java bean存储在redis数据库。因为需要对数据库中的某个字段进行修改,所以在redis上不能用简单的string类型存储,而要以hash类型存储。这就需要在向数据库写入java bean对象之前要将java bean按字段序列化为一个Map<String,String> 而在从数据库读取后,又需要将所有字段合并反序列化还原成一个java bean对象。

序列化

实现步骤: 1.用JSON.toJSONString(Object)将java bean序列化为json String 2.用JSON.parseObject(String)将上一步的String反序列化为一个JSONObject(其实也是一个Map<String,Object>,JSONObject实现了Map接口). 3.对上一步中的JSONObject中每个字段调用JSON.toJSONString(Object)进行序列化,最终生成符合redis数据库hash类型存储要求的Map<String,String>

反序列化

实现步骤: 1.从redis获取所有字段数据,也就是一个Map<String,String>.对Map中每个字段的json string调用 JSON.parse(String)反序列化,生成一个Map<String,Object>,其中的Object就是字段反序列化生成的对象 2.调用 com.alibaba.fastjson.util.TypeUtils.cast(Object , Type , ParserConfig)方法将上一步的Map

实现代码及测试

package net.gdface.facelog.message;

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.junit.Test;

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.TypeReference;
import com.alibaba.fastjson.serializer.SerializerFeature;

public class TestFastjson {

    @Test
    public void testJSONObject() {
        Group group = new Group();
        group.setId(0L);
        group.setName("admin");

        User guestUser = new User();
        guestUser.setId(2L);
        guestUser.setName("guest");

        User rootUser = new User();
        rootUser.setId(3L);
        rootUser.setName("root");

        group.addUser(guestUser);
        group.addUser(rootUser);
        {
            ////////// 按字段序列化 ///////////////
            // 序列化步骤1
            String jsonstr = JSON.toJSONString(group);
            System.out.println(jsonstr);
            // 序列化步骤2
            Map<String, Object> jsonObj = JSON.parseObject(jsonstr);
            // 序列化步骤3
            for (Entry<String, Object> entry : jsonObj.entrySet()) {
                entry.setValue(JSON.toJSONString(entry.getValue()));
            }
            // 循环结束后 jsonObj 的数据类型为Map<String,String>
            ////////// 按字段反序列化 ///////////////
            Map<String, Object> deJsonObj = new HashMap<String, Object>();
            // 反序列化步骤1
            for (Entry<String, Object> entry : jsonObj.entrySet()) {
                String json = (String) entry.getValue();
                Object field = JSON.parse( json);
                deJsonObj.put(entry.getKey(), field);
            }
            // 循环结束时 Map<String, Object>中的每个字段对应的Object都是反序列化后的对象
            // 反序列化步骤2
            Group dgroup = com.alibaba.fastjson.util.TypeUtils.cast(deJsonMap, Group.class, null);
            System.out.println(JSON.toJSONString(dgroup));
        }
    }

    public interface Person<T>{

    }
    public static class User implements Person<String>{

        private Long   id;
        private String name;


        public Long getId() {
            return id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

    }
    public static  class Group {

        private Long       id;
        private String     name;
        private List<User> users = new ArrayList<User>();
        public Date date=new Date();
        public java.sql.Date sqldate=new java.sql.Date(date.getTime());
        public byte[] array=new byte[]{22,33,3,2,3,1,5,-1};
        public String nullStr=null;
        private String privString="private string";
        public Long getId() {
            return id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public List<User> getUsers() {
            return users;
        }

        public void setUsers(List<User> users) {
            this.users = users;
        }

            public void addUser(User user) {
                users.add(user);
            }
    }
}

总结

上面这个实现方式在序列化阶段有两次的序列化和一次反序列化,从性能角度看是不完美的。如果想实现一次序列化反序列化,需要对fastjson有更多的了解。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • fastjson:javabean按字段(field)序列化存储为Map并反序列化改进

    需求说明 最近的项目应用到redis数据库,需要将java bean存储在redis数据库。因为需要对数据库中的某个字段进行修改,所以在redis上不能用简单的...

    用户1148648
  • Redis数据类型之Hash哈希类型

    Hash的应用场景:(存储一个用户信息对象数据) 1、 常用于存储一个对象 2、 为什么不用string存储一个对象?

    兮动人
  • 博客——使用 Redis 实现博客编辑的自动保存草稿功能

    RedisServiceImpl 实现类(因为文章参数类继承了文章类,因此反射获取属性的时候需要获取父类属性):

    凡人飞
  • java高级特性:使用反射实现万能序列化1

    很多时候我们需要将一个类的实例变成二进制数据存储或是通过网络发送,这个过程叫序列化。如果将二进制数据解析成位于内存中的类实例或是相关数据结构,那叫反序列化。所有...

    望月从良
  • 从入门到精通,超强 RedisTemplate 方法详解!

    要使用 RedisTemplate,必须要先引入它,下面是它的「maven依赖」。

    业余草
  • 将List转化为Map的通用方法—泛型方法的简单运用

    有时候我们需要将List转化为Map,将数据散列存储,以提高查询效率。但是集合类中所存放的对象类型是不同的,因此,针对不同类型,我们常会写多个逻辑重复的转化方法...

    java达人
  • Redis应用场景

    Redis开创了一种新的数据存储思路,使用Redis,我们不用在面对功能单调的数据库时,把精力放在如何把大象放进冰箱这样的问题上,而是利用Redis灵活多变的数...

    张善友
  • 用了几年的 Fastjson,我最终替换成了Jackson!

    作者:larva-zhh 来源:www.cnblogs.com/larva-zhh/p/11544317.html

    Java技术栈
  • Hessian 反序列化及相关利用链

    前不久有一个关于Apache Dubbo Http反序列化的漏洞,本来是一个正常功能(通过正常调用抓包即可验证确实是正常功能而不是非预期的Post),通过Po...

    Seebug漏洞平台
  • Redis笔记(二):Redis数据类型

    Redis支持五种数据类型:string(字符串),hash(哈希),list(列表),set(集合)及zset(sorted set:有序集合)。

    朝雨忆轻尘
  • 高效的序列化/反序列化数据方式 Protobuf

    上篇文章中其实已经讲过了 encode 的过程,这篇文章以 golang 为例,从代码实现的层面讲讲序列化和反序列化的过程。

    一缕殇流化隐半边冰霜
  • 值得收藏!Redis五大数据类型应用场景(一)

    Redis开创了一种新的数据存储思路,使用Redis,我们不用在面对功能单调的数据库时,把精力放在如何把大象放进冰箱这样的问题上,而是利用Redis灵活多变的数...

    java思维导图
  • SparkRDD转DataSet/DataFrame的一个深坑

    原需求:希望在map函数中将每一个rdd转为DataSet或者DataFrame。

    王知无-import_bigdata
  • SparkRDD转DataSet/DataFrame的一个深坑

    原需求:希望在map函数中将每一个rdd转为DataSet或者DataFrame。

    大数据真好玩
  • 除了FastJson,你还有选择: Gson简易指南

    这个周末被几个技术博主的同一篇公众号文章 fastjson又被发现漏洞,这次危害可导致服务瘫痪! 刷屏,离之前的漏洞事件没多久,FastJson 又出现严重 B...

    用户4172423
  • 除了FastJson,你还有选择: Gson简易指南

    前几天被几个技术博主的同一篇公众号文章 fastjson又被发现漏洞,这次危害可导致服务瘫痪! 刷屏,离之前漏洞事件没多久,fastjson 又出现严重 Bu...

    闻人的技术博客
  • HiveQL快速使用

    --define可以定义用户变量 --hivevar可以定义用户遍历 --hiveconf使用key-value得到hive-site.xml配值的变量

    黑白格
  • ToString数据如何反序列化

    不知道小伙伴们有没有这样的困扰,平常开发中写单测,要mock一个复杂的对象,并且也知道了该对象的toString格式数据(比如从日志中获取),但是该怎么构建这个...

    luoxn28
  • 数据脱敏——基于Java自定义注解实现日志字段脱敏

    版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/...

    DannyHoo

扫码关注云+社区

领取腾讯云代金券