准备好了单元测试模块,就写个底层的数据访问的控制Handle类来试试吧。也就是通常我们所说的DAO类。一般而言,每个模型/实体类需要建立一个DAO类。 不过经验告诉我们,一些通用的CRUD方法几乎在每个DAO类中都是一样的,那么也就是说我们通常所写的DAO会有些代码冗余的问题,而且根据业务需要,日后的维护工作有可能添加一些新的实体类,写地越多,DAO类就越多,写的代码就越多余还浪费时间。如何将写好的一个DAO类进行代码重用成为了问题。其实笔者见过一个的系统里有做过将Hibernate框架人应用做成静态类,用到时直接HibernateUtil.save()的。今天,咱也做一个通用的DAO放一些通用的CRUD方法进去,用来做通用的CRUD操作。这样做既省去了不少代码量,加强了代码可读性,而且还方便快捷地架构起这个WebModel,可以重用到很多的系统中。当然,遇到特殊的Entity 需要建个别的DAO时再当别论。

        首先是接口的编写,我在接口的编写时加了很多的JavaDoc,就不贴出代码了,上传为附件方便读者下载吧。此处列出几个方法做简单说明

 
  1. package com.xxxxx.webmodel.pin; 
  2.  
  3. import java.util.List; 
  4. import java.util.Map; 
  5.       public interface IPersistencePin { 
  6.     public void createPersistingEntity(Object persistingObj) throws Exception; 
  7.     public <PerE> PerE retrievePersistedEntity(Class<PerE> persistedClass, 
  8.             long id) throws Exception; 
  9.     public <PerE> PerE retrievePersistedEntity(Class<PerE> persistedClass, 
  10.             String identifier) throws Exception; 
  11.     public <PerE> List<PerE> retrievePersistedEntityAllInList( 
  12.             Class<PerE> persistedClass) throws Exception; 
  13.     public void updatePersistedEntity(Object persistingObj) throws Exception; 
  14.     public void deletePersistedEntity(Object persistingObj) throws Exception; 

      接口的实现:

 
  1. package com.xxxxx.webmodel.pin.impl; 
  2.  
  3. /*省略多个import 语句*/ 
  4. import com.gxino.webmodel.pin.IPersistencePin; 
  5. @Repository 
  6. public class HibernatePersistencePin implements Serializable,IPersistencePin { 
  7.     private static final long serialVersionUID = -295800418896667101L; 
  8.     private SessionFactory sessionFactory; 
  9.     public static long getSerialversionuid() { 
  10.         return serialVersionUID; 
  11.     } 
  12.  
  13.     public SessionFactory getSessionFactory() { 
  14.         return sessionFactory; 
  15.     } 
  16.     @Resource 
  17.     public void setSessionFactory(SessionFactory sessionFactory) { 
  18.         this.sessionFactory = sessionFactory; 
  19.     } 
  20. }    
 

       其中,第一个方法 createPersistingEntity(Object persistingObj) MapHibernate 中的任意Pojo的对象持久化,即将某实体类的一个对象的各项属性值存入数据库。

       

 
  1. @Override 
  2.   public void createPersistingEntity(Object persistingObj) throws Exception { 
  3.     Session session = null
  4.     boolean needCloseSession = false
  5.     try
  6.         session = sessionFactory.getCurrentSession(); 
  7.         if(session==null)throw new HibernateException("No current Session"); 
  8.     }catch(HibernateException he){ 
  9.         session = sessionFactory.openSession(); 
  10.         needCloseSession = true
  11.     } 
  12.     try
  13.         session.save(persistingObj); 
  14.         if(needCloseSession)session.close(); 
  15.     }catch(HibernateException he){ 
  16.         throw new Exception("Save Pojo failed, because of "+he.getMessage()); 
  17.     } 

       笔者在该方法中做了currentSessionopenSession双处理,也就是说该方法既可以是业务逻辑进行事务中的某一句,也可以无事务信赖地被单独调用。后面的update 方法与delete方法以及通过id从数据库中查出该对象的方法是如出一辙,只是session.save(persistingOjb)换成其他语句,就不贴代码了。贴一下另两个方法的第二个Try块中的代码吧:

 

1<PerE> PerE retrievePersistedEntity(Class<PerE> persistedClass,

String uniquePropertyName, Object propertyValue)

