AutoTable, Hibernate自动建立表替代方案

news/2024/10/19 0:22:37/

痛点

之前一直使用JPA为主要ORM技术栈,主要是因为Mybatis没有实体逆向建表功能。虽然Mybatis有从数据库建立实体,但是实际应用却没那么美好:当实体变更时,往往不会单独再建立一个数据库重新生成表,然后把表再逆向为实体。最终的结果往往是维护一份数据库SQL,再同时维护一份实体对象,两者没有自动建立关联。

方案

如果能够自动建立表,并自动维护系统初始的数据,该有多方便啊。

笔者实际的经验,十年前已经实现Hibernate自动建表+DBUnit自动初始数据(包括图片和相对数据,可见笔者其它文章)。

然而世界在发展,痛点终究有大牛出来解决,在Mybatis领域,最近出现了一个的替代解决方案:MybatisPlusExt,简称MPE。其中的自动建表已被MPE作者单独一个项目处理,叫做AutoTableAuto Table)自动维护表结构icon-default.png?t=N7T8https://autotable.tangzc.com/

迁移步骤

配置文件

autotable也有springboot starter。重新建表的逻辑,也有JAP类似的参数,因此,很容易可以进行迁移,改动点如下:

JPA

spring:jpa:database-platform: ${app.dataSource.hibernateDialect}generate-ddl: falseshow-sql: falseopen-in-view: falseproperties:hibernate.jdbc.time_zone: ${app.timeZone:GMT+8}

以及config里的配置Bean:

	@Value("${app.init.mode:none}")private String initMode;@Bean@ConfigurationProperties(prefix = "spring.datasource")public JpaVendorAdapter jpaVendorAdapter() {return new HibernateJpaVendorAdapter();}@Beanpublic LocalContainerEntityManagerFactoryBean entityManagerFactory(DataSource dataSource, JpaVendorAdapter jpaVendorAdapter){LocalContainerEntityManagerFactoryBean bean=new LocalContainerEntityManagerFactoryBean();      bean.setDataSource(dataSource);bean.setPackagesToScan(new String[] {"org.ccframe.subsys.*.domain.entity"});bean.setJpaVendorAdapter(jpaVendorAdapter);bean.getJpaPropertyMap().put("hibernate.hbm2ddl.auto",initMode);return bean;}

AutoTable

先导入starter依赖:


implementation ("com.tangzc:auto-table-spring-boot-starter:1.7.4") // 自动建表

然后书写格式:

auto-table:show-banner: falsemode: ${app.init.mode}model-package: org.ccframe.subsys.*.domain.entityindex-prefix: IDX_

Entity实体

以一个典型的带普通和Unique索引的实体User为例:

JPA

JPA+Hibernate方案

@Entity
@Table(name = "SYS_USER", indexes = {@Index(columnList = "USER_MOBILE"),@Index(columnList = "USER_EMAIL"),
}, uniqueConstraints = {@UniqueConstraint(columnNames = {"PLATFORM_ID","LOGIN_ID","USER_PSW"}),
})
@Getter
@Setter
@ToString
public class User extends BaseEntity{private static final long serialVersionUID = 6662916002685367792L;public static final String USER_ID = "userId";public static final String PLATFORM_ID = "platformId";public static final String LOGIN_ID = "loginId";public static final String USER_HEAD_PICT_ID = "userHeadPictId";public static final String USER_NAME = "userName";public static final String USER_PSW = "userPsw";public static final String USER_MOBILE = "userMobile";public static final String USER_EMAIL = "userEmail";public static final String USER_STATUS_CODE = "userStatusCode";public static final String IF_ADMIN = "ifAdmin";public static final String ROLE_CODE_STR = "roleCodeStr";@Id@GenericGenerator(name = "userId", strategy = "org.ccframe.commons.base.RedisIDGenerator")@GeneratedValue(generator = "userId")@Column(name = "USER_ID", nullable = false, length = 10)private Integer userId;@Column(name = "PLATFORM_ID", nullable = false, length = 10)private Integer platformId;@Column(name = "LOGIN_ID", nullable = false, length = 38)@Field(type = FieldType.Keyword)private String loginId;@Column(name = "USER_HEAD_PICT_ID", nullable = true, length = 10)private java.lang.Integer userHeadPictId;@Column(name = "USER_NAME", nullable = false, length = 32)private String userName;@Column(name = "USER_PSW", nullable = false, length = 128)private String userPsw;@Column(name = "USER_MOBILE", nullable = true, length = 17)private String userMobile;@Column(name = "USER_EMAIL", nullable = true, length = 70)private String userEmail;@Column(name = "IF_ADMIN", nullable = false, length = 2)@Field(type = FieldType.Keyword)private String ifAdmin;@Column(name = "USER_STATUS_CODE", nullable = false, length = 2)private String userStatusCode;@Column(name = "ROLE_CODE_STR", nullable = false, length = 80)private String roleCodeStr;
}

AutoTable

mybatis-plus + autotable的方案

