`

HbernateTemplate类使用详解

 
阅读更多
如何使用HbernateTemplate
HibernateTemplate提供持久层访问模板化,使用HibernateTemplate无须实现特定接口,它只需要提供一个SessionFactory的引用,就可执行持久化操作。SessionFactoyr对象可通过构造参数传入,或通过设值方式传入。如下:
//获取Spring上下文
ApplicationContext ctx = new FileSystemXmlApplicationContext("bean.xml");
//通过上下文获得SessionFactory
SessionFactory sessionFactory = (SessionFactory) ctx.getBean(“sessionFactory”);

然后创建HibernateTemplate实例。HibernateTemplate提供如下三个构造函数
      HibernateTemplate();
      HibernateTemplate(org.hibernate.SessionFactory sessionFactory);
     HibernateTemplate(org.hibernate.SessionFactory sessionFactory, boolean allowCreate);

第一个构造函数,构造一个默认的HibernateTemplate实例,因此,使用HibernateTemplate实例之前,还必须使用方法setSessionFactory(SessionFactory sessionFactory)来为HibernateTemplate传入SessionFactory的引用。
第二个构造函数,在构造时已经传入SessionFactory引用。
第三个构造函数,其boolean型参数表明:如果当前线程已经存在一个非事务性的Session,是否直接返回此非事务性的Session。
对于在Web应用,通常启动时自动加载ApplicationContext,SessionFactory和DAO对象都处在Spring上下文管理下,因此无须在代码中显式设置,可采用依赖注入解耦SessionFactory和DAO,依赖关系通过配置文件来设置,如下所示:
<?xml version="1.0" encoding="gb2312"?>
<!--  Spring配置文件的DTD定义-->
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
    "http://www.springframework.org/dtd/spring-beans.dtd">
<!--  Spring配置文件的根元素是beans-->
<beans>
         <!--定义数据源,该bean的ID为dataSource-->
         <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
                   <!--  指定数据库驱动-->
           <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>
                   <!--  指定连接数据库的URL-->                 
           <property name="url"><value>jdbc:mysql://wonder:3306/j2ee</value></property>
                   <!--  root为数据库的用户名-->
           <property name="username"><value>root</value></property>
                   <!--  pass为数据库密码-->
           <property name="password"><value>pass</value></property>
    </bean>
         <!--定义Hibernate的SessionFactory-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
           <!--  依赖注入数据源,注入正是上文定义的dataSource>
           <property name="dataSource"><ref local="dataSource"/></property>
                   <!--  mappingResouces属性用来列出全部映射文件>
           <property name="mappingResources">
                <list>
                     <!--以下用来列出所有的PO映射文件-->
                                     <value>lee/Person.hbm.xml</value>
                </list>
           </property>
          <!--定义Hibernate的SessionFactory的属性 -->
           <property name="hibernateProperties">
                    <props>
                                     <!--  指定Hibernate的连接方言-->
                                     <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
                                     <!--  不同数据库连接,启动时选择create,update,create-drop-->
                         <prop key="hibernate.hbm2ddl.auto">update</prop>
                    </props>
        </property>
</bean>
<!--  配置Person持久化类的DAO bean-->
<bean id="personDao" class="lee.PersonDaoImpl">
           <!--  采用依赖注入来传入SessionFactory的引用>
           <property name="sessionFactory"><ref local="sessionFactory"/></property>
</bean>
</beans>

DAO实现类中,可采用更简单的方式来取得HibernateTemplate的实例。代码如下:
public class PersnDAOImpl implements PersonDAO
{
    //以私有的成员变量来保存SessionFactory。
private SessionFactory sessionFactory;
         //设值注入SessionFactory必需的setter方法
    public void setSessionFactory(SessionFactory sessionFactory)
{
           this.sessionFactory = sessionFactory;
    }
 
    public List loadPersonByName(final String name)
{
           HibernateTemplate hibernateTemplate =
                    new HibernateTemplate(this.sessionFactory);
                   //此处采用HibernateTemplate完成数据库访问
    }
}

10.6.1 HibernateTemplate的常规用法
HibernateTemplate提供非常多的常用方法来完成基本的操作,比如通常的增加、删除、修改、查询等操作,Spring 2.0更增加对命名SQL查询的支持,也增加对分页的支持。大部分情况下,使用Hibernate的常规用法,就可完成大多数DAO对象的CRUD操作。下面是HibernateTemplate的常用方法简介:
      void delete(Object entity):删除指定持久化实例
      deleteAll(Collection entities):删除集合内全部持久化类实例
      find(String queryString):根据HQL查询字符串来返回实例集合
      findByNamedQuery(String queryName):根据命名查询返回实例集合
      get(Class entityClass, Serializable id):根据主键加载特定持久化类的实例
      save(Object entity):保存新的实例
      saveOrUpdate(Object entity):根据实例状态,选择保存或者更新
      update(Object entity):更新实例的状态,要求entity是持久状态
      setMaxResults(int maxResults):设置分页的大小

下面是一个完整DAO类的源代码:
public class PersonDAOHibernate implements PersonDAO
{
    //采用log4j来完成调试时的日志功能
private static Log log = LogFactory.getLog(NewsDAOHibernate.class);
    //以私有的成员变量来保存SessionFactory。
private SessionFactory sessionFactory;
//以私有变量的方式保存HibernateTemplate
private HibernateTemplate hibernateTemplate = null;
         //设值注入SessionFactory必需的setter方法
    public void setSessionFactory(SessionFactory sessionFactory)
{
           this.sessionFactory = sessionFactory;
    }
         //初始化本DAO所需的HibernateTemplate
public HIbernateTemplate getHibernateTemplate()
{
         //首先,检查原来的hibernateTemplate实例是否还存在
         if ( hibernateTemplate == null)
         {
                   //如果不存在,新建一个HibernateTemplate实例
                   hibernateTemplate = new HibernateTemplate(sessionFactory);
         }
         return hibernateTemplate;
}
         //返回全部的人的实例
    public List getPersons()
         {       
                   //通过HibernateTemplate的find方法返回Person的全部实例
           return getHibernateTemplate().find("from Person");
    }
         /**
          * 根据主键返回特定实例
          * @ return 特定主键对应的Person实例
          * @ param 主键值
    public News getNews(int personid)
         {
           return (Person)getHibernateTemplate().get(Person.class, new Integer(personid));
    }
         /**
          * @ person 需要保存的Person实例
          */
    public void savePerson(Person person)
         {                
                   getHibernateTemplate().saveOrUpdate(person);
    }
         /**
          * @ param personid 需要删除Person实例的主键
          * /
    public void removePerson(int personid)
         {
                   //先加载特定实例
           Object p = getHibernateTemplate().load(Person.class, new Integer(personid));
                   //删除特定实例
           getHibernateTemplate().delete(p);
    }
}

10.6.2 Hibernate的复杂用法HibernateCallback
HibernateTemplate还提供一种更加灵活的方式来操作数据库,通过这种方式可以完全使用Hibernate的操作方式。HibernateTemplate的灵活访问方式是通过如下两个方法完成:
      Object execute(HibernateCallback action)
      List execute(HibernateCallback action)
这两个方法都需要一个HibernateCallback的实例,HibernateCallback实例可在任何有效的Hibernate数据访问中使用。程序开发者通过HibernateCallback,可以完全使用Hibernate灵活的方式来访问数据库,解决Spring封装Hibernate后灵活性不足的缺陷。HibernateCallback是一个接口,该接口只有一个方法doInHibernate(org.hibernate.Session session),该方法只有一个参数Session。
通常,程序中采用实现HibernateCallback的匿名内部类来获取HibernateCallback的实例,方法doInHibernate的方法体就是Spring执行的持久化操作。具体代码如下:
public class PersonDaoImpl implements PersonDao
{
    //私有实例变量保存SessionFactory
         private SessionFactory sessionFactory;
         //依赖注入必须的setter方法
    public void setSessionFactory(SessionFactory sessionFactory)
         {
           this.sessionFactory = sessionFactory;
    }
         /**
          * 通过人名查找所有匹配该名的Person实例
          * @param name 匹配的人名
          * @return 匹配该任命的全部Person集合
          */
    public List findPersonsByName(final String name)
         {
           //创建HibernateTemplate实例
                   HibernateTemplate hibernateTemplate =
                         new HibernateTemplate(this.sessionFactory);
           //返回HibernateTemplate的execute的结果
                   return (List) hibernateTemplate.execute(
                //创建匿名内部类
                   new HibernateCallback()
                   {
              public Object doInHibernate(Session session) throws HibernateException
                            {
                   //使用条件查询的方法返回
                                     List result = session.createCriteria(Person.class)
                                                                     .add(Restrictions.like(“name”, name+”%”)
                                                                           .list();
                                    return result;
                     }
                });
    }
}

注意:方法doInHibernate方法内可以访问Session,该Session对象是绑定到该线程的Session实例。该方法内的持久层操作,与不使用Spring时的持久层操作完全相同。这保证对于复杂的持久层访问,依然可以使用Hibernate的访问方式。
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics