百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

系列总结:JPA核心接口综合实战案例

ccwgpt 2024-09-21 13:43 27 浏览 0 评论

JPA入门精要系列:

(1):JPA框架组件核心组成

(2):EntityManager概要及API一览表

(3):EntityManager API功能详解(二)

(4):EntityManager API功能详解(三)

本文由牛旦教育原创出品,欢迎收藏、点赞、分享。

转载请注明来源@牛旦教育IT课堂

0.导引

作为JPA系列文章的小结,本文针对EntityManager核心接口,在此给出部分相关核心方法的具体的应用示例。本示例假设在同一个持久化单元配置下完成,且在非EE服务器模式下进行(EE模式下,我会在专栏中详细介绍,敬请期待~)



1. JPA环境简介

根据前面的介绍,我简洁起见,我直接在Eclipse 2019企业版IDE中创建JPA项目,然后引入必要的jar包。项目环境为:

JDK8、JPA2.1(当前JPA版本2.2,相对2.1变动不大;另需引入Java EE8标准包javaee-api-8.0.jar)以及MySQL8的驱动mysql-connector-java-8.0.15.jar。

注意:mysql8与mysql5的驱动器和连接字符串不同,若不注意区分可能导致连接失败。

以上环境你可以通过Maven工程来配置实现,这里给出依赖部分配置参考,如下所示:

<properties>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
        <endorsed.dir>${project.build.directory}/endorsed</endorsed.dir>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <failOnMissingWebXml>false</failOnMissingWebXml>
        <jakartaee>8.0</jakartaee>
    </properties>
    
    <dependencies>
        <dependency>
            <groupId>javax</groupId>
            <artifactId>javaee-api</artifactId>
            <version>${jakartaee}</version>
            <scope>provided</scope>
        </dependency>
        <dependency>
            <groupId>org.eclipse.persistence</groupId>
            <artifactId>eclipselink</artifactId>
            <version>2.7.5</version>
            <scope>compile</scope>
        </dependency>        
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.15</version>
        </dependency>
……
    </dependencies>

JPA项目的核心持久化配置文件persistence.xml如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.1" xmlns="http://xmlns.jcp.org/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence http://xmlns.jcp.org/xml/ns/persistence/persistence_2_1.xsd">
    <persistence-unit name="default" transaction-type="RESOURCE_LOCAL" >
        <class>com.newdayedu.jpa.demo.entity.Book</class>    
       
        <properties>
            <property name="javax.persistence.jdbc.driver" value="com.mysql.cj.jdbc.Driver"/>
            <property name="javax.persistence.jdbc.url"
             value="jdbc:mysql://localhost:3306/mytest?characterEncoding=utf8&useSSL=false&serverTimezone=UTC&rewriteBatchedStatements=true&allowPublicKeyRetrieval=true"/>
            <property name="javax.persistence.jdbc.user" value="mytest"/>
            <property name="javax.persistence.jdbc.password" value="mytest"/>
            <property name="eclipselink.jdbc.exclusive-connection.mode" value="Transactional"/>
        </properties>
    </persistence-unit>
</persistence>

1.2 获取EntityManagerFactory实例

获取实体管理器工厂(EntityManagerFactory),是一项代价很重的操作,所以,建议要要把对应的实体管理器工厂换成下来,以便复用。通常可以考虑在应用程序启动时初始化一次,然后伴随应用程序的整个生命周期而重复使用。特别注意:实体管理器工厂(EntityManagerFactory)是线程安全的。通常我们在非EE环境下,创建JPA帮助器类,通过如下代码获取工厂的实例:

public class JPAHelper {
    private static  EntityManagerFactory emf = null ;
    static {     
       emf = Persistence.createEntityManagerFactory("default");
        }
//或者这样获取指定的实体工厂实例
public static final EntityManagerFactory getEMFactory(String PUName) {
       if(PUName==null||"".equals(PUName.trim())) {
           return emf ;
       }
       return Persistence.createEntityManagerFactory(PUName);     
        }
}
 
 

在EE环境下,通过注入方式实现,类似如下所示(这里暂不展开)。

@PersistenceContext(unitName = "default")
    private EntityManager em;
 
    protected EntityManager getEntityManager() {
        return em;
    }


1.3 获取EntityManager实体

可在JPAHelper类中,提供简单的方法,用如下方式获取JPA的实体管理器:

public static EntityManager getDefaultEntityManager() {
       return emf.createEntityManager() ;
}

但是实体管理器不是线程的,若需要确保实体管理器的线程安全,尤其在Servlet环境中注入是要注意。这里可以借助ThreadLocal类,以如下方式实现增强线程安全性(下面的代码依然在JPAHelper中):

