一篇文章理清Java持久化脉络(关于JDBC、JPA、Hibernate、Spring Data JPA)

news2024/9/20 3:09:27

Java持久化

    • 一、JDBC、JPA、Spring Data JPA 的定义及关系
    • 二、JDBC(古老的东西,只需简单知道是啥)
      • 1.1 JDBC概念
      • 1.2 JDBC示例
    • 三、JPA(第二代持久化,代表是Hibernate等框架)
      • 3.1 JPA概念
      • 3.2 JPA示例
    • 四、Spring Data JPA(第三代,只需学习这个的使用)
    • 五、总结

JDBC、JPA、Hibernate、Spring Data JPA,这个经常听到的词汇,具体是什么,他们是什么关系,以及如何通过他们实现java的持久化,一文带你理清Java持久化的脉络。

一、JDBC、JPA、Spring Data JPA 的定义及关系

JDBC,是一组Java接口,是Java提供的面向关系的持久化规范。各个数据库厂商(Oracle、MySQL等)提供实现,JDBC是面向SQL的,就是在代码力去执行SQL语句。

JPA,也是一组接口,是Java提供的面向对象的持久化规范。各个ORM(也就是Hibernate,MyBatis等)提供实现,JPA是面向对象的,也就是我们熟知的,定义一个实体,然后就可以和关系型数据库进行交互。

Spring Data JPA,ORM的又一层抽象,可以通过方法名约定进一步简化样板代码,底层是通过调用Hibernate等ORM实现的。Spring Data JPA默认是通过调用Hibernate实现的。

二、JDBC(古老的东西,只需简单知道是啥)

1.1 JDBC概念

JDBC是一组java api,只是接口没有实现,各数据库厂商提供实现。开发只需要面向jdbc api即可。

要使用不同的数据库,只需要切换不同的实现类即可。也就是使用不同的驱动包。

jdbc就是建立在sql之上的,sql能干的,它都能干,例如DDL、DML、DQL等,能更好的对数据进行调优,允许使用数据的所有特性,而这是其他框架不鼓励甚至禁止的。

1.2 JDBC示例

不用管具体代码含义,只大概看看,简单一条查询,都干了什么。创建数据库连接–先写SQL语句–关闭连接处理异常等。每一个操作都需要这样执行一遍。

Musician georgeHarrison = new Musician(0, "George Harrison");<br><br>  // 建立数据库连接等

String myDriver = "org.gjt.mm.mysql.Driver";
String myUrl = "jdbc:mysql://localhost/test";
Class.forName(myDriver);
Connection conn = DriverManager.getConnection(myUrl, "root", "");
// 纯SQL语句
String query = " insert into users (id, name) values (?, ?)";
PreparedStatement preparedStmt = conn.prepareStatement(query);
preparedStmt.setInt     (1, 0);
preparedStmt.setString (2, "George Harrison");
preparedStmt.setString (2, "Rubble");
preparedStmt.execute();<br><br>// 必须手动关闭连接
conn.close();
// 还有一些异常处理

每一个查询、操作,即使只查询一条数据,也需要写这一堆代码,极其繁琐。因此催生了JPA的诞生。

三、JPA(第二代持久化,代表是Hibernate等框架)

3.1 JPA概念

JPA是面向对象的持久化规范。不用像JPA一样手动创建连接,编写SQL等,只需要创建与表对应的实体,并给实体打上相应的标记,JPA的实现框架就能自动生成SQL,进行持久化相关操作。

不过它的底层也是通过JDBC实现的。
在这里插入图片描述

JPA与HIbernate的关系?

题外话,现在我们都知道Hibernate是JPA的实现,但是追随历史,这是一个先有鸡还是先有蛋的问题。

Hibernate是2002年发布的,而JPA是2006年发布的,所以JPA的很多规范其实是参考了Hibernate,JPA发布后,Hibernate 3.0又根据JPA规范进行调整,最终被认证为JPA的实现之一。由此可见Hibernate 的江湖地位。

3.2 JPA示例

  • ORM简单示例

简单演示一下使用Hibernate等这种ORM后,保存数据只需要定义一个Entity,然后保存即可。

// 定义实体,标记上相应的注解
@Entity
public class Musician {
   @Id
   private Long id;
}



Musician georgeHarrison = new Musician(0, "George Harrison");
// 直接一行皆可,不需要处理数据库连接,不需要写一堆SQL
musicianManager.save(georgeHarrison);
  • Spring继承Hibernate的详细配置

业务操作很简单,下面演示一下spring怎样继承Hibernate。

实现ORM,需要配置三个东西:数据源DataSource、SessionFactory、事务。

通过xml配置:

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- 数据源配置 -->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.cj.jdbc.Driver" />
        <property name="url" value="jdbc:mysql://localhost:3306/yourdb" />
        <property name="username" value="username" />
        <property name="password" value="password" />
    </bean>

    <!-- Hibernate SessionFactory 配置 -->
    <bean id="sessionFactory" class="org.springframework.orm.hibernate5.LocalSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="packagesToScan" value="com.yourpackage.model" />
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</prop>
                <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.hbm2ddl.auto">update</prop>
            </props>
        </property>
    </bean>

    <!-- 事务管理器 -->
    <bean id="transactionManager" class="org.springframework.orm.hibernate5.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory" />
    </bean>

</beans>

通过代码配置:

@Configuration
@EnableTransactionManagement
public class HibernateConfig {

    @Bean
    public DataSource dataSource() {
        DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://localhost:3306/yourdb");
        dataSource.setUsername("username");
        dataSource.setPassword("password");
        return dataSource;
    }

    @Bean
    public LocalSessionFactoryBean sessionFactory() {
        LocalSessionFactoryBean sessionFactory = new LocalSessionFactoryBean();
        sessionFactory.setDataSource(dataSource());
        sessionFactory.setPackagesToScan("com.yourpackage.model");
        sessionFactory.setHibernateProperties(hibernateProperties());
        return sessionFactory;
    }

    private Properties hibernateProperties() {
        Properties properties = new Properties();
        properties.put("hibernate.dialect", "org.hibernate.dialect.MySQL5Dialect");
        properties.put("hibernate.show_sql", "true");
        properties.put("hibernate.hbm2ddl.auto", "update");
        return properties;
    }

    @Bean
    public HibernateTransactionManager transactionManager(SessionFactory sessionFactory) {
        HibernateTransactionManager txManager = new HibernateTransactionManager();
        txManager.setSessionFactory(sessionFactory);
        return txManager;
    }
}

Hibernate 有一个称为上下文会话的功能,其中 Hibernate 本身管理Session每个事务的一个当前会话。一般会通过为每一个实体再创建一个Dao类来调用SessionFactory。

public class ProductDaoImpl implements ProductDao {

 private SessionFactory sessionFactory;

 public void setSessionFactory(SessionFactory sessionFactory) {
  this.sessionFactory = sessionFactory;
 }

 public Collection loadProductsByCategory(String category) {
  return this.sessionFactory.getCurrentSession()
    .createQuery("from test.Product product where product.category=?")
    .setParameter(0, category)
    .list();
 }
}

四、Spring Data JPA(第三代,只需学习这个的使用)

JPA构建在JDBC之上,帮我们省略了数据库连接的管理、sql编写等繁琐的样板代码。而Spring Data JPA是构建在JPA之上的,能够进一步减少样板代码的编写,让我们尽量只关注业务逻辑。

看上面的Hibernate,Dao层其实也有许多样板代码,通过使用SessionFactory 来执行查询和操作等,而这在spring data jpa中进一步做了简化。其中包括:

1、简化配置

一般通过spring boot使用spring data jpa,其中配置项只有这些。

spring:
  datasource:
    url: jdbc:mysql://localhost:3306/yourdb
    username: username
    password: password
    driver-class-name: com.mysql.cj.jdbc.Driver

  jpa:
    hibernate:
      ddl-auto: update
    show-sql: true
    properties:
      hibernate:
        dialect: org.hibernate.dialect.MySQLDialect

可以发现少了事务的配置,这是因为默认使用spring data 的JpaTransactionManager,它可以满足大部分需求,无需再单独配置。

2、Repository 模式极大精简传统Dao层操作

其中包括,直接写接口声明方法,就可以实现数据库访问;自动化CEUD操作,findById(), save(), delete()这些无需手动编写;支持通过方法名称自动生成查询(如 findByLastName(String lastName))。还包括分页排序、自动创建表等。

Spring Data 进一步简化了这一过程,并可以完全删除 DAO 实现。DAO 的接口现在是我们唯一需要明确定义的工件。

传统的Dao层,可以直接使用以下接口实现。

public interface IFooRepository extends JpaRepository<Foo, Long> {

    Foo findByName(String name);
@Query("SELECT f FROM Foo f WHERE LOWER(f.name) = LOWER(:name)")
  @Query("SELECT f FROM Foo f WHERE LOWER(f.name) = LOWER(:name)")

