您的位置:首页 > 数据库 > Redis

深浅克隆系列化与redis序列化

yulungggg 2019-07-06 14:38 253 查看 https://blog.csdn.net/yulunggg

深浅克隆系列化与redis序列化.........................................................................................1
1. 一、什么是对象克隆? ..............................................................................................2
2. 二、如何实现对象克隆..............................................................................................2
3. 三、什么是浅克隆和深克隆......................................................................................2
4. 总结: ..........................................................................................................................3
5. Redis序列化方式比较.................................................................................................3
5.1. 1. spring-data-redis支持的序列化策略...............................................................3


1. 一、什么是对象克隆?
首先我们需要知道,什么是对象的克隆,或者说复制。一个业务逻辑,需要一个新的对象,但是类型和值都是
之前的,也就是说,新状态和之前完全一样。使用new和赋值语句或者set注入都是可以的,但是,这会花费大
量开销去做,效率低,并且还会产生冗余代码。
恰好java语言本身契合了原型设计模式,给我们提供了一个clone方法在Object对象中,只要需要克隆的对象实现
Cloneable接口,那么我们只要简单的调用一下该方法就可以获得表面完全不同的对象。
2. 二、如何实现对象克隆
上文也是提到,克隆对象基本只要两步:
实现Cloneable接口
实现clone()方法,并调用父类clone()
需要注意,Object的clone()方法是在java平台层实现的native方法,具有开销小,速度快的特点。而且,原始的Obje
ct方法是被protected修饰的,在这里需要修改为public,如果不这么做,浅克隆时没有问题,深克隆就会遇到权限
不够的问题。java继承还有个原则,就是子类覆写父类方法,访问修饰符权限不能低于父类。。
3. 三、什么是浅克隆和深克隆
浅拷贝:仅仅克隆基本类型变量,而不克隆引用类型的变量
深克隆:既克隆基本类型变量,也克隆引用类型变量
其实本来原生的clone()方法也是,复制了引用的,但是为什么引用类型还是同一个呢?
就是因为java中,只有值传递,没有引用传递,引用在内存中还是值,是地址的hash值,一个特殊标记唯一堆内存
块的值。
4. 总结:
1.浅克隆:只复制基本类型的数据,引用类型的数据只复制了引用的地址,引用的对象并没有复制,在新的对
象中修改引用类型的数据会影响原对象中的引用。直接使用clone方法,再嵌套的还是浅克隆,因为有些引用
类型不能直接克隆。
2.深克隆:是在引用类型的类中也实现了clone,是clone的嵌套,并且在clone方法中又对没有clone方法的引用类
型又做差异化复制,克隆后的对象与原对象之间完全不会影响,但是内容完全相同。
3.使用序列化也能完成深复制的功能:对象序列化后写入流中,此时也就不存在引用什么的概念了,再从流中读
取,生成新的对象,新对象和原对象之间也是完全互不影响的。
5. Redis序列化方式比较
5.1. 1. spring-data-redis支持的序列化策略
spring-data-redis默认采用的序列化策略有两种,一种是String的序列化策略,一种是JDK的序列化策略。
参考博客 https://blog.csdn.net/pcwblover008/article/details/79915205
redisTemplate可自定义各种key和各种value的序列化方式:
defaultSerializer 默认序列化策略
key 普通key,非hash
value 普通value,非hash
hashKey hash的filed
hashValue hash的value
spring-data-redis的序列化类有下面这几个:
GenericToStringSerializer: 可以将任何对象泛化为字符串并序列化
Jackson2JsonRedisSerializer: 跟JacksonJsonRedisSerializer实际上是一样的
JacksonJsonRedisSerializer: 序列化object对象为json字符串
JdkSerializationRedisSerializer: 序列化java对象(被序列化的对象必须实现Serializable接口),无法转义成对象
StringRedisSerializer: 简单的字符串序列化
GenericToStringSerializer:类似StringRedisSerializer的字符串序列化
GenericJackson2JsonRedisSerializer:类似Jackson2JsonRedisSerializer,但使用时构造函数不用特定的类参考以
上序列化,自定义序列化类;
博客https://stackoverflow.com/questions/13215024/weird-redis-key-with-spring-data-jedis?answertab=votes#tabtop
中,使用了Jackson2JsonRedisSerializer, 需要在配置文件中, 配置一个类,用于反序列化. 这是致命缺陷!!
所以, 我们推荐使用GenericJackson2JsonRedisSerializer
StringRedisTemplate默认采用的是String的序列化策略,保存的key和value都是采用此策略序列化保存的。Strin
gRedisSerializer
RedisTemplate默认采用的是JDK的序列化策略,保存的key和value都是采用此策略序列化保存的。JdkSerializa
tionRedisSerializer
key和hashKey: 推荐使用 StringRedisSerializer: 简单的字符串序列化
hashValue: 推荐使用
GenericJackson2JsonRedisSerializer:类似Jackson2JsonRedisSerializer,但使用时构造函数不用特定的类

标签: