JPA 基础

数据库驱动 ==> JDBC 规范 ==> ORM 框架 ==> JPA 规范 ==> spring-data-jpa

ORM 思想

JPA 的使用步骤

jpa 依赖

<properties>
<project.hibernate.version>5.4.2.Final</project.hibernate.version>
</properties> <!-- hibernate对jpa的支持包 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-entitymanager</artifactId>
<version>${project.hibernate.version}</version>
</dependency> <!-- c3p0 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-c3p0</artifactId>
<version>${project.hibernate.version}</version>
</dependency> <!-- Mysql and MariaDB -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.6</version>
</dependency>

jpa 的持久化配置

<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://java.sun.com/xml/ns/persistence" version="1.0">
<!-- 持久化单元 -->
<persistence-unit name="myJPA" transaction-type="RESOURCE_LOCAL">
<!-- JPA 的实现者 -->
<provider>org.hibernate.jpa.HibernatePersistenceProvider</provider> <properties>
<!-- 数据库信息
驱动 javax.persistence.jdbc.driver
数据库地址 javax.persistence.jdbc.url
用户名 javax.persistence.jdbc.user
密码 javax.persistence.jdbc.password
-->
<property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver" />
<property name="javax.persistence.jdbc.url" value="jdbc:mysql:///jpa_db" />
<property name="javax.persistence.jdbc.user" value="root" />
<property name="javax.persistence.jdbc.password" value="root" /> <!-- jpa 实现者的配置
显示 sql : hibernate.show_sql
格式化 sql : hibernate.format_sql
创建表: hibernate.hbm2ddl.auto 库必须存在,hibernate 不会自动创建库
create: 运行时创建,有则先删除再创建
update: 运行时创建,有则不创建
none: 不创建
-->
<property name="hibernate.show_sql" value="true" />
<property name="hibernate.format_sql" value="true" />
<property name="hibernate.hbm2ddl.auto" value="update" /> </properties> </persistence-unit>
</persistence>

实体类

package com.mozq.jpa.domain;

import org.hibernate.annotations.GeneratorType;

import javax.persistence.*;

@Entity
@Table
public class Customer {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long custId;
@Column
private String custName;
@Column
private String custSource;
@Column
private String custLevel;
@Column
private String custIndustry;
@Column
private String custPhone;
@Column
private String custAddress;
//省略构造器和 get/set
}

使用 jpa 的 API 进行操作

JPA实体的四种状态 https://www.jianshu.com/p/636954880af8

JPA中的实体对象拥有四种状态:

  • 瞬时状态(transient)
  • 持久状态(persistent)
  • 游离状态(detached)
  • 删除状态 (deleted)

瞬时状态

瞬时状态的实体就是一个普通的java对象,和持久化上下文无关联,数据库中也没有数据与之对应。

托管状态

使用 EntityManager 进行 find 或者 persist 操作返回的对象即处于托管状态,此时该对象已经处于持久化上下文中,因此任何对于该实体的更新都会同步到数据库中。

游离状态

当事务提交后,处于托管状态的对象就转变为了游离状态。此时该对象已经不处于持久化上下文中,因此任何对于该对象的修改都不会同步到数据库中。但是数据库中有这个对象对应的记录。

删除状态

当调用EntityManger对实体进行delete后,该实体对象就处于删除状态。其本质也就是一个瞬时状态的对象。

持久:处于持久的对象被 EntityManager 管理,当事务被提交时,则对持久对象的任何修改都将同步到数据库中。

public interface EntityManager {
/*
瞬时对象
游离对象将报异常
*/
void persist(Object var1); // 对象持久化,交给 EntityManager 管理。数据库中必须不存在。瞬时 ==> 持久
<T> T merge(T var1);// 根据 id 是否存在,插入新对象或更新现有对象。瞬时 ==> 持久 / 游离 ==> 持久 void remove(Object var1);
<T> T find(Class<T> entityClass, Object id);// 立即加载,返回真实对象
<T> T getReference(Class<T> entityClass, Object id);// 延迟加载,返回代理对象 void flush();
/* obj 必须是受管对象,用于将数据库数据同步到受管对象,如果是其他状态的对象将抛异常。*/
void refresh(Object var1);
void clear();
void detach(Object var1); void close();//关闭
EntityTransaction getTransaction();// 获取事务对象
}
# void refresh(Object var1);
java.lang.IllegalArgumentException: Entity not managed
# persist 方法异常
Caused by: org.hibernate.PersistentObjectException: detached entity passed to persist: com.mozq.jpa.domain.Customer Customer customer = new Customer();
customer.setCustId(1L);
customer.setCustName("刘备");
em.persist(customer);//此时数据库中存在 1L 记录,报错。
@Test
public void testSave(){
// Persistence 创建实体类管理器工厂
EntityManagerFactory factory = Persistence.createEntityManagerFactory("myJPA");
// 实体类管理器
EntityManager em = factory.createEntityManager();
// 事务对象
EntityTransaction tx = em.getTransaction();
tx.begin(); // CRUD
Customer customer = new Customer();
customer.setCustName("刘备"); em.persist(customer); // 保存 tx.commit();
// 释放资源
em.close();
factory.close();
}

JPQL 查询

// 查询全部
Query query = em.createQuery("from Customer");
List resultList = query.getResultList();
System.out.println(resultList);
// 排序
Query query = em.createQuery("from Customer order by custAddress desc ,custName asc, custId");
List resultList = query.getResultList();
/*
order by
customer0_.custAddress desc,
customer0_.custName asc,
customer0_.custId
*/
// 分页
Query query = em.createQuery("from Customer order by custId");
query.setFirstResult(2);
query.setMaxResults(3);
/*
order by
customer0_.custId limit ?, ?
*/

Hibernate 主键生成策略和创建表策略

public enum GenerationType {
TABLE,
SEQUENCE,
IDENTITY,
AUTO; private GenerationType() {
}
}
hibernate.hbm2ddl.auto create/update/none
# <property name="hibernate.hbm2ddl.auto" value="create" /> # 先删除表,再创建表
# @GeneratedValue(strategy = GenerationType.IDENTITY)
drop table if exists Customer
create table Customer (
custId bigint not null auto_increment, # 使用自增主键,生成的 sql 不会有主键字段。需要数据库支持自增。
custAddress varchar(255),
custIndustry varchar(255),
custLevel varchar(255),
custName varchar(255),
custPhone varchar(255),
custSource varchar(255),
primary key (custId)
) engine=InnoDB
insert
into
Customer
(custAddress, custIndustry, custLevel, custName, custPhone, custSource)
values
(?, ?, ?, ?, ?, ?)
# <property name="hibernate.hbm2ddl.auto" value="update" />
# @GeneratedValue(strategy = GenerationType.IDENTITY) # 使用自增主键,生成的 sql 不会有主键字段。需要数据库支持自增。
create table Customer ( # 如果表不存在则会发送创建表的语句。
custId bigint not null auto_increment,
custAddress varchar(255),
custIndustry varchar(255),
custLevel varchar(255),
custName varchar(255),
custPhone varchar(255),
custSource varchar(255),
primary key (custId)
) engine=InnoDB
insert
into
Customer
(custAddress, custIndustry, custLevel, custName, custPhone, custSource)
values
(?, ?, ?, ?, ?, ?)

GenerationType.AUTO + create

drop table if exists Customer
drop table if exists hibernate_sequence create table Customer (
custId bigint not null,
custAddress varchar(255),
custIndustry varchar(255),
custLevel varchar(255),
custName varchar(255),
custPhone varchar(255),
custSource varchar(255),
primary key (custId)
) engine=InnoDB create table hibernate_sequence (
next_val bigint
) engine=InnoDB insert into hibernate_sequence values ( 1 ) select
next_val as id_val
from
hibernate_sequence for update 锁 update
hibernate_sequence
set
next_val= ?
where
next_val=? insert into
Customer
(custAddress, custIndustry, custLevel, custName, custPhone, custSource, custId)
values
(?, ?, ?, ?, ?, ?, ?)

JPA

Caused by: org.hibernate.id.IdentifierGenerationException: ids for this class must be manually assigned before calling save(): com.mozq.jpa.domain.Customer

JPA 基础的更多相关文章

  1. JPA基础

    目录 目录 1 一.JPA基础 2 1.1 JPA基础 2 1.2JPA开发过程 3 1.3 实体的生命周期及实体管理器常用方法 4 二.环境搭建 5 2.1 添加JPA支持 6 2.2 添加配置文件 ...

  2. 【Spring Data 系列学习】Spring Data JPA 基础查询

    [Spring Data 系列学习]Spring Data JPA 基础查询 前面的章节简单讲解了 了解 Spring Data JPA . Jpa 和 Hibernate,本章节开始通过案例上手 S ...

  3. (二) JPA基础

    一.什么是JAP JPA(Java Persistence API)是SUN官方推出的Java持久化规范,它为Java开发人员提供了一种对象/关联映射工具来管理Java应用中的关系数据.它的出现主要是 ...

  4. JPA学习笔记1——JPA基础

    1.JPA简介: Java持久化规范,是从EJB2.x以前的实体Bean(Entity bean)分离出来的,EJB3以后不再有实体bean,而是将实体bean放到JPA中实现.JPA是sun提出的一 ...

  5. JPA学习笔记1——JPA基础 (转自CSDN)

    http://blog.csdn.net/chjttony/article/details/6086298 1.JPA简介: Java持久化规范,是从EJB2.x以前的实体Bean(Entity be ...

  6. Spring Data Jpa (二)JPA基础查询

    介绍Spring Data Common里面的公用基本方法 (1)Spring Data Common的Repository Repository位于Spring Data Common的lib里面, ...

  7. Spring Data JPA 基础第二篇

    主要调用工具类JpaUtils类 package cn.itcast.utils;import javax.persistence.EntityManager;import javax.persist ...

  8. jpa基础知识掌握-分页-sql

    https://blog.csdn.net/liuchuanhong1/article/details/52042477

  9. spring data jpa 创建方法名进行简单查询

    版权声明:本文为博主原创文章,未经博主允许不得转载. spring data jpa 可以通过在接口中按照规定语法创建一个方法进行查询,spring data jpa 基础接口中,如CrudRepos ...

随机推荐

  1. Ubuntu安装支持PCL、LAS的CloudCompare

    git clone --recursive https://github.com/cloudcompare/trunk.git cd trunk mkdir build cd build cmake ...

  2. Spring 常犯的十大错误,这坑你踩过吗?

    阅读本文大概需要 9 分钟. 1.错误一:太过关注底层 我们正在解决这个常见错误,是因为 “非我所创” 综合症在软件开发领域很是常见.症状包括经常重写一些常见的代码,很多开发人员都有这种症状. 虽然理 ...

  3. C/C++深度优先搜索(递归树模拟)

    //C++深度优先搜索(递归树模拟) #define _CRT_SECURE_NO_WARNINGS #include <iostream> #define MAX_N 1000 usin ...

  4. PostgreSQL事务特性之嵌套事务

    嵌套事务的实现是基于SAVEPOINT.ROLLBACK TO SAVEPOINT和RELEASE SAVEPOINT的,也就是设置一个保存点,可以回滚到保存点和释放保存点. 测试表的初始状态如下: ...

  5. 笔记本CPU性能排行

    截图如下: 1. 图1 2. 图2 3. 4. 5. 6. 7. 8. 谢谢浏览!

  6. 图解微信小程序---实现行的删除和增加操作

    图解微信小程序之实现行的删除和增加操作 代码笔记部分 第一步:在项目的app.json中创建一个新的页面(页面名称英文,可自定义) 第二步:在创建的新页面中编写页面(注意bindtap属性值,因为是我 ...

  7. kvm虚拟机日常管理与配置

    1.  查看KVM虚拟机配置文件及运行状态 (1) KVM虚拟机默认配置文件位置: /etc/libvirt/qemu/ autostart目录是配置kvm虚拟机开机自启动目录.    (2) vir ...

  8. HBase统计表行数(RowCount)的四种方法

    背景:对于其他数据存储系统来说,统计表的行数是再基本不过的操作了,一般实现都非常简单:但对于HBase这种key-value存储结构的列式数据库,统计 RowCount 的方法却有好几种不同的花样,并 ...

  9. mini QQ(项目一)

    一个多人聊天工具(C/S结构),实现了如下功能: 一个可视化窗口,支持鼠标点击事件 注册功能,用户可以注册自己的聊天账号, 注册信息包括: 账号名(可以用姓名来替代账号,支持中文), 密码(聊天框输入 ...

  10. 对于js中事件冒泡的理解分析

    一. 事件 事件的三个阶段:事件捕获 -> 事件目标 -> 事件冒泡 捕获阶段:先由文档的根节点document往事件触发对象,从外向内捕获事件对象: 目标阶段:到达目标事件位置(事发地) ...