  Foo retrieveByName(@Param("name")String name);
}

五、总结

java的持久化也是经历了从自行车到汽车的转变,从JDBC直接调用sql,到jpa封装jdbc,使用面向对象的思想处理持久化,再到spring data jpa进一步对jpa进行抽象,目前的学习成本和编码成本已经很低了,只需要学习spring data jpa即可。
了解这些发现,有助于我们理解底层实现,

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2042880.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

GB/T 38082-2019 生物降解塑料购物袋检测

生物降解塑料购物袋是指以生物降解树脂为主要原料制得的&#xff0c;具有提携结构的&#xff0c;在销售、服务等场所用于盛装及携提商品的袋制品。 GB/T 38082-2019 生物降解塑料购物袋检测项目&#xff1a; 检测项目 测试标准 尺寸偏差 GB/T 38082 感官 GB/T 38082 提掉…

html+css+js网页设计 大一电商6个页面 带js 有轮播图,增删改查等功能

htmlcssjs网页设计 大一电商6个页面 带js 有轮播图&#xff0c;增删改查等功能 网页作品代码简单&#xff0c;可使用任意HTML编辑软件&#xff08;如&#xff1a;Dreamweaver、HBuilder、Vscode 、Sublime 、Webstorm、Text 、Notepad 等任意html编辑软件进行运行及修改编辑等…

JTAG标准笔记:IEEE1149.1、IEEE1149.4、IEEE1149.5、IEEE1149.6、 IEEE1500等协议之前的发展和联系

JTAG (Joint Test Action Group) 是一种串行通信协议。对于典型的串行通信&#xff0c;总线较少&#xff0c;线路数通常为1到4条&#xff0c;数据是以位为单位依次传输的。笔记中大部分图片来自JTAG标准介绍UP的视频 IEEE 1149.1&#xff0c;通常称为JTAG&#xff08;Joint Tes…

安防监控/视频汇聚平台EasyCVR如何配置,实现默认获取设备的子码流?

安防视频监控/视频集中存储/云存储/磁盘阵列EasyCVR平台基于云边端一体化架构&#xff0c;兼容性强、支持多协议接入&#xff0c;包括国标GB/T 28181协议、部标JT808、GA/T 1400协议、RTMP、RTSP/Onvif协议、海康Ehome、海康SDK、大华SDK、华为SDK、宇视SDK、乐橙SDK、萤石云SD…

设计模式22-迭代器模式

设计模式22-迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;动机定义结构定义结构结构图解释注意事项 C代码推导多态属性&#xff08;虚函数&#xff09;实现迭代器1. **返回值问题**2. **对象切割问题**3. **内存管理问题**4. **迭代器生命周期问题**5. **接口…

static、extern,const关键字

1、static关键字 static关键字&#xff1a;延长生命周期&#xff0c;限制作用域 static修饰局部变量&#xff1a;静态局部变量 static修饰全局变量&#xff1a;静态全局变量 static修饰函数&#xff1a;静态函数 2、extern关键字 extern&#xff1a;引用其他文件 .c 中的全局…

对敲期权组合如何操作?

对敲期权组合按照你说的对沖敲出期权应该是一种期权套利行为&#xff0c;在买入的同时卖出一个执行价格不同的期权进行对冲&#xff0c;或者在卖出一张期权合约的时候同时买进一张执行价不动的同类期权进行对中&#xff0c;这样亏报有限&#xff0c;是种套利行为&#xff0c;下…

Java基础之进制转换

1 进制基础 概念&#xff1a; ​ 进制就是进位制&#xff0c;是人们规定的一种进位方法&#xff0c;二进制逢2进1&#xff0c;八进制是逢8进1&#xff0c;十进制逢10进1&#xff0c;十六进制逢16进1。 不同进制形式&#xff1a; 二进制 0b或0B开头&#xff0c;由0和1组成 八…

爬虫配置代理:保护隐私有效地抓取数据

爬虫配置代理的详细指南 在进行网络爬虫时&#xff0c;使用代理可以帮助我们更有效地抓取数据&#xff0c;避免IP被封禁&#xff0c;并提高隐私保护。本文将详细介绍如何在爬虫中配置代理&#xff0c;包括不同的代理类型、如何选择合适的代理以及在Python中实现代理的具体步骤…

中国软件评测中心:2024年最新人工智能大语言模型技术发展研究报告 (附文档)

