(二)JPA 连接工厂、主键生成策略、DDL自动更新
2、JPA连接工厂
通过之前的 代码 实现已经清楚的发现了整个的JPA实现步骤,但是这个步骤似乎有一些繁琐了,毕竟最终所关心的一定是EntityManager对象实例,而要想获取到此对象的实例,那么要经过许多的步骤,这样如果每一次都重复的进行处理,会非常的繁琐了,那么就需要进行代码的抽象规定。
// 创建JPA Entity工厂
EntityManagerFactory factory =
Persistence.createEntityManagerFactory("YootkJPA");
// JPA操作对象
EntityManager entityManager = factory.createEntityManager();
对等概念: DataSource -> EntityMannagerFactory
Connection -> EntityMannager,每一个对象的实例都表示一个Session的操作
所以此时可以考虑将部分的代码移交给JPA的专属连接管理类,用这个类可以基于ThreadLocal实现EntityManager存储,这样每一次通过该类的方法获取EntityManager的时候如果不关闭,则获取到的是同一个实例。
对于上叙测试类的优化。
- 一个工具类,简化EntityManagerFactory的创建与关闭
public class JPAEntityFactory {
/**
* JPA持久单元
*/
private static final String PERSISTENCE_UNIT = "YootkJPA";
/**
* 等同于 数据源
*/
private static EntityManagerFactory entityManagerFactory;
/**
* EntityManager 等同于 连接
*/
private static ThreadLocal<EntityManager> entityManagerThreadLocal =
new ThreadLocal<>();
static {
// 初始化创建数据源 静态代码块只会创建一次
rebuildEntityManagerFactory();
}
/**
* 获取连接
*/
public static EntityManager getEntityManager() {
// 本机线程获取连接
EntityManager entityManager = entityManagerThreadLocal.get();
if (entityManager == null) {
// 从数据源获取新的连接
entityManager = getEntityManagerFactory().createEntityManager();
// 存入本地线程
entityManagerThreadLocal.set(entityManager);
}
return entityManager;
}
private static EntityManagerFactory getEntityManagerFactory() {
if (entityManagerFactory == null) {
// 创建数据源
rebuildEntityManagerFactory();
}
return entityManagerFactory;
}
private static void rebuildEntityManagerFactory() {
// 持久单元 创建 数据源
entityManagerFactory = Persistence.createEntityManagerFactory(PERSISTENCE_UNIT);
}
/**
* 关闭EntityManager
*/
public static void close() {
// 获取实例
EntityManager entityManager = entityManagerThreadLocal.get();
if (entityManager != null) {
// 关闭连接 entityManager类似于Connection
entityManager.close();
// 删除本地线程中的连接
entityManagerThreadLocal.remove();
}
}
}
修改测试类:
@Test
public void testAdd2() {
// 获取连接
EntityManager entityManager = JPAEntityFactory.getEntityManager();
// 开启事务
entityManager.getTransaction().begin();
Course course = new Course();
course.setCname("Spring编程实战");
// 字符串 日期对象
course.setStart(DateUtil.stringToDate("2022-09-19"));
course.setEnd(DateUtil.stringToDate("2022-12-30"));
course.setCredit(2);
course.setNum(88);
// 执行插入
entityManager.persist(course);
// 提交事务
entityManager.getTransaction().commit();
// 关闭连接
JPAEntityFactory.close();
}
3、主键生成
在JPA开发之中,主键数据的生成主要是基于
@Id注解定义的,而在实际的项目开发之中,数据表的设计结构是有所不同的,所以JPA为了适应这些不同的数据表的定义,也提供有不同的主键生成策略。

3、DDL自动更新
在实际的开发之中你是否会出现这样的一种比较 尴尬 的问题,在进行开发的时候有人修改数据表,而后当前的实体类结构和数据表的结构不统一,但是在JPA设计的时候,充分的考虑到了这种数据表修改的问题(表可能存在,也可能不存在,或者表的结构可能修改了),所以在这样的环境下就需要让代码可以自动的进行数据表的纠正。
在传统的项目开发之中,常规的做法是先进行数据表的创建,而后在围绕数据表进行业务功能的实现。在每次业务发生改变时,也是先进行表结构的修改,而后再进行程序的变更,这样的数据库维护是非常繁琐的,考虑到数据库更新以及 数据库移植 方面的设计,在 Hibernate 之中提供了 DDL 自动创建以及表更新策略

JPA现在主要是基于 Hibernate 实现,那么 Hibernate 开发框架最早的一个特点就在于 可移植性,也就是说一个项目是在MySQL数据库下开发的,那么通过简单的配置修改,可以让代码直接在Oracle数据库中运行。
对于当前市面上可以见到的ORM开发框架来讲,只有JPA标准规定了数据库移植性的话题,而Hibernate 实现了JPA标准,所以只有Hibernate 开发框架具有移植性的功能,而像大家所熟悉的MyBatis是没有这样的功能。
DDL更新策略

3.1、使用
去到JPA配置文件中,修改DDL更新策略、
3.1.1、create
每次加载时,根据实体类生成表,如果表存在于数据库,会先删除
<!-- JPA更新策略 -->
<property name="hibernate.hbm2ddl.auto" value="create"/>

查看执行日志信:可以看到 drop table if exists course (删除表,然后会依据实体类,重新创建表)
3.1.2、update
如果表不存在,重建表。
存在:如:实体类某个字段,在数据表中不存在,这个时候会添加。但是,删除实体类的某个字段,数据库对应的字段并不会删除。
修改实体类: 添加教师

修改配置文件-更新策略为update