private static final ThreadLocal<EntityManager> _threadLocal = new ThreadLocal<>();
   
    public static EntityManager getEntityManager() {
       EntityManager em = _threadLocal.get();
       if(em==null) {
           em = emf.createEntityManager() ;
           _threadLocal.set(em);
       }
       return em ;
    }

1.4 用JPA操作数据库之持久化

这里我们创建一个Book类实体,以便对其实例进行持久化,实体类:

@Entity
@Table(name="books")
@NamedQuery(name="Book.findAll", query="SELECT b FROM Book b")
public class Book implements Serializable {
    private static final long serialVersionUID = 1L;
 
    @Id
    private int bkid;
 
    private String bkcomm;
 
    private String bkname;
 
    private BigDecimal bkprice;
 
    @Temporal(TemporalType.TIMESTAMP)
    private Date bkpubdate;
 
    private String bkpublisher;
 
    private String bktype;
 
    private String bkwriter;
 
    public Book() {
    }
//getter/setter略
}

注意,通过注解进行标准,并对应数据库的books表。

实现数据的持久化,代码如下所示:

public static void main(String[] args) {
       EntityManager em = null;
       Book book = new Book();
       // 为book赋值,以便进行有效持久化
       book.setBkid(101);
       book.setBkname("大秦帝国");
       book.setBktype("历史类");
       book.setBkprice(new BigDecimal(345.5));
       book.setBkpubdate(new Date());
       book.setBkwriter("孙皓辉");
       book.setBkpublisher("华文出版社");
       try {
           em = getEntityManager();
           em.getTransaction().begin();
           em.persist(book);
           em.getTransaction().commit();
           Logger.getLogger(JPAHelper.class.getName()).log(Level.INFO, "数据持久化成功!");
       } catch (Exception ex) {
           ex.printStackTrace();
       } finally {
           close(em);//关闭实体管理器
       }
      
    }

1.5 JPA的命名查询

这个示例,主要来介绍基于实体类的命名查询的实现。

所谓命名查询,就是在实体类上的这行注解,用来查找所有书籍信息:

@NamedQuery(name="Book.findAll", query="SELECT b FROM Book b")

示例中,我用实体管理器来创建命名查询实例,并返回结果。待如下所示:

public static void main(String[] args) {
       EntityManager em = null;
       em = getEntityManager();
       @SuppressWarnings("unchecked")
       List<Book> bkList = em.createNamedQuery("Book.findAll").getResultList();
       for(Book bk:bkList) {
           Logger.getLogger("JPAHelper.main()").log(Level.INFO, "书籍信息:"+bk.toString());
       }
}

运行上面的代码,会输出所有查到的书籍信息。

另外,我在实体上只编写了一个命名查询语句。如果要田间多个怎么办呢?你可以把所有的命名查询用下面注释进行封装,如下所示(直接替换原来命名查询注释即可):

@NamedQueries({
   @NamedQuery(name = "Book.findAll", query = "SELECT b FROM Book b"),
   @NamedQuery(name = "Book.findBookById", query = "SELECT b FROM Book b WHERE b.bkid = :bkid"),
  @NamedQuery(name = "Book.findBookByName", query = "SELECT b FROM Book b WHERE b.bkname = :bkname")})

或再在原来NamedQuery注释后并列的增加需要的命名注释也可。

关于命名查询的说明:在命名查询上使用的两个元数据注解@NamedQueries和@NamedQuery很容易理解,记住前者可以包含任意需要的NameQuery子注解。而@NamedQuery两个核心内容就是名查询的名子以及对应的JPQL语句。JPQL是JPA化的SQL语句,但不能直接写出原生的SQL语句,如要进行原生的SQL查询,实体管理器有另外的接口API支持(请参照EntityManager API功能详解部分内容)。相关内容我会在专栏里进行详解。

如果要根据ID查找唯一的一本书,可以按下面语句来实现:

Book bk = em.find(Book.class,4) ;
System.out.println("按ID查找到的数据:"+bk.toString());

好了。关于JPA的基本操作,就实战的这里,还有更多的内容,我会在专栏中完整的、详细的介绍,请本关注头条号以及专栏。

继续加油,努力提供更优的“多、快、好、省”的IT相关技术学习内容!

跟牛旦学IT技术,其实编程没有那么难,关键是如何进入这扇门。 收藏一下,分享出去吧^_^

相关推荐

团队管理“布阵术”:3招让你的团队战斗力爆表!

为何古代军队能够以一当十?为何现代企业有的团队高效似“特种部队”,有的却松散若“游击队”?**答案正隐匿于“布阵术”之中!**今时今日,让我们从古代兵法里萃取3个核心要义,助您塑造一支战斗力爆棚的...

知情人士回应字节大模型团队架构调整

【知情人士回应字节大模型团队架构调整】财联社2月21日电,针对原谷歌DeepMind副总裁吴永辉加入字节跳动后引发的团队调整问题,知情人士回应称:吴永辉博士主要负责AI基础研究探索工作,偏基础研究;A...

