單元測(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();
}
}