一、ORM概述[了解]
ORM(Object-Relational Mapping) 表示对象关系映射。只要有一套程序能够做到建立对象与数据库的关联,操作对象就可以直接操作数据库数据,就可以说这套程序实现了ORM对象关系映射
简单的说:ORM就是建立实体类和数据库表之间的关系,从而达到操作实体类就相当于操作数据库表的目的。
Mybatis(ibatis)、Hibernate、Jpa都是使用ORM思想
二、hibernate与JPA的概述[了解]
2.1    hibernate概述
Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的orm框架,hibernate可以自动生成SQL语句,自动执行,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库。
2.2    JPA概述
JPA的全称是Java Persistence API, 即Java 持久化API,是SUN公司推出的一套基于ORM的规范,内部是由一系列的接口和抽象类构成。
JPA通过JDK 5.0注解描述对象-关系表的映射关系,并将运行期的实体对象持久化到数据库中。 JPA框架中支持大数据集、事务、并发等容器级事务
2.3    JPA与hibernate的关系
JPA规范本质上就是一种ORM规范,注意不是ORM框架——因为JPA并未提供ORM实现,它只是制订了一些规范,提供了一些编程的API接口,但具体实现则由服务厂商来提供实现。           

JPA和Hibernate的关系就像JDBC和JDBC驱动的关系,JPA是规范,Hibernate除了作为ORM框架之外,它也是一种JPA实现。也就是说,如果使用JPA规范进行数据库操作,底层需要hibernate作为其实现类完成数据持久化工作。
三、JPA的实现
maven工程导入坐标
<properties> 	<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 	<project.hibernate.version>5.0.7.Final</project.hibernate.version> </properties>
  <dependencies> 	 	<dependency> 		<groupId>junit</groupId> 		<artifactId>junit</artifactId> 		<version>4.12</version> 		<scope>test</scope> 	</dependency>
  	 	<dependency> 		<groupId>org.hibernate</groupId> 		<artifactId>hibernate-entitymanager</artifactId> 		<version>${project.hibernate.version}</version> 	</dependency>
  	 	<dependency> 		<groupId>org.hibernate</groupId> 		<artifactId>hibernate-c3p0</artifactId> 		<version>${project.hibernate.version}</version> 	</dependency>
  	 	<dependency> 		<groupId>log4j</groupId> 		<artifactId>log4j</artifactId> 		<version>1.2.17</version> 	</dependency>
  	 	<dependency> 		<groupId>mysql</groupId> 		<artifactId>mysql-connector-java</artifactId> 		<version>5.1.6</version> 	</dependency> 	 	<dependency>        <groupId>org.projectlombok</groupId>        <artifactId>lombok</artifactId>        <version>1.18.10</version>     </dependency> </dependencies>
   | 
 
创建客户的数据库表
 CREATE TABLE cst_customer (   cust_id bigint(32) NOT NULL AUTO_INCREMENT COMMENT '客户编号(主键)',   cust_name varchar(32) NOT NULL COMMENT '客户名称(公司名称)',   cust_source varchar(32) DEFAULT NULL COMMENT '客户信息来源',   cust_industry varchar(32) DEFAULT NULL COMMENT '客户所属行业',   cust_level varchar(32) DEFAULT NULL COMMENT '客户级别',   cust_address varchar(128) DEFAULT NULL COMMENT '客户联系地址',   cust_phone varchar(64) DEFAULT NULL COMMENT '客户联系电话',   PRIMARY KEY (`cust_id`) ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
 
  | 
 
创建客户的实体类
public class Customer implements Serializable {      	private Long custId; 	private String custName; 	private String custSource; 	private String custIndustry; 	private String custLevel; 	private String custAddress; 	private String custPhone; 	 	public Long getCustId() { 		return custId; 	} 	public void setCustId(Long custId) { 		this.custId = custId; 	} 	public String getCustName() { 		return custName; 	} 	public void setCustName(String custName) { 		this.custName = custName; 	} 	public String getCustSource() { 		return custSource; 	} 	public void setCustSource(String custSource) { 		this.custSource = custSource; 	} 	public String getCustIndustry() { 		return custIndustry; 	} 	public void setCustIndustry(String custIndustry) { 		this.custIndustry = custIndustry; 	} 	public String getCustLevel() { 		return custLevel; 	} 	public void setCustLevel(String custLevel) { 		this.custLevel = custLevel; 	} 	public String getCustAddress() { 		return custAddress; 	} 	public void setCustAddress(String custAddress) { 		this.custAddress = custAddress; 	} 	public String getCustPhone() { 		return custPhone; 	} 	public void setCustPhone(String custPhone) { 		this.custPhone = custPhone; 	} }
  | 
 
3.1编写实体类和数据库表的映射配置[重点]
在实体类上使用JPA注解的形式配置映射关系
 
 
  @Entity  @Table(name="cst_customer")  @Data public class Customer { 	 	@Id 	@GeneratedValue(strategy=GenerationType.IDENTITY)  	@Column(name="cust_id")  	private Long custId; 	 	@Column(name="cust_name")  	private String custName; 	 	@Column(name="cust_source") 	private String custSource; 	 	@Column(name="cust_industry") 	private String custIndustry; 	 	@Column(name="cust_level") 	private String custLevel; 	 	@Column(name="cust_address") 	private String custAddress; 	 	@Column(name="cust_phone") 	private String custPhone; 	 }
 
  | 
 
常用注解的说明
@Entity 	作用:指定当前类是实体类。 @Table 	作用:指定实体类和表之间的对应关系。 	属性: 		name:指定数据库表的名称 @Id 	作用:指定当前字段是主键。 @GeneratedValue 	作用:指定主键的生成方式。。 	属性: 		strategy :指定主键生成策略。 @Column 	作用:指定实体类属性和数据库表之间的对应关系 	属性: 		name:指定数据库表的列名称。 		unique:是否唯一   		nullable:是否可以为空   		inserttable:是否可以插入   		updateable:是否可以更新   		columnDefinition: 定义建表时创建此列的DDL   		secondaryTable: 从表名。如果此列不建在主表上(默认建在主表),该属性定义该列所在从表的名字搭建开发环境[重点]
   | 
 
     
3.2配置JPA的核心配置文件
在java工程的src路径下创建一个名为META-INF的文件夹,在此文件夹下创建一个名为persistence.xml的配置文件
<?xml version="1.0" encoding="UTF-8"?> <persistence xmlns="http://java.sun.com/xml/ns/persistence" version="2.0">     
 
 
 
 
 
      <persistence-unit name="myJpa" transaction-type="RESOURCE_LOCAL">                  <provider>org.hibernate.jpa.HibernatePersistenceProvider</provider>
                   <properties>             
 
 
 
 
              <property name="javax.persistence.jdbc.user" value="root"/>             <property name="javax.persistence.jdbc.password" value="xmgl0609"/>             <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>             <property name="javax.persistence.jdbc.url" value="jdbc:mysql:///jpa"/>
              
 
 
 
 
 
 
              <property name="hibernate.show_sql" value="true" />             <property name="hibernate.hbm2ddl.auto" value="update" />         </properties>     </persistence-unit> </persistence>
   | 
 
实现操作
@Test 	public void test() { 		
 
 
  		EntityManagerFactory factory = Persistence.createEntityManagerFactory("myJpa"); 		 		EntityManager em = factory.createEntityManager(); 		 		EntityTransaction tx = em.getTransaction(); 		 		tx.begin(); 		Customer c = new Customer(); 		c.setCustName("xxx"); 		 		em.persist(c); 		 		tx.commit(); 		 		em.close(); 		factory.close(); 	}
   | 
 
3.3JPA中的主键生成策略
通过注解来映射hibernate实体的,主键标识为@Id, 其生成规则由@GeneratedValue设定的。
JPA提供的四种标准用法为TABLE,SEQUENCE,IDENTITY,AUTO。
具体说明如下:
IDENTITY:主键由数据库自动生成(主要是自动增长型)
用法:
@Id   @GeneratedValue(strategy = GenerationType.IDENTITY)  private Long custId;
   | 
 
TABLE:使用一个特定的数据库表格来保存主键
用法:
@Id   @GeneratedValue(strategy = GenerationType.TABLE, generator="payablemoney_gen")   @TableGenerator(name = "pk_gen",       table="tb_generator",       pkColumnName="gen_name",       valueColumnName="gen_value",       pkColumnValue="PAYABLEMOENY_PK",       allocationSize=1   )
   | 
 
private Long custId;
 
      @Target({TYPE, METHOD, FIELD})        @Retention(RUNTIME)       public @interface TableGenerator {                String name();                String table() default "";                String catalog() default "";         String schema() default "";                String pkColumnName() default "";                String valueColumnName() default "";                String pkColumnValue() default "";                int initialValue() default 0;                int allocationSize() default 50;         UniqueConstraint[] uniqueConstraints() default {};       }
   | 
 
//这里应用表tb_generator,定义为 :    CREATE TABLE  tb_generator (        id NUMBER NOT NULL,        gen_name VARCHAR2(255) NOT NULL,        gen_value NUMBER NOT NULL,        PRIMARY KEY(id)      )
   | 
 
四、JPA的API介绍
4.1    Persistence对象
Persistence对象主要作用是用于获取EntityManagerFactory对象的 。通过调用该类的createEntityManagerFactory静态方法,根据配置文件中持久化单元名称创建EntityManagerFactory。
 @Test String unitName = "myJpa"; EntityManagerFactory factory= Persistence.createEntityManagerFactory(unitName);
 
  | 
 
4.2    EntityManagerFactory
EntityManagerFactory 接口主要用来创建 EntityManager 实例
 EntityManager em = factory.createEntityManager();
 
  | 
 
由于EntityManagerFactory 是一个线程安全的对象(即多个线程访问同一个EntityManagerFactory 对象不会有线程安全问题),并且EntityManagerFactory 的创建极其浪费资源,所以在使用JPA编程时,我们可以对EntityManagerFactory 的创建进行优化,只需要做到一个工程只存在一个EntityManagerFactory 即可
static  {                  factory = Persistence.createEntityManagerFactory("myJpa");     }
  | 
 
4.3    EntityManager
在 JPA 规范中, EntityManager是完成持久化操作的核心对象。实体类作为普通 java对象,只有在调用 EntityManager将其持久化后才会变成持久化对象。EntityManager对象在一组实体类与底层数据源之间进行 O/R 映射的管理。它可以用来管理和更新 Entity Bean, 根椐主键查找 Entity Bean, 还可以通过JPQL语句查询实体。
我们可以通过调用EntityManager的方法完成获取事务,以及持久化数据库的操作
方法说明:
getTransaction : 获取事务对象 persist : 保存操作 merge : 更新操作 remove : 删除操作 find/getReference : 根据id查询
   | 
 
4.4    EntityTransaction
在 JPA 规范中, EntityTransaction是完成事务操作的核心对象,对于EntityTransaction在我们的java代码中承接的功能比较简单
begin:开启事务 commit:提交事务 rollback:回滚事务
   | 
 
五、抽取JPAUtil工具类
package cn.itcast.dao;
  import javax.persistence.EntityManager; import javax.persistence.EntityManagerFactory; import javax.persistence.Persistence;
  public final class JPAUtil { 	 	private static EntityManagerFactory em; 	 	static { 		 		em = Persistence.createEntityManagerFactory("myPersistUnit"); 	}
  	
 
 
 
  	public static EntityManager getEntityManager() { 		return em.createEntityManager(); 	} }
   | 
 
六、使用JPA完成CRUD
package cn.itcast.test;
  import cn.itcast.domain.Customer; import cn.itcast.utils.JpaUtils; import org.junit.Test;
  import javax.persistence.EntityManager; import javax.persistence.EntityManagerFactory; import javax.persistence.EntityTransaction; import javax.persistence.Persistence;
  public class JpaTest {
      
 
 
 
 
 
 
 
 
 
      @Test     public void testSave() {
 
 
 
          EntityManager em = JpaUtils.getEntityManager();                  EntityTransaction tx = em.getTransaction();          tx.begin();                  Customer customer = new Customer();         customer.setCustName("xxx");         customer.setCustIndustry("xx");                  em.persist(customer);                   tx.commit();                  em.close();  
      }
      
 
 
 
 
 
 
 
 
      @Test     public  void testFind() {                  EntityManager entityManager = JpaUtils.getEntityManager();                  EntityTransaction tx = entityManager.getTransaction();         tx.begin();                  
 
 
 
          Customer customer = entityManager.find(Customer.class, 1l);                          tx.commit();                  entityManager.close();     }
      
 
 
 
 
 
 
 
 
 
      @Test     public  void testReference() {                  EntityManager entityManager = JpaUtils.getEntityManager();                  EntityTransaction tx = entityManager.getTransaction();         tx.begin();                  
 
 
 
          Customer customer = entityManager.getReference(Customer.class, 1l);         System.out.print(customer);                  tx.commit();                  entityManager.close();     }
 
      
 
 
      @Test     public  void testRemove() {                  EntityManager entityManager = JpaUtils.getEntityManager();                  EntityTransaction tx = entityManager.getTransaction();         tx.begin();         
                   Customer customer = entityManager.find(Customer.class,1l);                  entityManager.remove(customer);
                   tx.commit();                  entityManager.close();     }
 
      
 
 
      @Test     public  void testUpdate() {                  EntityManager entityManager = JpaUtils.getEntityManager();                  EntityTransaction tx = entityManager.getTransaction();         tx.begin();         
                   Customer customer = entityManager.find(Customer.class,1l);                  customer.setCustIndustry("xxx");         entityManager.merge(customer);
                   tx.commit();                  entityManager.close();     }
  }
   | 
 
七、JPQL实现CRUD操作
JPQL全称Java Persistence Query Language
基于首次在EJB2.0中引入的EJB查询语言(EJB QL),Java持久化查询语言(JPQL)是一种可移植的查询语言,旨在以面向对象表达式语言的表达式,将SQL语法和简单查询语义绑定在一起。
其特征与原生SQL语句类似,并且完全面向对象,通过类名和属性访问,而不是表名和表的属性。
代码如下:
package cn.itcast.test;
  import cn.itcast.domain.Customer; import cn.itcast.utils.JpaUtils; import org.junit.Test;
  import javax.persistence.EntityManager; import javax.persistence.EntityTransaction; import javax.persistence.Query; import java.util.List;
 
 
 
  public class JpqlTest {
      
 
 
 
      @Test     public void testFindAll() {                  EntityManager em = JpaUtils.getEntityManager();                  EntityTransaction tx = em.getTransaction();         tx.begin();                  String jpql = "from Customer ";         Query query = em.createQuery(jpql);
                   List list = query.getResultList();
          for (Object obj : list) {             System.out.print(obj);         }
                   tx.commit();                  em.close();     }
 
      
 
 
 
 
 
 
 
 
      @Test     public void testOrders() {                  EntityManager em = JpaUtils.getEntityManager();                  EntityTransaction tx = em.getTransaction();         tx.begin();                  String jpql = "from Customer order by custId desc";         Query query = em.createQuery(jpql);
                   List list = query.getResultList();
          for (Object obj : list) {             System.out.println(obj);         }
                   tx.commit();                  em.close();     }
 
      
 
 
 
      @Test     public void testCount() {                  EntityManager em = JpaUtils.getEntityManager();                  EntityTransaction tx = em.getTransaction();         tx.begin();                           String jpql = "select count(custId) from Customer";         Query query = em.createQuery(jpql);                  
          
 
 
          Object result = query.getSingleResult();
          System.out.println(result);
                   tx.commit();                  em.close();     }
 
      
 
 
 
      @Test     public void testPaged() {                  EntityManager em = JpaUtils.getEntityManager();                  EntityTransaction tx = em.getTransaction();         tx.begin();                           String jpql = "from Customer";         Query query = em.createQuery(jpql);                           query.setFirstResult(0);                  query.setMaxResults(2);
          
          
 
 
          List list = query.getResultList();
          for(Object obj : list) {             System.out.println(obj);         }
                   tx.commit();                  em.close();     }
 
      
 
 
 
 
      @Test     public void testCondition() {                  EntityManager em = JpaUtils.getEntityManager();                  EntityTransaction tx = em.getTransaction();         tx.begin();                           String jpql = "from Customer where custName like ? ";         Query query = em.createQuery(jpql);                           query.setParameter(1,"传智播客%");
          
          
 
 
          List list = query.getResultList();
          for(Object obj : list) {             System.out.println(obj);         }
                   tx.commit();                  em.close();     }
  }
   |