Spring数据库访问之ORM(一)

  Spring数据库访问中另外一大模块就是ORM,ORM即对象/关系映射。Spring支持大多数ORM框架,比如Hibernate,JPA,JDO,TopLink和iBatis(Spring2支持iBatis2,现MyBatis3的Spring支持由MyBatis社区开发,并非Spring)。 
    首先我们从单独使用ORM框架开始,来介绍Spring对ORM的支持,以Hibernate为例。使用ORM框架,需要提供持久化类,以课程管理为背景,如下设计课程类: 
Java代码   收藏代码
  1. package org.ourpioneer.course.bean;  
  2. import java.sql.Date;  
  3. /** 
  4.  * 课程信息描述bean 
  5.  *  
  6.  * @author Nanlei 
  7.  *  
  8.  */  
  9. public class Course {  
  10.     private Long id;  
  11.     private String title;  
  12.     private java.util.Date startDate;  
  13.     private java.util.Date endDate;  
  14.     private int fee;  
  15.     //必须提供无参默认构造方法  
  16.     public Course() {  
  17.         super();  
  18.     }  
  19.     //省略其它构造方法,getter和setter等方法  
  20. }  

    作为对数据访问的一种模式,我们仍然应用DAO模式,写DAO类已经很熟练了,如下设计即可: 
Java代码   收藏代码
  1. package org.ourpioneer.course.dao;  
  2. import java.util.List;  
  3. import org.ourpioneer.course.bean.Course;  
  4. public interface CourseDAO {  
  5.     public void save(Course course);  
  6.     public void delete(Course course);  
  7. public void update(Course course);  
  8.     public Course findById(Long courseId);  
  9.     public List<Course> findAll();  
  10. }  

    非常简单的设计,包含CRUD操作,那么实现类中我们用Hibernate帮助我们进行数据库访问操作也非常简单: 
Java代码   收藏代码
  1. package org.ourpioneer.course.dao;  
  2. import java.util.List;  
  3. import org.hibernate.Query;  
  4. import org.hibernate.Session;  
  5. import org.hibernate.SessionFactory;  
  6. import org.hibernate.Transaction;  
  7. import org.hibernate.cfg.Configuration;  
  8. import org.ourpioneer.course.bean.Course;  
  9. public class CourseDAOImpl implements CourseDAO {  
  10.     private SessionFactory sessionFactory;  
  11.     public CourseDAOImpl() {  
  12.         Configuration cfg = new Configuration().configure();  
  13.         sessionFactory = cfg.buildSessionFactory();  
  14.     }  
  15.     public List<Course> findAll() {  
  16.         Session session = sessionFactory.openSession();  
  17.         try {  
  18.             Query query = session.createQuery("from Course");  
  19.             return query.list();  
  20.         } finally {  
  21.             session.close();  
  22.         }  
  23.     }  
  24.     public Course findById(Long courseId) {  
  25.         Session session = sessionFactory.openSession();  
  26.         try {  
  27.             return (Course) session.get(Course.class, courseId);  
  28.         } finally {  
  29.             session.close();  
  30.         }  
  31.     }  
  32.     public void save(Course course) {  
  33.         Session session = sessionFactory.openSession();  
  34.         Transaction tx = session.beginTransaction();  
  35.         try {  
  36.             tx.begin();  
  37.             session.saveOrUpdate(course);  
  38.             tx.commit();  
  39.         } catch (RuntimeException e) {  
  40.             tx.rollback();  
  41.             throw e;  
  42.         } finally {  
  43.             session.close();  
  44.         }  
  45.     }  
  46. }  

    这里只展示几个方法作为代表,其它方法类似可以写出,非常简单。首先是构造方法,初始化实现类时创建Hibernate的配置对象,new Configuration().configure()时,Hibernate会在类路径的根路径下自动寻找名为hibernate.cfg.xml的配置文件并加载,之后就是创建Hibernate的Session对象,利用Session对象提供和衍生出的方法来进行数据库操作。下面来看配置文件,这是比较重要的,因为通过配置文件,把数据库信息和实体Bean的信息都告诉Hibernate,可以省去我们很多在数据库设计上的事情。 
