针对DAO层单元测试中断言复杂对象的解决方案

你还在为单元测试中断言大对象而烦恼吗?那么请看。。。
现状:
目前对于 查询 的测试基本上都是通过SQL文(insert语句)准备数据,然后通过代码获取对象,然后检查对象中的数据是否正确。

当是如果一张表的数据过多,想要全部检查每个字段是否正确映射到对象的字段,需要耗费大量的繁琐又无趣的Assert语句。

如信贷系统中的DA_DRAWNDN_CONTRACT表有90多个字段,要每一个字段都Assert过来几乎不现实。

解决方案:
1、通过解析准备数据的SQL(一般都在.sql文件中),获取每个字段对应的值;

2、再解析ibatis的配置文件(配置sql文的xml文件),获取每个字段对应的对象属性名;

3、通过放射机制获取需要比较的对象对应属性的值,进行Assert。

具体实现:
通过继承单元测试准备数据的基类AbstractPreparedDatabaseTransactionalTests,增加方法
/**
*检查对象是否正确
*@param t 需要校验的对象
*@param i 与.sql文件中的第几条数据比较
*/
protected <T> void assertObject(T t, int i)

目前已经实现的代码附件:

AssertObjectTransactionalTests.java

具体应用:
1.继承AssertObjectTransactionalTests

2.通过重载String getsqlMapPath()方法给出ibatis对应的xml配置文件在哪里

如:

/**
* 获取SqlMap的路径
*/
protected String getsqlMapPath() {
return "META-INF/sqlmap/CLMS_ACCT_REL_SqlMap.xml";
}
3.比较通过方法assertObject比较具体的对象和.sql文件中第几条准备数据

如.sql中有

insert into clms_acct_rel (ACCT_REL_ID, GMT_CREATED, GMT_CREATOR, GMT_MODIFIED, GMT_MODIFIER, IS_DELETED, ACCT_NO,
ACCT_REL_NO, ACCT_CUST_NAME, REL_TYPE, ACCT_TYPE, RELATION, USE_TYPE, REPAY_USE_TYPE, REL_START_DATE, REL_DUE_DATE,
AUTO_REPAY_AMT, AUTO_REPAY_RATE, SEND_SEQ_NO, REPAY_SEQ_NO, EMAIL, BANK_PROVINCE, BANK_CITY, BANK_ID, BANK_NAME, REL_STATUS)
values (909090, to_date('11-01-2011', 'dd-mm-yyyy'), 'test', to_date('21-01-2011', 'dd-mm-yyyy'), 'test', '0', 'ACCT000001',
'REL000001', 'CUST_NAME', '2', '3', '4', '1', '5', to_date('20-01-2011', 'dd-mm-yyyy'), to_date('22-01-2011', 'dd-mm-yyyy'),
10000.00000000, 50.000000, 1, 3, 'www@test1.com', '浙江', '杭州', '1', '建设银行', '1');
testCase:

/**
* 测试AcctRelDO.getById
*/
public void testGetbyId() {
AcctRelDO ret = acctRelDao.getById(909090L);
Assert.assertNotNull(ret);
assertObject(ret, 1);
}
其他
目前该类已经在信贷系统的UT中使用,还在继续完善中。

另外在AssertObjectTransactionalTests增加了一个方法<T> void assertObject(T t1, T t2),用来断言两个对象中的所有属性值是否相等。

此方法主要真对那些没有equals() 方法的对象,使用方法。。。不解释


package com.alifi.hades.dal.dao;

import java.io.InputStreamReader;
import java.io.LineNumberReader;
import java.lang.reflect.Method;
import java.math.BigDecimal;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;

import junit.framework.Assert;

import org.springframework.core.io.Resource;
import org.w3c.dom.Document;
import org.w3c.dom.NodeList;

public class AssertObjectTransactionalTests extends SqlScriptPreparedDatabaseTransactionalTests {

private static final String DATE_TYPE = "DATE";
private static final String VARCHAR_TYPE = "VARCHAR";
private static final String CHAR_TYPE = "CHAR";
private static final String DECIMAL_TYPE = "DECIMAL";
protected SimpleDateFormat dateFormat = new SimpleDateFormat("dd-MM-yyyy");

protected String getsqlMapPath() {
return null;
}

/**
* 检查对象是否正确
*
* @param t 需要校验的对象
* @param i 与.sql文件中的第几条数据比较
*/
protected <T> void assertObject(T t, int i) {

//解析sql文件
try {

//如果没有给出sqlMap路径抛出异常
if (getsqlMapPath() == null) {
throw new Exception();
}
//获取所有的准备数据sql文
List<String> lstStatement = getStatement();

//解析sql文获取字段对应的值
Map<String, String> col_val = parseStatement(lstStatement.get(i - 1));

//解析sqlMap和比较数据
parseSqlMap(getsqlMapPath(), col_val, t);

} catch (Exception ex) {
Assert.fail();
}
}

/**
* 解析sql文件获取sql文列表
*
* @return sql文列表
* @throws Exception
*/
protected List<String> getStatement() throws Exception {
List<String> lstStatement = new ArrayList<String>();

LineNumberReader reader = null;
Resource resource = applicationContext.getResource(getDataLocation());
reader = new LineNumberReader(new InputStreamReader(resource.getInputStream(), "UTF-8"));
String line = null;
StringBuilder statement = new StringBuilder();
while ((line = reader.readLine()) != null) {
line = line.trim();
if (line.endsWith(";")) {
statement.append(" ").append(line.substring(0, line.length() - 1));
lstStatement.add(statement.toString());
statement = new StringBuilder();
} else {
statement.append(" ").append(line);
}
}
return lstStatement;
}

/**
* 解析sql文获取每一字段对应的值
*
* @param statement sql文
* @return 每一字段对应的值
* @throws Exception
*/
protected Map<String, String> parseStatement(String statement) throws Exception {

Map<String, String> col_val = new HashMap<String, String>();

String[] statementPart = statement.split("values");
String columnStatement = statementPart[0].substring(statementPart[0].indexOf('(') + 1,
statementPart[0].lastIndexOf(')'));
String valueStatement = statementPart[1].substring(statementPart[1].indexOf('(') + 1,
statementPart[1].lastIndexOf(')'));

String[] columns = columnStatement.split(",");
valueStatement = valueStatement.replaceAll("to_date *\\(", "")
.replaceAll(", *'dd-mm-yyyy'\\)", "").replaceAll("'", "");
String[] values = valueStatement.split(",");

for (int i = 0; i < columns.length; i++) {
col_val.put(columns[i].trim(), ("".equals(values[i].trim()) || "null".equals(values[i]
.trim())) ? null : values[i].trim());
}

return col_val;
}

/**
* 通过解析SqlMap获取column对应的属性名和类型,并通过放射规则和对象t进行比较
*
* @param <T>
* @param sqlMapPath sqlMap路径
* @param col_val 每一字段对应的值
* @param t 需要比较的对象
* @throws Exception
*/
protected <T> void parseSqlMap(String sqlMapPath, Map<String, String> col_val, T t)
throws Exception {

DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
DocumentBuilder builder = factory.newDocumentBuilder();
Resource resource = applicationContext.getResource(sqlMapPath);
Document doc = builder.parse(resource.getInputStream());
NodeList nl = doc.getElementsByTagName("result");

String column;
String type;
String property;
Object value;
for (int i = 0; i < nl.getLength(); i++) {
column = nl.item(i).getAttributes().getNamedItem("column").getNodeValue();
type = nl.item(i).getAttributes().getNamedItem("jdbcType").getNodeValue();
property = nl.item(i).getAttributes().getNamedItem("property").getNodeValue();

if (DATE_TYPE.equals(type)) {
value = t
.getClass()
.getMethod(
"get" + property.substring(0, 1).toUpperCase()
+ property.substring(1)).invoke(t);
if (col_val.get(column) != null) {
Assert.assertEquals(dateFormat.parse(col_val.get(column)), (Date) value);
} else {
Assert.assertEquals(col_val.get(column), value);
}
} else if (VARCHAR_TYPE.equals(type) || CHAR_TYPE.equals(type)) {
value = t
.getClass()
.getMethod(
"get" + property.substring(0, 1).toUpperCase()
+ property.substring(1)).invoke(t);
if (col_val.get(column) != null) {
Assert.assertEquals(col_val.get(column), String.valueOf(value).trim());
} else {
Assert.assertEquals(col_val.get(column), value);
}
} else if (DECIMAL_TYPE.equals(type)) {

value = t
.getClass()
.getMethod(
"get" + property.substring(0, 1).toUpperCase()
+ property.substring(1)).invoke(t);
if (col_val.get(column) != null) {
Assert.assertEquals(0, new BigDecimal(value.toString())
.compareTo(new BigDecimal(col_val.get(column))));
} else {
Assert.assertEquals(col_val.get(column), value);
}
}
}
}

/**
* 比较两个对象里的属性值
*
* @param t1 对象1
* @param t2 对象2
*/
protected <T> void assertObject(T t1, T t2) {
Method[] methods = t1.getClass().getMethods();

try {
for (Method method : methods) {
if (method.getName().startsWith("get") && !"getClass".equals(method.getName())) {
if (method.invoke(t1) == null) {
continue;
}
if (method.getReturnType() == String.class) {
Assert.assertEquals(method.invoke(t1).toString().trim(), method.invoke(t2)
.toString().trim());
} else if (method.getReturnType() == Date.class) {
Assert.assertEquals(dateFormat.format(method.invoke(t1)),
dateFormat.format(method.invoke(t1)));
} else {
Assert.assertEquals(0, new BigDecimal(method.invoke(t1).toString())
.compareTo(new BigDecimal(method.invoke(t2).toString())));
}
}
}
} catch (Exception e) {
Assert.fail();
}
}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值