@TableName("SYS_USER")
@AutoTable("SYS_USER")
@TableIndex(name = "UK66q7srks5eylhocxej5gs68mb", type= IndexTypeEnum.UNIQUE, fields = {"tenantId","loginId","userPsw"})
@TableIndex(name = "IDXbby41q9neesp2i6hatmlud01b", fields = "userMobile")
@TableIndex(name = "IDXhjkdbn8wxvwcdp7ohh7dch6i1", fields = "userEmail")
@Getter
@Setter
@ToString
public class User extends BaseEntity{private static final long serialVersionUID = 6662916002685367792L;public static final String USER_ID = "userId";public static final String TENANT_ID = "tenantId";public static final String LOGIN_ID = "loginId";public static final String USER_AVATAR = "userAvatar";public static final String USER_NAME = "userName";public static final String USER_PSW = "userPsw";public static final String USER_MOBILE = "userMobile";public static final String USER_EMAIL = "userEmail";public static final String USER_STATUS_CODE = "userStatusCode";public static final String IF_ADMIN = "ifAdmin";public static final String ROLE_CODE_STR = "roleCodeStr";@TableId(type = IdType.ASSIGN_ID)@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 120) @ColumnNotNullprivate Long userId;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 38) @ColumnNotNull@Field(type = FieldType.Keyword)private String loginId;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 48) @ColumnNotNullprivate String userAvatar;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 32) @ColumnNotNullprivate String userName;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 128) @ColumnNotNullprivate String userPsw;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 17)private String userMobile;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 70)private String userEmail;@Field(type = FieldType.Keyword)@ColumnType(value = MysqlTypeConstant.CHAR, length = 1)private String ifAdmin;@ColumnType(value = MysqlTypeConstant.CHAR, length = 1)private String userStatusCode;@ColumnType(value = MysqlTypeConstant.VARCHAR, length = 80)private String roleCodeStr;}

总结

AutoTable能够很好的兼容JPA的格式,实现自动建表的迁移。

但是有几个注意点:

1)索引需要进行命名,hibernate的是采用自动前缀+25位字符来自动实现索引的命名,我们不用去关心索引的名称。而迁移到AutoTable需要去起个不重复的名字。这个问题不大

2)hibernate采用方言的模式,可以兼容大部分数据库。而AutoTable的字段类型,需要指定数据库类型,如果要切换数据库,需要做实体代码定义的改动


http://www.ppmy.cn/news/1454085.html

相关文章

ffmpeg 转文件为h265

查看 nvidia-smi ---------------------------------------------------------------------------------------- | NVIDIA-SMI 551.61 Driver Version: 551.61 CUDA Version: 12.4 | |-------------------------------------------------------------------------------------…

【Docker学习】docker run的端口映射-p和-P选项

docker run的端口映射选项分为-p(小写,全称--publish),-P(大写,全称--publish-all),之前认为只有改变容器发布给宿主机的默认端口号才会进行-p的设置,而不改变默认端口号…

对命令模式的理解

目录 一、场景1、文本编辑器并不是一个好的例子,设备控制器才是2、设备控制器的demo 二、不用命令模式1、代码2、问题 三、使用命令模式1、代码2、当需求变化时2.1 新增代码2.2 优点 四、进一步思考1、省略对Command的建模可以吗?2、命令模式的价值 一、…

http 一定是基于TCP连接的吗?

写在前面 最近收到一个同学的留言,问题就是 HTTP是不是基于TCP的? 先说结论: HTTP 1.0、1.1、2.0 版本是基于TCP的。 HTTP 3.0 是基于UDP的。 很多人这么一看,呦?这不是常见的八股文吗? HTTP 1.0、1.1、2…

Spring框架的扩展点

Spring框架是一个非常流行的Java应用程序框架,它提供了一系列的扩展点,使得开发者可以自定义和增强框架的功能。这些扩展点包括: Bean生命周期回调 Spring允许通过实现特定的接口或使用注解来在Bean的生命周期的不同阶段(如初始…

13_Scala面向对象编程_伴生对象

文章目录 1.伴生对象1.1 scala的一个性质,scala文件中的类都是公共的;1.2 scala使用object关键字也可以声明对象; 3.关于伴生对象和类4.权限修饰符,scala仅有private;5.伴生对象可以访问伴生类中的私有属性;6.案例7.伴…

程序设计——前后端分离实现简单表白墙

文章目录 一、前端页面样式代码二、前后端衔接1. 后端创建 maven 项目2. 针对前后端交互的解释以及后端代码的实现针对 post 请求解释前后端衔接针对 Get 请求解释前后端衔接 3.后端与数据库的联系以及对数据的存取单独封装数据库连接代码解释后端存储 save 数据的代码解释后端…

项目经理【人】原则

系列文章目录 【引论一】项目管理的意义 【引论二】项目管理的逻辑 【环境】概述 【环境】原则 【环境】任务 【环境】绩效 【人】概述 【人】原则 一、共创模式 1.1 共创模式 二、干系人的影响力强度和态度 2.1 干系人影响力 2.2 干系人态度 2.3 干系人管理 三、干系人权力…