JPA是一套对象持久化规范,可以通过多种工具实现它。比如常见的Hibernate。按说初学就该照猫画虎的写例子,可是总改不了探究为什么的好奇心。于是在经历了不少曲折和困惑之后,终于把Hibernate和eclipselink两种实现的例子都完成了,也体会了一点二者的区别。
先看在eclipse里的默认jpa项目的做法:
新建JPAProject,填写项目名称和Targetruntime(因为只是个简单的例子,不需要部署到服务器上,就选了jre),再点几次next之后,就过不去了,因为要求必须使用一个UserLibrary。这就是要添加实现jpa的类库。如果使用eclipselink可以选择下载,如果使用hibernate,可以新建一个,然后把hibernate包里的各个jar文件加进来。最终二者就是在persistence.xml文件的描述上有些许区别,这也是JPA规范的目的所在——使代码尽量不要绑定特定的ORM框架。下面是示例代码:
<?xml version="1.0" encoding="UTF-8"?> <persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"> <persistence-unit name="jpa-link2" transaction-type="RESOURCE_LOCAL"> <class>com.jpalink2.zjc.Connect</class> <properties> <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/> <property name="javax.persistence.jdbc.url" value="jdbc:mysql://localhost:3306/jpa"/> <property name="javax.persistence.jdbc.user" value="java"/> <property name="javax.persistence.jdbc.password" value="111111"/> <!-- 这个设置不知道为什么不起作用,不能自动建表,:( --> <property name="eclipelink.ddl-generation" value="drop-and-create-tables"/> </properties> </persistence-unit> </persistence>
package com.jpalink2.zjc; import javax.persistence.*; /** * Entity implementation class for Entity: Connect * */ @Entity @Table(name="contacts") public class Connect { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private int id; private String name; private String tel; public String getName() { return this.name; } public void setName(String name) { this.name = name; } public String getTel() { return this.tel; } public void setTel(String tel) { this.tel = tel; } public static void main(String[] arg){ EntityManagerFactory emf =Persistence.createEntityManagerFactory("jpa-link2"); Connect c=new Connect(); c.setName("hanmeimei"); c.setTel("13912423846"); EntityManager em = emf.createEntityManager(); em.getTransaction().begin(); em.persist(c); em.getTransaction().commit(); em.close(); emf.close(); } }
Hibernate示例代码:
<?xml version="1.0" encoding="UTF-8"?> <persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"> <persistence-unit name="jpa-hibernate"> <properties> <!--配置Hibernate方言 --> <property name="hibernate.dialect" value="org.hibernate.dialect.MySQL5Dialect" /> <!--配置数据库驱动 --> <property name="hibernate.connection.driver_class" value="com.mysql.jdbc.Driver" /> <!--配置数据库用户名 --> <property name="hibernate.connection.username" value="java" /> <!--配置数据库密码 --> <property name="hibernate.connection.password" value="111111" /> <!--配置数据库url --> <property name="hibernate.connection.url" value="jdbc:mysql://localhost:3306/jpa" /> <!--设置外连接抓取树的最大深度 --> <property name="hibernate.max_fetch_depth" value="3" /> <!--自动输出schema创建DDL语句 --> <property name="hibernate.hbm2ddl.auto" value="update" /> </properties> </persistence-unit> </persistence>
package com.jpahibernate.zjc; import javax.persistence.*; /** * Entity implementation class for Entity: Connect * */ @Entity @Table(name="contacts") public class Connect { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private int id; private String name; private String tel; public String getName() { return this.name; } public void setName(String name) { this.name = name; } public String getTel() { return this.tel; } public void setTel(String tel) { this.tel = tel; } public static void main(String[] arg){ EntityManagerFactory emf =Persistence.createEntityManagerFactory("jpa-hibernate"); Connect c=new Connect(); c.setName("zhangxiaoga"); c.setTel("13912423846"); EntityManager em = emf.createEntityManager(); em.getTransaction().begin(); em.persist(c); em.getTransaction().commit(); em.close(); emf.close(); } }
除了包名不同,其他代码一模一样。两个项目也都用的同一张表,先执行这个项目可以自动创建表,再执行上面的项目就不会报错了。
几个小插曲:
noinstall版本的mysql在修改my.ini的时候,增加basedir和datadir路径需要用/分隔,如:d:/soft/mysql,否则启动不了服务
选择创建普通java project也是可以完成示例的。而如果选择创建的是JPA Project,每添加一个@Entity注释的类,系统就会自动去检查是否增加到了 persistence.xml中(没有添加也可以执行,但作为有点强迫症的程序员,看着这个红叉实在是难受),如果不希望系统检查,删掉.project中的语句——<nature>org.eclipse.wst.common.project.facet.core.nature</nature>,重启eclipse即可。