Xml代码   收藏代码
  1. <?xml version='1.0' encoding='UTF-8'?>  
  2. <!DOCTYPE hibernate-configuration PUBLIC  
  3.           "-//Hibernate/Hibernate Configuration DTD 3.0//EN"  
  4.   "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">  
  5. <hibernate-configuration>  
  6.     <session-factory>  
  7.         <property name="connection.driver_class">com.mysql.jdbc.Driver</property>  
  8.         <property name="connection.url">jdbc:mysql:///test</property>  
  9.         <property name="connection.username">root</property>  
  10.         <property name="connection.password">123</property>  
  11.         <property name="dialect">org.hibernate.dialect.MySQLDialect</property>  
  12.         <property name="show_sql">true</property>  
  13.         <property name="hbm2ddl.auto">update</property>  
  14.         <mapping resource="org/ourpioneer/course/hbm/course.hbm.xml" />  
  15.     </session-factory>  
  16. </hibernate-configuration>  

    这里我们告诉Hibernate使用mysql数据库,并配置数据库信息,所用方言,并在执行应用程序时在控制台打印出还原的SQL语句。使用hbm2ddl.auto可以让Hibernate根据实体Bean的配置信息来自动建表,这是很方便的,最后的mapping就是配置实体bean映射信息的文件,我们来看一下: 
Xml代码   收藏代码
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"  
  3. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">  
  4. <hibernate-mapping package="org.ourpioneer.course.bean">  
  5.     <class name="Course" table="course">  
  6.         <id name="id" type="java.lang.Long" column="ID">  
  7.             <generator class="identity" />  
  8.         </id>  
  9.         <property name="title" type="java.lang.String" column="TITLE"  
  10.             length="100" not-null="true" />  
  11.         <property name="startDate" type="java.sql.Date" column="STARTDATE"  
  12.             not-null="true" />  
  13.         <property name="endDate" type="java.sql.Date" column="ENDDATE"  
  14.             not-null="true" />  
  15.         <property name="fee" type="java.lang.Integer" column="FEE"  
  16.             not-null="true" />  
  17.     </class>  
  18. </hibernate-mapping>  

    这里首先配置了实体bean的所在位置,类信息还有数据库中表对应的关系。之后创建主键信息id,使用了MySQL的自增主键特性,剩下就是对各个字段的描述了,都见名知意,易于理解。 
    准备好这些内容,就可以编写测试程序了,在项目中引入Hibernate的相关依赖。这里说明一下,Maven默认仓库是Apache的,其中的Hibernate版本还在3.3.2.GA(本文编写时),而当前的Hibernate官方版本已经是3.6.0.Final了,我们想使用新版本,该怎么办?很简单,配置Maven仓库的位置,让其可以发现3.6.0.Final版的Hibernate并下载依赖。JBoss官方也提供一个Maven仓库,其中就有最新版的Hibernate,那么我们在项目的POM中配置一下这个地址: 
Xml代码   收藏代码
  1. <repositories>  
  2.     <repository>  
  3.         <releases>  
  4.             <updatePolicy>always</updatePolicy>  
  5.         </releases>  
  6.         <snapshots>  
  7.             <updatePolicy>always</updatePolicy>  
  8.         </snapshots>  
  9.         <id>Jboss</id>  
  10.         <name>Jboss Repository</name>  
  11. <url>https://repository.jboss.org/nexus/content/groups/public</url>  
  12.     </repository>  
  13. </repositories>  

    之后,为项目引入其它必要的依赖,使用Maven管理,我们不需自己再去寻找各种依赖了,非常简单的管理,如图所示: 
Spring数据库访问之ORM(一)_第1张图片 
    下面来看示例程序: 
Java代码   收藏代码
  1. package org.ourpioneer.course;  
  2. import java.util.GregorianCalendar;  
  3. import java.util.List;  
  4. import org.ourpioneer.course.bean.Course;  
  5. import org.ourpioneer.course.dao.CourseDAO;  
  6. import org.ourpioneer.course.dao.CourseDAOImpl;  
  7. public class Demo {  
  8.     public static void main(String[] args) {  
  9.         CourseDAO courseDAO = new CourseDAOImpl();  
  10.         Course course = new Course();  
  11.         course.setTitle("Spring ORM");  
  12.         course.setStartDate(new GregorianCalendar(201111).getTime());  
  13.         course.setEndDate(new GregorianCalendar(201121).getTime());  
  14.         course.setFee(100);  
  15.         courseDAO.save(course);  
  16.         List<Course> courses = courseDAO.findAll();  
  17.         Long courseId = courses.get(0).getId();  
  18.         course = courseDAO.findById(courseId);  
  19.         System.out.println(course);  
  20.         courseDAO.delete(course);  
  21.     }  
  22. }  

    首先创建的是Course对象,并设置其中的属性,使用save方法将其持久化到数据库中,之后通过findAll方法查询数据库中的全部记录,当然现在只有一条。并拿到Id,在通过findById方法获取出来,然后打印结果。最终删除记录。执行该程序,我们可以得到如下输出信息: 