人工智能作为引领新一轮科技产业革命的战略性技术和新质生产力重要驱动力&#xff0c;正在引发经济、社会、文化等领域的变革和重塑&#xff0c;2023 年以来&#xff0c;以 ChatGPT、GPT-4 为代表的大模型技术的出台&#xff0c;因其强大的内容生成及多轮对话能力&#xff0c;引…

python-A+B again

[题目描述] 小理有一个非常简单的问题给你&#xff0c;给你两个整数 A 和 B&#xff0c;你的任务是计算 AB。输入格式&#xff1a; 输入共 2∗T1 行。 输入的第一行包含一个整数 T 表示测试实例的个数&#xff0c;然后 2∗T 行&#xff0c;分别表示 A 和 B 两个正整数。注意整数…

调研在深度学习中如何读代码

这里调研了四个up主的内容&#xff0c;对他们讲的内容摘了一下主要的内容。想要看原文的画可以看原篇。 1.如何学习别人的代码&#xff08;代码量较大时&#xff09;_怎么学习别人的代码-CSDN博客 想要掌握的好&#xff0c;光阅读是不够的&#xff0c;一定要动手写、训练模型…

k8s 部署RuoYi-Vue-Plus之minio搭建

1.直接部署一个pod 需要挂载存储款, 可参考 之前文章设置 https://blog.csdn.net/weimeibuqieryu/article/details/140183843 2.部署yaml 创建部署文件 minio-deploy.yaml apiVersion: v1 kind: PersistentVolume metadata:name: minio-pvnamespace: ruoyi #使用ns ruoyi s…

MyCAT读写分离实现

1. 添加一个新的虚拟主机&#xff0c;设置ip为10.1.1.60,主机名为 mycat.yuanyu.zhangmin.关闭防火墙 SELinux NetworkManager 2. 上传jdk和mycat安装包 3. 解压并且添加到指定的位置 4. 查看并且配置jdk环境 、5. 测试启动myca就可以了 6. 找到server.xml和schema.xml 7. 配…

Python酷库之旅-第三方库Pandas(081)

目录 一、用法精讲 336、pandas.Series.str.rpartition方法 336-1、语法 336-2、参数 336-3、功能 336-4、返回值 336-5、说明 336-6、用法 336-6-1、数据准备 336-6-2、代码示例 336-6-3、结果输出 337、pandas.Series.str.slice方法 337-1、语法 337-2、参数 …

RCE---eval长度限制绕过技巧

目录 题目源码 方法一&#xff1a;命令执行的利用 方法二&#xff1a;file_put_contents&#xff08;本地文件包含的利用&#xff09; 方法三&#xff1a;usort(…$_GET); 题目源码 <?php $param $_REQUEST[param]; if(strlen($param)<17 && stripos($par…

【题解】【一题多解】—— [NOIP1998 普及组] 三连击

【题解】—— [NOIP1998 普及组] 三连击 [NOIP1998 普及组] 三连击题目背景题目描述输入格式输出格式输入输出样例输入 #1输出 #1 提示 解法1.直接提交答案解法2.普通枚举2.1.题意分析2.2.AC代码 解法3.全排列枚举3.1.题意分析3.2.AC代码 解法4.深度优先搜索4.1.题意分析4.2.AC…

宝塔面板屏蔽 Censys,防止源站 IP 泄露

Censys 搜索引擎很强大。Censys 每天都会扫描 IPv4 地址空间&#xff0c;以搜索所有联网设备并收集相关的信息&#xff0c;并返回一份有关资源&#xff08;如设备、网站和证书&#xff09;配置和部署信息的总体报告。 在 IP 前加上 https 访问时&#xff0c;Nginx 会自动返回该…

嵌入式学习 20(Linux高级编程——文件——misc)

文件操作相关函数 一、symlink 函数 int symlink(const char *oldpath, const char *newpath); 功能&#xff1a; 创建一个指向 oldpath 文件的新的符号链接&#xff08;软链接&#xff09;文件。 参数&#xff1a; • oldpath&#xff1a;被链接指向的原始文件的路径。 • …

spring事务失效问题可以这样解决

今天咱们就来聊聊在Spring事务管理中你可能踩过的坑&#xff0c;并教你如何规避这些陷阱。 事务提交的陷阱 有时&#xff0c;事务方法在抛出异常后没有回滚&#xff0c;而是被提交了。 这通常是由于异常被捕获但没有显式抛出&#xff0c;导致Spring误以为事务正常完成。 错…