豆包大模型团队开源RLHF框架,训练吞吐量最高提升20倍

强化学习(RL)对大模型复杂推理能力提升有关键作用,但其复杂的计算流程对训练和部署也带来了巨大挑战。近日,字节跳动豆包大模型团队与香港大学联合提出HybridFlow。这是一个灵活高效的RL/RL...

创业团队如何设计股权架构及分配(创业团队如何设计股权架构及分配方案)

创业团队的股权架构设计,决定了公司在随后发展中呈现出的股权布局。如果最初的股权架构就存在先天不足,公司就很难顺利、稳定地成长起来。因此,创业之初,对股权设计应慎之又慎,避免留下巨大隐患和风险。两个人如...

消息称吴永辉入职后引发字节大模型团队架构大调整

2月21日,有消息称前谷歌大佬吴永辉加入字节跳动,并担任大模型团队Seed基础研究负责人后,引发了字节跳动大模型团队架构大调整。多名原本向朱文佳汇报的算法和技术负责人开始转向吴永辉汇报。简单来说,就是...

31页组织效能提升模型,经营管理团队搭建框架与权责定位

分享职场干货,提升能力!为职场精英打造个人知识体系,升职加薪!31页组织效能提升模型如何拿到分享的源文件:请您关注本头条号,然后私信本头条号“文米”2个字,按照操作流程,专人负责发送源文件给您。...

异形柱结构(异形柱结构技术规程)

下列关于混凝土异形柱结构设计的说法,其中何项正确?(A)混凝土异形柱框架结构可用于所有非抗震和抗震设防地区的一般居住建筑。(B)抗震设防烈度为6度时,对标准设防类(丙类)采用异形柱结构的建筑可不进行地...

职场干货:金字塔原理(金字塔原理实战篇)

金字塔原理的适用范围:金字塔原理适用于所有需要构建清晰逻辑框架的文章。第一篇:表达的逻辑。如何利用金字塔原理构建基本的金字塔结构受众(包括读者、听众、观众或学员)最容易理解的顺序:先了解主要的、抽象的...

底部剪力法(底部剪力法的基本原理)

某四层钢筋混凝土框架结构,计算简图如图1所示。抗震设防类别为丙类,抗震设防烈度为8度(0.2g),Ⅱ类场地,设计地震分组为第一组,第一自振周期T1=0.55s。一至四层的楼层侧向刚度依次为:K1=1...

结构等效重力荷载代表值(等效重力荷载系数)

某五层钢筋混凝土框架结构办公楼,房屋高度25.45m。抗震设防烈度8度,设防类别丙类,设计基本地震加速度0.2g,设计地震分组第二组,场地类别为Ⅱ类,混凝土强度等级C30。该结构平面和竖向均规则。假定...

体系结构已成昭告后世善莫大焉(体系构架是什么意思)

实践先行也理论已初步完成框架结构留余后人后世子孙俗话说前人栽树后人乘凉在夏商周大明大清民国共和前人栽树下吾之辈已完成结构体系又俗话说青出于蓝而胜于蓝各个时期任务不同吾辈探索框架结构体系经历有限肯定发展...

框架柱抗震构造要求(框架柱抗震设计)

某现浇钢筋混凝土框架-剪力墙结构高层办公楼,抗震设防烈度为8度(0.2g),场地类别为Ⅱ类,抗震等级:框架二级,剪力墙一级,混凝土强度等级:框架柱及剪力墙C50,框架梁及楼板C35,纵向钢筋及箍筋均采...

梁的刚度、挠度控制(钢梁挠度过大会引起什么原因)

某办公楼为现浇钢筋混凝土框架结构,r0=1.0,混凝土强度等级C35,纵向钢筋采用HRB400,箍筋采用HPB300。其二层(中间楼层)的局部平面图和次梁L-1的计算简图如图1~3(Z)所示,其中,K...

死要面子!有钱做大玻璃窗,却没有钱做“柱和梁”,不怕房塌吗?

活久见,有钱做2层落地大玻璃窗,却没有钱做“柱子和圈梁”,这样的农村自建房,安全吗?最近刷到个魔幻施工现场,如下图,这栋5开间的农村自建房,居然做了2个全景落地窗仔细观察,这2个落地窗还是飘窗,为了追...

不是承重墙,物业也不让拆?话说装修就一定要拆墙才行么

最近发现好多朋友装修时总想拆墙“爆改”空间,别以为只要避开承重墙就能随便砸!我家楼上邻居去年装修,拆了阳台矮墙想扩客厅,结果物业直接上门叫停。后来才知道,这种配重墙拆了会让阳台承重失衡,整栋楼都可能变...

取消回复欢迎 发表评论: