JPA(Java Persistence API)是一种Java持久化解决方案,负责把数据保存到数据库。

概述

    JPA是存储业务实体关联的实体的来源。它显示了如何定义一个面向普通Java对象(POJO)作为一个实体,以及如何与管理关系实体。

    在 EJB 3 之前,EJB主要包含三种类型:

  •     会话Bean
  •     消息驱动Bean
  •     实体Bean

    但自 EJB 3.0 开始,实体Bean被单独分离出来,形成了新的规范:JPA。所以,JPA完全可以脱离 EJB 3 来使用。

    知道了JPA的由来,那么也就很清楚其实它就是一种标准、规范,而不是具体的实现(如Hibernate)。

    JPA是Java官方提出的、随Java EE 5 规范一同发布的。在此之前,已经存在很多ORM框架了,他们都有不同的实现,这也是JAP出现的必然条件:

    ORM框架的出现,是直接存储对象称为可能,它们将Java对象拆分成SQL语句,并利用JDBC保存到数据库。但是不同的框架,使用起来却是很大不同的,这也导致开发者需要学习各种不同的ORM框架(虽然你可能觉得精通一个ORM框架就够了,但你也不能保证不会碰到需要维护别人的程序的时候,当然,他们很可能就使用了不同的框架)。而JAP规范,正式为了解决这个问题:规范ORM框架,是ORM框架有统一的接口和用法。

    至于规范,我们可以把它当作接口来理解,它规定了应该怎么做,但不包含任何实现。所以JPA并不能单独使用,必须指定实现了JPA的框架。

    JAP既然已经解释清楚了,好处就很显然:程序不再依赖某个具体的ORM框架。比如:使用JPA+Hibernate的程序,简单的修改配置,即可改为JPA+TopLink,以为它们都实现了JPA规范。

实例

    Eclipselink 也是一种比较强大的ORM框架,由Oracle企业级的ORM平台TopLink代码捐献给Eclipse社区而来的。所以下面会以Eclipselink 作JPA实现来介绍。

    首先创建工程:

    01

    输入项目名字,选择JDK1.7以上版本,并选择JPA版本:

    02

    继续点下一步,知道出现下面这个界面,选择使用EclipseLink库,如果没有,请点游标的小图标下载,知道可以如图勾选,然后点finish:

    03

    这时,你应该能得到如下项目结构:

    04

    按自己的习惯,新建包。

    接下来编写一个实体:

package com.anxpp.demo.jpa;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;
/**
 * 实体
 * @author anxpp
 */
@Entity //注解表示实体
@Table  //对应数据库的person表
public class Person {
    //自增长主键
    @Id
    @GeneratedValue(strategy= GenerationType.AUTO)  
    private int id;
    private String name;
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "id:" + id + ",name:" + name;
    }
}

    这时应该会报错,因为persistence.xml文件中还没配置实体,而且其他配置信息也还没有,下面就为其添加位置,添加后完整的persistence.xml文件内容如下(为了演示输出时不会有太多内容,日志设为OFF):

<?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="JPADemo">
        <class>com.anxpp.demo.jpa.Person</class>
        <properties>
            <property name="javax.persistence.jdbc.url" value="jdbc:mysql://localhost:3306/jpademo" />
            <property name="javax.persistence.jdbc.user" value="root" />
            <property name="javax.persistence.jdbc.password" value="root" />
            <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver" />
            <property name="eclipselink.logging.level" value="OFF" />
            <property name="eclipselink.ddl-generation" value="create-tables" />
        </properties>
    </persistence-unit>
</persistence>

    很显然,上面配置的需要访问的数据库是MySQL,所以我们需要为项目添加jdbc驱动,本人使用的 mysql-connector-java-5.1.9.jar ,读者可以按自己想要的添加。

    接下来就是CRUD测试方法的编写,下面将代码一并贴出(代码仅供介绍JPA,编码方式没有任何参考价值):

package com.anxpp.demo.jpa;
import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
/**
 * 测试使用
 * @author anxpp
 */
public class TestUse {
    public static void main(String args[]){
        TestUse testUse = new TestUse();
        testUse.save();     //添加一条
        testUse.findAll();  //查看结果1
        testUse.save();     //再添加一条
        testUse.findAll();  //查看结果2
        testUse.update();   //更新数据
        testUse.findAll();  //查看结果3
        testUse.del();      //删除数据
        testUse.findAll();  //查看结果4
        testUse.close();
        System.out.println("-------------------------");
    }
    EntityManagerFactory emfactory = Persistence.createEntityManagerFactory("JPADemo");
    EntityManager entitymanager = emfactory.createEntityManager();
    public void save() {
        entitymanager.getTransaction().begin();
        Person person = new Person();
        person.setName("anxpp");
        entitymanager.persist(person);
        entitymanager.getTransaction().commit();
    }
    public void update() {
        entitymanager.getTransaction().begin();
        Person person=entitymanager.find(Person.class,1);
        person.setName(person.getName() + "+");
        entitymanager.getTransaction().commit();
    }
    public void findAll(){
        entitymanager.getTransaction().begin();
        @SuppressWarnings("unchecked")
        List<Person> persons=entitymanager.createQuery("select p from Person p").getResultList();
        System.out.println("当前查询结果:" + persons);
        entitymanager.getTransaction().commit();
    }
    public void del() {
        entitymanager.getTransaction().begin();
        Person person=entitymanager.find( Person.class, 1 );
        entitymanager.remove(person);
        entitymanager.getTransaction().commit();
    }
    public void close(){
        entitymanager.close();
        emfactory.close();
    }
}

    以下是输出结果:

当前查询结果:[id:1,name:anxpp]
当前查询结果:[id:1,name:anxpp, id:2,name:anxpp]
当前查询结果:[id:1,name:anxpp+, id:2,name:anxpp]
当前查询结果:[id:2,name:anxpp]
-------------------------

    但是JPA毕竟是一种重量级的,需要Jave EE 容器支持(如Jboss)。

    JPA有一套标准的API,这里不多介绍,其实并没有很多人去详细了解这些。后面会介绍Spring Data JPA,也应是我们开发时框架的首选。相对传统的Spring+Hibernate,Spring Data JPA+Hibernate是一种更先进,也更方便简单的用法,相应的文章,也会在最近一两天编辑完成,完成后会在本文添加链接,欢迎大家阅读,互相学习探讨。