spring data jpaHibernate开启全球唯一UUID设置

Posted 曲高终和寡

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring data jpaHibernate开启全球唯一UUID设置相关的知识,希望对你有一定的参考价值。

 

spring data jpa、Hibernate开启全球唯一UUID设置

原文链接:https://www.cnblogs.com/blog5277/p/10662079.html

原文作者:博客园--曲高终和寡

*******************如果你看到这一行,说明爬虫在本人还没有发布完成的时候就抓走了我的文章,导致内容不完整,请去上述的原文链接查看原文****************

 

UUID这个东西,具体的我就不讲了,网上的内容很多,我这里就随便放一个大家看一下就了解了

https://www.jianshu.com/p/d77f3ef0868a

这里面有关于发号器,UUID的介绍,具体各位是使用UUID的哪个版本,或者使用雪花ID,或者使用其他别的方式,请各位自行判断

 

在我的业务里面呢:

1.准备部署至少两台服务器,且两台服务器时区不一样,一台阿里云香港,东八区,一台美国洛杉矶,多少区忘了,反正有13个小时的时差,

——雪花ID的方案就被我否了,因为我不能保证美国那台服务器的时区不会根据冬夏令时来回跳,雪花ID在时钟回拨的情况下会一直等候,我也不需要雪花ID有序这个优点。

2.同样为了容灾,我准备在两台服务器分别部署数据库

——主键自增的方案就被我否了,迁移有点麻烦了

3.我不需要主键来排序,我选用的PostgresQL也不会因为主键是字符型的而导致插入、读取变快

——所以我不需要一切主键需要有序的生成方案,UUID完美符合我的需求

4.既然选用了UUID,我就希望它在任何情况下都不会重复,全球唯一

——那么UUID的生成方案2.3.4.5就都不满足我的要求了,虽然概率极地,但是还是有理论上重复的可能性的

5.UUID的方案1存在一定的危险性,是根据MAC地址(或者IP)生成的,那么理论上能根据MAC地址找到所在地

——。。。然后呢。。。?我一台服务器/IP在香港,一台服务器/IP在洛杉矶,想通过这俩地方来找我怕是不太可能嗷。。。更何况我写的东西根本就不会有多少人用,更没有什么法律风险不担心被查水表,更没有什么价值。。。?谁会来搞我?

 

所以,最终选用了方案1的UUID生成方法,不可能重复,我开发的时候就不用考虑这种极小概率会导致的BUG,放心大胆的用就是了。

而生成方案1的UUID有很多工具包,比如Apache common 的UUID 包,还有github上一个项目 JAU,项目地址:

https://github.com/cowtowncoder/java-uuid-generator

我太懒了,不想在项目里引入一个包,就为了生成UUID。。。所以就查了一下,原来HIbernate对生成UUID方案1还做了优化:

Version1变种 – Hibernate

Hibernate的CustomVersionOneStrategy.java,解决了之前version 1的两个问题

- 时间戳(6bytes, 48bit):毫秒级别的,从1970年算起,能撑8925年….
- 顺序号(2bytes, 16bit, 最大值65535): 没有时间戳过了一秒要归零的事,各搞各的,short溢出到了负数就归0。
- 机器标识(4bytes 32bit): 拿localHost的IP地址,IPV4呢正好4个byte,但如果是IPV6要16个bytes,就只拿前4个byte。
- 进程标识(4bytes 32bit): 用当前时间戳右移8位再取整数应付,不信两条线程会同时启动。

值得留意就是,机器进程和进程标识组成的64bit Long几乎不变,只变动另一个Long就够了。

Hibernate解决了Version1的两个问题,哪两个问题呢:

但好像Version 1就没考虑过一台机器上起了两个进程这类的问题,也没考虑相同时间戳的并发问题

很好,HIbernate解决了这俩问题,那完美符合我的需求,并且本来项目就引入了JPA,官方支持岂不美哉?

废话说了那么多,配置只需要两行的事,然而这个配置国内网站几乎找不到,困扰了我很久怎么在Hibernate里面设置选用CustomVersionOneStrategy.java,作为主键生成策略的

/**
 * @Author:ShadowSaint
 * @Date:19-4-3 上午4:56
 * @Description: TODO
 */
@MappedSuperclass
@Data
@Inheritance(strategy = InheritanceType.TABLE_PER_CLASS)
public class BaseDomain {
    @Id
    @GenericGenerator(
            name = "uuid2",
            strategy = "uuid2",
            parameters = {@org.hibernate.annotations.Parameter(
                    name = "uuid_gen_strategy_class",
                    value = "org.hibernate.id.uuid.CustomVersionOneStrategy")
            })
    @GeneratedValue(generator = "uuid2")
    private String id;
    @Column
    private LocalDateTime gmtCreate;
    @Column
    private LocalDateTime gmtModified;

    public void initInsert() {
        this.gmtCreate = LocalDateTime.now();
        this.gmtModified = LocalDateTime.now();
    }

    public void initUpdate() {
        this.gmtModified = LocalDateTime.now();
    }
}

 

 

这样就可以了,以后每一个DO都继承这个BaseDomain,ID就自动生成Hibernate优化版的版本1UUID了

 

以上是关于spring data jpaHibernate开启全球唯一UUID设置的主要内容,如果未能解决你的问题,请参考以下文章

在spring boot / spring data elasticsearch中禁用自动索引创建

没有 Spring Boot 的 Spring Security JWT 示例

java.lang.IllegalStateException:无法加载 ApplicationContext Spring Boot + JUnit 测试

Spring data jpa-未定义名为“entityManagerFactory”的bean;注入自动装配的依赖项失败

jpa基本常识

如何在 Spring 生态系统中使用 JPA hibernate 连接到不同的 mysql 数据库模式?