Spring数据库访问之ORM(一)_第2张图片 
    我们之前并没有在数据库中建表,而Hibernate在执行插入之前会为我们自动建表,然后执行插入操作,两次查询操作,并打印出对象信息,最后执行了删除操作,从SQL语句中可以看到Hibernate最终的执行结果是什么。而此时回到数据库中,会发现一个建好的表。 
Spring数据库访问之ORM(一)_第3张图片 
    Hibernate的简单ORM映射操作就介绍完了,下面来看使用JPA注解和Hibernate的API来持久化对象,首先修改持久化类: 
Java代码   收藏代码
  1. package org.ourpioneer.course.bean;  
  2. import java.sql.Date;  
  3. import javax.persistence.Column;  
  4. import javax.persistence.Entity;  
  5. import javax.persistence.GeneratedValue;  
  6. import javax.persistence.GenerationType;  
  7. import javax.persistence.Id;  
  8. import javax.persistence.Table;  
  9.   
  10. /** 
  11.  * 课程信息描述bean 
  12.  *  
  13.  * @author Nanlei 
  14.  *  
  15.  */  
  16. @Entity  
  17. @Table(name = "course")  
  18. public class Course {  
  19.     @Id  
  20.     @GeneratedValue(strategy = GenerationType.IDENTITY)  
  21.     @Column(name = "ID")  
  22.     private Long id;  
  23.     @Column(name = "TITLE", length = 100, nullable = false)  
  24.     private String title;  
  25.     @Column(name = "STARTDATE",nullable=false)  
  26.     private java.util.Date startDate;  
  27.     @Column(name = "ENDDATE",nullable=false)  
  28.     private java.util.Date endDate;  
  29.     @Column(name = "FEE",nullable=false)  
  30.     private int fee;  
  31.     // 其余内容不变,省略  
  32. }  

    使用JPA的注解,首先对类进行注解,使用@Entity,并关联数据库表,使用@Table。下面就是对字段进行主键了,标识符(主键)字段要使用@Id,还要指定生成策略和对应的列名,剩下的字段只需指定列信息即可。现在告诉Hibernate我们使用JPA注解,而不使用映射文件了,如下配置: 
Xml代码   收藏代码
  1. <!--  
  2. <mapping resource="org/ourpioneer/course/hbm/course.hbm.xml" /> 
  3.  -->  
  4. <mapping class="org.ourpioneer.course.bean.Course"/>  

    修改DAO实现类的构造方法,使用注解配置方式创建SessionFactory,如下即可: 
Java代码   收藏代码
  1. public CourseDAOImpl() {  
  2.     // Configuration cfg = new Configuration().configure();  
  3.     Configuration cfg = new AnnotationConfiguration().configure();  
  4.     sessionFactory = cfg.buildSessionFactory();  
  5. }  

    此时再次执行测试方法,反馈的信息没有任何变化,但是我们就使用了JPA注解而并非Hibernate的映射信息了。下面来看看使用Hibernate作为JPA引擎的持久化步骤。先配置依赖,引入: 
Xml代码   收藏代码
  1. <dependency>  
  2.     <groupId>org.hibernate</groupId>  
  3.     <artifactId>hibernate-entitymanager</artifactId>  
  4.     <version>3.6.0.Final</version>  
  5.     <type>jar</type>  
  6.     <scope>compile</scope>  
  7. </dependency>  
  8. <dependency>  
  9.     <groupId>jboss</groupId>  
  10.     <artifactId>jboss-archive-browsing</artifactId>  
  11.     <version>5.0.0alpha-200607201-119</version>  
  12.     <type>jar</type>  
  13.     <scope>compile</scope>  
  14. </dependency>  

    如果在Java EE容器中运行JPA,可以通过容器来配置JPA,如果是在Java SE中运行JPA,那么需要在类路径的META-INF下配置persistence.xml来配置持久化单元,在本例中我们使用Hibernate作为JPA的引擎,就可以这么来写: 
