您的位置:軟件測(cè)試 > 開(kāi)源軟件測(cè)試 > 開(kāi)源單元測(cè)試工具 >
對(duì)DAO編寫單元測(cè)試
作者:網(wǎng)絡(luò)轉(zhuǎn)載 發(fā)布時(shí)間:[ 2013/2/19 15:12:51 ] 推薦標(biāo)簽:

  單元測(cè)試作為保證軟件質(zhì)量及重構(gòu)的基礎(chǔ),早已獲得廣大開(kāi)發(fā)人員的認(rèn)可。單元測(cè)試是一種細(xì)粒度的測(cè)試,越來(lái)越多的開(kāi)發(fā)人員在提交功能模塊時(shí)也同時(shí)提交相應(yīng)的單元測(cè)試。對(duì)于大多數(shù)開(kāi)發(fā)人員來(lái)講,編寫單元測(cè)試已經(jīng)成為開(kāi)發(fā)過(guò)程中必須的流程和佳實(shí)踐。

  對(duì)普通的邏輯組件編寫單元測(cè)試是一件容易的事情,由于邏輯組件通常只需要內(nèi)存資源,因此,設(shè)置好輸入輸出即可編寫有效的單元測(cè)試。對(duì)于稍微復(fù)雜一點(diǎn)的組件,例如Servlet,我們可以自行編寫模擬對(duì)象,以便模擬HttpRequest和HttpResponse等對(duì)象,或者,使用EasyMock之類的動(dòng)態(tài)模擬庫(kù),可以對(duì)任意接口實(shí)現(xiàn)相應(yīng)的模擬對(duì)象,從而對(duì)依賴接口的組件進(jìn)行有效的單元測(cè)試。

  在J2EE開(kāi)發(fā)中,對(duì)DAO組件編寫單元測(cè)試往往是一件非常復(fù)雜的任務(wù)。和其他組件不通,DAO組件通常依賴于底層數(shù)據(jù)庫(kù),以及JDBC接口或者某個(gè)ORM框架(如Hibernate),對(duì)DAO組件的測(cè)試往往還需引入事務(wù),這更增加了編寫單元測(cè)試的復(fù)雜性。雖然使用EasyMock也可以模擬出任意的JDBC接口對(duì)象,或者ORM框架的主要接口,但其復(fù)雜性往往非常高,需要編寫大量的模擬代碼,且代碼復(fù)用度很低,甚至不如直接在真實(shí)的數(shù)據(jù)庫(kù)環(huán)境下測(cè)試。不過(guò),使用真實(shí)數(shù)據(jù)庫(kù)環(huán)境也有一個(gè)明顯的弊端,我們需要準(zhǔn)備數(shù)據(jù)庫(kù)環(huán)境,準(zhǔn)備初始數(shù)據(jù),并且每次運(yùn)行單元測(cè)試后,其數(shù)據(jù)庫(kù)現(xiàn)有的數(shù)據(jù)將直接影響到下一次測(cè)試,難以實(shí)現(xiàn)“即時(shí)運(yùn)行,反復(fù)運(yùn)行”單元測(cè)試的良好實(shí)踐。

  本文針對(duì)DAO組件給出一種較為合適的單元測(cè)試的編寫策略。在JavaEE開(kāi)發(fā)網(wǎng)(http://www.javaeedev.com)的開(kāi)發(fā)過(guò)程中,為了對(duì)DAO組件進(jìn)行有效的單元測(cè)試,我們采用HSQLDB這一小巧的純Java數(shù)據(jù)庫(kù)作為測(cè)試時(shí)期的數(shù)據(jù)庫(kù)環(huán)境,配合Ant,實(shí)現(xiàn)了自動(dòng)生成數(shù)據(jù)庫(kù)腳本,測(cè)試前自動(dòng)初始化數(shù)據(jù)庫(kù),極大地簡(jiǎn)化了DAO組件的單元測(cè)試的編寫。
在Java領(lǐng)域,JUnit作為第一個(gè)單元測(cè)試框架已經(jīng)獲得了廣泛的應(yīng)用,無(wú)可爭(zhēng)議地成為Java領(lǐng)域單元測(cè)試的標(biāo)準(zhǔn)框架。本文以新的JUnit 4版本為例,演示如何創(chuàng)建對(duì)DAO組件的單元測(cè)試用例。

  JavaEEdev的持久層使用Hibernate 3.2,底層數(shù)據(jù)庫(kù)為MySQL。為了演示如何對(duì)DAO進(jìn)行單元測(cè)試,我們將其簡(jiǎn)化為一個(gè)DAOTest工程:
對(duì)DAO編寫單元測(cè)試 圖-1

  由于將Hibernate的Transaction綁定在Thread上,因此,HibernateUtil類負(fù)責(zé)初始化SessionFactory以及獲取當(dāng)前的Session:

 

 public class HibernateUtil {

  private static final  SessionFactory sessionFactory;

  static {

  try {

  sessionFactory =  new AnnotationConfiguration()

  .configure()

  .buildSessionFactory();

  }

  catch(Exception e) {

  throw new  ExceptionInInitializerError(e);

  }

  }
  public static Session  getCurrentSession() {

  return  sessionFactory.getCurrentSession();

  }

  }
 

  HibernateUtil還包含了一些輔助方法,如:

 public static Object query(Class clazz, Serializable id);

  public static void createEntity(Object entity);

  public static Object queryForObject(String hql, Object[] params);

  public static List queryForList(String hql, Object[] params);
 

  在此不再多述。

  實(shí)體類User使用JPA注解,代表一個(gè)用戶:

  @Entity

  @Table(name="T_USER")

  public class User {

  public static final  String REGEX_USERNAME = "[a-z0-9][a-z0-9-][a-z0-9]";

  public static final  String REGEX_PASSWORD = "[a-f0-9]";

  public static final  String REGEX_EMAIL = "([0-9a-zA-Z]([-.w]*[0-9a-zA-Z])*@([0-9a-zA-Z][-w]*[0-9a-zA-Z].)+[a-zA-Z])"; 
  private String  username;     // 用戶名

  private String  password;     // MD5口令

  private boolean  admin;       // 是否是管理員

  private String  email;        // 電子郵件

  private int  emailValidation; // 電子郵件驗(yàn)證碼

  private long  createdDate;    // 創(chuàng)建時(shí)間

  private long  lockDate;       // 鎖定時(shí)間  
  public User() {} 
  public User(String  username, String password, boolean admin, long lastSignOnDate) {

  this.username = username;

  this.password =  password;

  this.admin = admin;

  } 
  @Id

  @Column(updatable=false,  length=20)

  @Pattern(regex=REGEX_USERNAME)

  public String  getUsername() { return username; }

  public void setUsername(String  username) { this.username = username; } 
  @Column(nullable=false,  length=32)

  @Pattern(regex=REGEX_PASSWORD)

  public String  getPassword() { return password; }

  public void  setPassword(String password) { this.password = password; } 
  @Column(nullable=false,  length=50)

  @Pattern(regex=REGEX_EMAIL)

  public String getEmail()  { return email; }

  public void  setEmail(String email) { this.email = email; } 
  @Column(nullable=false)

  public boolean getAdmin()  { return admin; }

  public void  setAdmin(boolean admin) { this.admin = admin; } 
  @Column(nullable=false,  updatable=false)

  public long  getCreatedDate() { return createdDate; }

  public void  setCreatedDate(long createdDate) { this.createdDate = createdDate; } 
  @Column(nullable=false)

  public int  getEmailValidation() { return emailValidation; }

  public void  setEmailValidation(int emailValidation) { this.emailValidation =  emailValidation; } 
  @Column(nullable=false)

  public long getLockDate()  { return lockDate; }

  public void  setLockDate(long lockDate) { this.lockDate = lockDate; }
  @Transient

  public boolean  getEmailValidated() { return emailValidation==0; } 
  @Transient

  public boolean  getLocked() {

  return !admin  && lockDate>0 && lockDate>System.currentTimeMillis();

  }

  }

上一頁(yè)1234下一頁(yè)
軟件測(cè)試工具 | 聯(lián)系我們 | 投訴建議 | 誠(chéng)聘英才 | 申請(qǐng)使用列表 | 網(wǎng)站地圖
滬ICP備07036474 2003-2017 版權(quán)所有 上海澤眾軟件科技有限公司 Shanghai ZeZhong Software Co.,Ltd