查看执行日志:
表不存在,创建表
create table course (
cid bigint not null auto_increment,
cname varchar(255),
credit integer,
end date,
num integer,
start date,
teacher varchar(255),
primary key (cid)
) engine=InnoDB
排除属性
假如,我们相使实体类中的某个字段,在执行时,不创建数据库中的对应字段。使用@Transient即可

(二)JPA 连接工厂、主键生成策略、DDL自动更新的更多相关文章
- (二)JPA实体类主键生成策略
在JPA中,配置实体类的主键的生成策略使用 @GeneratedValue @Id @Column(name = "id") @GeneratedValue(strategy = ...
- JPA中的主键生成策略
通过annotation(注解)来映射hibernate实体的,基于annotation的hibernate主键标识为@Id, 其生成规则由@GeneratedValue设定的.这里的@id和@Gen ...
- 160727、自定义hibernate主键生成策略生成字符串+数字自增长
需求:需要自增长注解如MyId0001.MyId0002.MyId0003 实现:实现这个接口org.hibernate.id.IdentifierGenerator 一.MyIdGenerator. ...
- hibernate的持久化类、主键生成策略
一.hibernate的持久化类 1.什么是持久化类: 持久化:将数据存储到关系型数据库. 持久化类:与数据库中的数据表建立了某种关系的java类.(持久化类=javabean+映射配置文件) 2.持 ...
- 三大框架 之 Hibernate生成策略与缓存策略(主键生成策略、持久化、持久化类划分、一级缓存、事物管理)
目录 Hibernate生成策略与缓存策略 主键生成策略 主键分类 主键的生成策略 持久化 什么是持久化 什么是持久化类 持久化类编写规则 持久化类的划分 三种状态区分 持久态对象特征 一级缓存 什么 ...
- 4、JPA table主键生成策略(在JPA中table策略是首推!!!)
用 table 来生成主键详解 它是在不影响性能情况下,通用性最强的 JPA 主键生成器.这种方法生成主键的策略可以适用于任何数据库,不必担心不同数据库不兼容造成的问题. initialValue不起 ...
- jpa table主键生成策略
用 table 来生成主键详解 它是在不影响性能情况下,通用性最强的 JPA 主键生成器.这种方法生成主键的策略可以适用于任何数据库,不必担心不同数据库不兼容造成的问题. initialValue不起 ...
- jpa基于按annotation的hibernate主键生成策略
JPA注解持久化类很方便,需要jar包:ejb3-persistence.jar下载 一.JPA通用策略生成器 通过annotation来映射hibernate实体的,基于annotation的hib ...
- 转数据库分库分表(sharding)系列(二) 全局主键生成策略
本文将主要介绍一些常见的全局主键生成策略,然后重点介绍flickr使用的一种非常优秀的全局主键生成方案.关于分库分表(sharding)的拆分策略和实施细则,请参考该系列的前一篇文章:数据库分库分表( ...
随机推荐
- 实战模拟│JWT 登录认证
目录 Token 认证流程 Token 认证优点 JWT 结构 JWT 基本使用 实战:使用 JWT 登录认证 Token 认证流程 作为目前最流行的跨域认证解决方案,JWT(JSON Web Tok ...
- MySQL语句与正则表达式
正则表达式的作用是匹配文本,将一个模式(正则表达式)与一个文本串进行比较.MySQL用WHERE子句对正则表达式提供了初步的支持,允许你指定正则表达式,过滤SELECT检索出的数据. 1.元字符说明 ...
- Solution -「校内题」Xorequ
0x00 前置芝士 数位dp考试里出现的小神题?? 显然考场会选择打表找规律. 数位dp + 矩阵快速幂 0x01 题目描述 给定正整数 \(n\),现有如下方程 \(x \bigoplus 3x = ...
- RabbitMQ细说之开篇
前言 关于消息中间件的应用场景,小伙伴们应该都耳熟能详了吧,比如经常提到的削峰填谷.分布式事务.异步业务处理.大数据分析等等,分布式消息队列成为其中比较关键的桥梁,也就意味着小伙伴们得掌握相关技能:当 ...
- 关于python文件打包成exe的调试问题
python文件使用pyinstaller打包的问题 常用pyinstaller相关命令 文件整体打包, 会自动打包相关依赖 pyinstaller -F file 分文件打包,只打包单个文件,其他文 ...
- 2535-springsecurity系列--关于授权角色“ROLE”前缀的问题
版本信息 <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring ...
- GIt后悔药:还原提交的操作(谨慎操作)
一.背景: 偶尔会遇到git的版本分支的文件被误改的情况,需要还原,此篇文章可能会帮助到你. PS: 来理解下 Git 工作区.暂存区和版本库概念,可以更好的理解以下的还原操作. * 工作区:就是你在 ...
- 记一次Linux server偶发CPU飙升问题的跟进与解决
背景 进入6月后,随着一个主要功能版本api的上线,服务端的QPS翻了一倍,平时服务器的CPU使用稳定在30%上下,高峰期则在60%上下,但是偶尔会有单台机器出现持续数分钟突然飙到90%以上,导致大量 ...
- Python3.7爬虫:实时api(百度ai)检测验证码模拟登录(Selenium)页面
原文转载自「刘悦的技术博客」https://v3u.cn/a_id_134 今天有同学提出了一个需求,老板让自动登录这个页面:https://www.dianxiaomi.com/index.htm, ...
- Spring提供的API实现文件上传
Spring为我们提供了文件上传接口MultipartRequest及其实现类StandardMultipartFile StandardMultipartFile是StandardMultipart ...