传入一个唯一键值的名值对找出类对象: 

 
  1. Criteria criteria = session.createCriteria(persistedClass); 
  2. criteria.add(Restrictions.eq(uniquePropertyName, propertyValue)); 
  3. @SuppressWarnings("unchecked"
  4. PerE result = (PerE)criteria.uniqueResult(); 
  5. if(needCloseSession)session.close(); 
  6. return result; 

 

       这里用到了HibernateCriteriaRestrictions,其实还有个Example类。 关于这一点,网络上有好多高手有解盲它的专题用法的帖子,笔者就不作大篇文章说它了。感觉这个比起写hql更 『面向对象』 一点儿,好用且易理解,所以特意贴在这里一个,以提醒自己与读者们可以这样去写。其实用了它的代码都是白话文,简单提一下: 从Session中创建一个某pojo人“基准”,用一些 “条件”,或是举例的方式,限定这个查询基准,之后按这个限定的标准查出具体对象。

      查找所有对象到一个List中也不多说了,criteriaadd任何限制,直接criteria.list();就好.

 

2<PerE> PerE retrievePersistedEntity(Class<PerE> persistedClass,

String identifier) throws Exception;

      无需给出唯一键的名,只需要唯一键的值,即去查询到该实体类对象。不过由于一个表中,可能有多个唯一键,这样就会产生冲突。例如,用户名和Email两个唯一性字段,用户Aaaa@aaa.com作用户名,用户Baaa@aaa.comemail,这种情况下查询到用户的结果就不仅是A而且还有B了。所以笔者将它在接口里就标了@deprecated

      同样,只贴第二个try块中应放的代码。

 
  1. String propertyName=null
  2. Method[] methodArray =persistedClass.getMethods(); 
  3. if(methodArray!=null)for(Method method:methodArray){ 
  4.     if(method.getName().startsWith("get")){ 
  5.         Criteria criteria = session.createCriteria(persistedClass); 
  6.         Annotation anno = method.getAnnotation(Column.class); 
  7.         if(anno.toString().contains("unique=true")){ 
  8.             propertyName = Character.toString(method.getName().charAt(3)).toLowerCase()+method.getName().substring(4); 
  9.             criteria.add(Restrictions.eq(propertyName, identifier)); 
  10.         } 
  11.         @SuppressWarnings("unchecked"
  12.         PerE result = (PerE)criteria.uniqueResult(); 
  13.         if(needCloseSession)session.close(); 
  14.         return result; 
  15.     } 
  16. if(needCloseSession)session.close(); 
  17. return null
 

      这方法的实现需要javax.persistence.Column这一注解到实体类中的每一个属性做配合。换句话说,HibernateMapping是注解方式的。 Column注解在JPA的规范中, hibernate框架有JPA的实现,在hibernate-jpa-2.0-api-x.x.x-Final.jar包中。 上面代码用到了反射,主要意思是:找到该Pojo类中所有属性的getter,通过查看该getter上是否注解有@Column并且注解的一个属性unique值为true,来找到相应的属性名,然后用名值对限定查询标准去查出实体类对象。

      至于怎么用JPA注解配置Pojo,也可以作为专题发表一篇文章了,笔者就不多叙述,直接在附件夹带一些实体类了,实体类如何mapsessionFactory中也在第一篇《xxx 注解化配置》的ApplicationContext.java中出现,代码在配置sessionFactory @Bean的方法中:

 
  1. @SuppressWarnings("rawtypes"
  2. Class[] entities = new Class[2]; 
  3. entities[0] = AccountEntity.class
  4. entities[1] = ActionLogEntity.class
  5. asfBean.setAnnotatedClasses(entities); 

 

      另外,有些读者可能已经发现,上述通用DAO中的实现中,每个方法也几乎重用了好多地方,只是改变了中间的核心部分,可不可以用一个固定的方法将每个方法的核心语句包住,就像是让Spring代管Transactional一样,很明显,我们会想到AOP,顺便将log也做到AOP中,不过这一篇就不多讨论了。先就这样用着。

      最后,别忘了写单元测试类。

 

 
 
  1. package com.xxxxx.webmodel.test.pin; 
  2.  
  3. import static org.junit.Assert.fail; 
  4.  
  5. import java.util.Date; 
  6. import java.util.HashMap; 
  7. import java.util.List; 
  8. import java.util.Map; 
  9.  
  10. import javax.annotation.Resource; 
  11.  
  12. import org.junit.Assert; 
  13. import org.junit.Before; 
  14. import org.junit.BeforeClass; 
  15. import org.junit.Ignore; 
  16. import org.junit.Test; 
  17. import org.junit.runner.RunWith; 
  18. import org.springframework.test.context.ContextConfiguration; 
  19. import org.springframework.test.context.TestExecutionListeners; 
  20. import org.springframework.test.context.junit4.SpringJUnit4Cla***unner; 
  21. import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; 
  22. import org.springframework.test.context.support.DirtiesContextTestExecutionListener; 
  23. import org.springframework.test.context.transaction.TransactionConfiguration; 
  24. import org.springframework.test.context.transaction.TransactionalTestExecutionListener; 
  25. import org.springframework.transaction.annotation.Transactional; 
  26.  
  27. import com.xxxxx.webmodel.entity.AccountEntity; 
  28. import com.xxxxx.webmodel.entity.ActionLogEntity; 
  29. import com.xxxxx.webmodel.pin.IPersistencePin; 
  30. import com.xxxxx.webmodel.util.ApplicationContext; 
  31. import com.xxxxx.webmodel.util.WebConfiguration; 
  32. @RunWith(SpringJUnit4Cla***unner.class
  33. @ContextConfiguration(classes={ApplicationContext.class}) 
  34. @TestExecutionListeners( { DependencyInjectionTestExecutionListener.class, DirtiesContextTestExecutionListener.class, TransactionalTestExecutionListener.class }) 
  35. @TransactionConfiguration(defaultRollback=true,transactionManager="hibernateTransactionManager"
  36. @Transactional 
  37. public class PersistencePinTest { 
  38.     private IPersistencePin persistPin; 
  39.     private Map<String,Object> usedStuff; 
  40.      
  41.     public IPersistencePin getPersistPin() { 
  42.         return persistPin; 
  43.     } 
  44.     @Resource 
  45.     public void setPersistPin(IPersistencePin persistPin) { 
  46.         this.persistPin = persistPin; 
  47.     } 
  48.      
  49.     @BeforeClass 
  50.     public static void init() throws Exception{ 
  51.             new WebConfiguration().onStartup(null); 
  52.     } 
  53.      
  54.     @Before 
  55.     public void setUp() throws Exception{ 
  56.         if(usedStuff==null
  57.             usedStuff = new HashMap<String,Object>(); 
  58.         AccountEntity account = new AccountEntity(); 
  59.         account.setUserName("junittest"); 
  60.         account.setPassword("junittest"); 
  61.         account.setEmail("junittest@junit.com"); 
  62.         account.setAlias("unitTest"); 
  63.         account.setAge((short)18); 
  64.         account.setGender("Male"); 
  65.         account.setCreatedAt(new Date()); 
  66.         persistPin.createPersistingEntity(account); 
  67.         usedStuff.put("userId"new Long(account.getId())); 
  68.         usedStuff.put("userName", account.getUserName()); 
  69.         usedStuff.put("userEmail", account.getEmail()); 
  70.          
  71.         ActionLogEntity actionLog = new ActionLogEntity(); 
  72.         actionLog.setAction("Junit Test"); 
  73.         actionLog.setActor(account); 
  74.         actionLog.setHappenTime(new Date()); 
  75.         try{persistPin.createPersistingEntity(actionLog);} 
  76.         catch(Exception e){ 
  77.             e.printStackTrace(); 
  78.             } 
  79.         usedStuff.put("actionId"new Long(actionLog.getId())); 
  80.         usedStuff.put("actionTime", actionLog.getHappenTime()); 
  81.     } 
  82.  
  83.     @Test 
  84.     @Ignore 
  85.     public final void testHibernateSession() throws Exception{ 
  86.         /*Please not execute @Before setUp() method, for test this, and NO              @TransactionConfiguration NO @Transactional*/ 
  87.         /*Also make other test methods @Ignore*/ 
  88.         AccountEntity account = new AccountEntity(); 
  89.         account.setUserName("junittest"); 
  90.         account.setPassword("junittest"); 
  91.         account.setEmail("junittest@junit.com"); 
  92.         account.setAlias("unitTest"); 
  93.         account.setAge((short)18); 
  94.         account.setGender("Male"); 
  95.         account.setCreatedAt(new Date()); 
  96.         persistPin.createPersistingEntity(account); 
  97.         persistPin.deletePersistedEntity(account); 
  98.     } 
 

具体的其他测试方法在附件的文件中,或是读者们自行去写会比笔者的要好很多。 

      附件中有源码,在很下面哦,请您仔细找一下