Xml代码   收藏代码
  1. <?xml version='1.0' encoding='UTF-8'?>  
  2. <persistence xmlns="http://java.sun.com/xml/ns/persistence"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  4.     xsi:schemaLocation="http://java.sun.com/xml/ns/persistence  
  5. http://java.sun.com/xml/ns/persistence/persistence_1_0.xsd"  
  6.     version="1.0">  
  7.     <persistence-unit name="course">  
  8.         <properties>  
  9.             <property name="hibernate.ejb.cfgfile" value="/hibernate.cfg.xml" />  
  10.         </properties>  
  11.     </persistence-unit>  
  12. </persistence>  

    加载配置文件仍然使用Hibernate的配置文件,或者也可以将其中的property写在persistence.xml中,那么因为在这里JPA已经可以获取持久化单元了,在Hibernate的配置中就需要配置持久化对象的映射了,去掉mapping信息即可。有了配置信息,还需要实现类,我们重写一个JPA的DAO实现,如下: 
Java代码   收藏代码
  1. package org.ourpioneer.course.dao;  
  2. import java.util.List;  
  3. import javax.persistence.EntityManager;  
  4. import javax.persistence.EntityManagerFactory;  
  5. import javax.persistence.EntityTransaction;  
  6. import javax.persistence.Persistence;  
  7. import javax.persistence.Query;  
  8. import org.ourpioneer.course.bean.Course;  
  9. public class CourseDAOImplJPA implements CourseDAO {  
  10.     private EntityManagerFactory entityManagerFactory;  
  11.     public CourseDAOImplJPA() {  
  12.         entityManagerFactory = Persistence.createEntityManagerFactory("course");  
  13.     }  
  14.     public void delete(Course course) {  
  15.         EntityManager manager = entityManagerFactory.createEntityManager();  
  16.         EntityTransaction tx = manager.getTransaction();  
  17.         try {  
  18.             tx.begin();  
  19.             manager.remove(manager.merge(course));  
  20.             tx.commit();  
  21.         } catch (RuntimeException e) {  
  22.             tx.rollback();  
  23.             throw e;  
  24.         } finally {  
  25.             manager.close();  
  26.         }  
  27.     }  
  28.     public List<Course> findAll() {  
  29.         EntityManager manager = entityManagerFactory.createEntityManager();  
  30.         try {  
  31.             Query query = manager  
  32.                     .createQuery("select course from Course course");  
  33.             return query.getResultList();  
  34.         } finally {  
  35.             manager.close();  
  36.         }  
  37.     }  
  38.     public Course findById(Long courseId) {  
  39.         EntityManager manager = entityManagerFactory.createEntityManager();  
  40.         try {  
  41.             return manager.find(Course.class, courseId);  
  42.         } finally {  
  43.             manager.close();  
  44.         }  
  45.     }  
  46.     public void save(Course course) {  
  47.         EntityManager manager = entityManagerFactory.createEntityManager();  
  48.         EntityTransaction tx = manager.getTransaction();  
  49.         try {  
  50.             tx.begin();  
  51.             manager.persist(course);  
  52.             tx.commit();  
  53.         } catch (RuntimeException e) {  
  54.             tx.rollback();  
  55.             throw e;  
  56.         } finally {  
  57.             manager.close();  
  58.         }  
  59.     }  
  60.     public void update(Course course) {  
  61.         EntityManager manager = entityManagerFactory.createEntityManager();  
  62.         EntityTransaction tx = manager.getTransaction();  
  63.         try {  
  64.             tx.begin();  
  65.             manager.merge(course);  
  66.             tx.commit();  
  67.         } catch (RuntimeException e) {  
  68.             tx.rollback();  
  69.             throw e;  
  70.         } finally {  
  71.             manager.close();  
  72.         }  
  73.     }  
  74. }  

    这里特别要注意的是delete方法,首先调用merge方法,否则当前对象是出于脱管态的,无法和Session进行关联,也就无法删除该对象。不加merge方法时候会抛出异常,大家可以测试一下,因为底层还是Hibernate进行的,Hibernate的持久化对象有三种状态,那么就要注意状态发生的变化。 

你可能感兴趣的:(spring,数据库,orm)