`
jimmy9495
  • 浏览: 300599 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

针对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();
        }
    }
}

分享到:
评论

相关推荐

    SpringBoot 多模块Dao层单元测试

    本篇文章将重点讲解如何在IDEA环境下,针对Spring多模块项目中的Dao层进行单元测试。 首先,我们要理解什么是单元测试。单元测试是一种软件测试方法,它针对程序中的最小可测试单元,如方法或类,进行独立验证。在...

    使用junit测试ssh中的dao

    标题“使用junit测试ssh中的dao”涉及到的是Java开发中的单元测试技术,特别是针对SSH(Struts2、Spring、Hibernate)框架中的数据访问对象(DAO)进行测试。SSH是Java Web开发中常用的三大开源框架,它们协同工作以...

    利用JUnit和Spring-test对SpringJDBC组件DAO层测试

    在针对SpringJDBC的DAO层进行测试时,我们需要创建模拟数据和预期结果,以检查DAO方法是否按预期工作。 Spring-test是Spring框架的一部分,专门用于测试Spring应用。它提供了如`@ContextConfiguration`、`@RunWith...

    Spring对于业务层的单元测试支持

    例如,你可以创建一个测试类,针对业务层的一个服务方法编写测试用例,通过`@Test`注解标记测试方法,并使用`assertThat`等断言库来验证方法的返回结果。 接下来,集成测试则关注组件间的交互,通常涉及到数据库、...

    JDBC相关单元测试及通用的Dao

    jdbc详细测试用例,包括connection ,statement,preparedstatement,resultset,BeanUtils,DBUtils,数据库连接池dbcp,cp03的单元测试,及dao层的统一封装,适合项目开发人员学习和项目中使用。

    自动测试所有dao单元测试

    NULL 博文链接:https://zhuliang1984723.iteye.com/blog/2260856

    junit测试spring,hibernate的dao层代码

    这样,`Spring` 会自动管理 `DAO` 实例,我们可以在测试方法中通过 `@Autowired` 注解注入 `DAO` 对象。同时,为了隔离测试,我们可以使用 `@Transactional` 注解,使得每次测试都在一个新的事务中执行,测试结束后...

    Service层和DAO层解析

    DAO层的设计应保持纯粹,避免包含复杂的业务逻辑,以便于维护和测试。 Service层则是面向功能的,它封装了业务逻辑,处理用户请求,并协调各个组件协同工作。例如,在银行系统中,一个存款功能可能会涉及到多个步骤...

    简单DAO层示例

    在IT行业中,DAO(Data Access Object)层是软件设计模式中的一个重要组成部分,它主要用于数据库操作,隔离了业务逻辑层与数据存储层之间的交互。在这个"简单DAO层示例"中,我们将探讨DAO的设计原则、实现方式以及...

    action层,dao层 ,service层,entity层1

    标题和描述中提到的"action层,dao层,service层,entity层"是这种分层架构的典型组成部分,下面将详细解释每一层的功能和它们之间的交互。 1. **Action层(控制层)**: 这一层主要负责接收用户的请求,进行业务...

    Spring的作用贯穿了整个中间层,将Web层、Service层、DAO层及PO无缝整合

    2. **分层设计**:Spring框架支持分层的设计模式,这意味着不同的业务功能可以被封装在不同的层中,例如Web层负责处理用户界面,Service层处理业务逻辑,DAO层处理数据访问。这样的设计有助于保持代码的清晰度,并...

    数据库持久层的UT测试

    在本场景中,我们关注的是MyBatis作为持久层框架的测试,以及DAO(数据访问对象)层的测试。这里使用的环境是JDK 1.6,并且项目使用Maven进行构建管理。测试策略包括针对MySQL和HSQL两种不同数据库的测试,允许通过...

    SSH单元测试代码整理

    4. **Mock对象**:使用Mockito等库模拟DAO层或Service层的对象,避免实际数据库交互,简化测试。 5. **断言**:使用JUnit提供的断言方法,如`assertEquals`,确保预期结果与实际结果一致。 6. **异常测试**:测试...

    用java的面向对象语言来操作关系型数据库 dao层.zip

    本压缩包文件“dao层.zip”显然关注的是数据访问对象(Data Access Object, DAO)层的设计,这是一个在Java应用中常见的用于数据库操作的抽象层。 DAO层的主要目的是将业务逻辑与数据存储机制分离,这样可以减少...

    Android-KBUnitTest是一款轻量级DAO单元测试框架

    KBUnitTest是一款轻量级DAO单元测试框架,开发者可以通过此框架,在Android Studio运行SQLiteDatabase、SharedPreference单元测试。KBUnitTest支持原生SQLiteDatabase操作及GreenDAO、Afinal、XUtils、DbFlow第三方...

    DAO 数据访问对象

    数据访问对象(DAO,Data Access Object)是一种设计模式,它在软件工程中被用来封装对数据库的操作,使得业务逻辑层与数据存储层之间的耦合度降低。DAO模式的主要目标是提供一个接口,通过这个接口,应用程序可以...

    DBUnit 进行单元测试

    在实际开发中,DBUnit还支持自定义操作、复杂查询结果验证等功能,可大大提高数据库测试的效率和准确性。结合持续集成工具如Jenkins,DBUnit能确保数据库层的代码质量,为整个项目的稳定性提供保障。因此,对于任何...

    对dbunit进行mybatis DAO层Excel单元测试(必看篇)

    在mybatis DAO层中,对数据库的单元测试尤为重要,因为它直接影响到软件的性能和可靠性。本文将围绕对dbunit进行mybatis DAO层Excel单元测试,介绍相关的知识点和实现方法。 一、Mybatis DAO层测试难点 在mybatis ...

    DAO单元测试

    今天我将展示一下我是如何在实际中对dao进行单元测试的首先我们来确认一下dao需要什么样的环境,我的dao是用Spring+hibernate来构建的,而对应的数据源是oracle9。所以要进行dao的测试我需要从Spring的连接oracle的...

    hibernate 单元测试批处理代码

    在Hibernate中,我们通常会测试持久化模型类、DAO(数据访问对象)和Service层。JUnit是一个流行的Java单元测试框架,我们可以结合Hibernate Test和Mockito库来创建高效的测试环境。Hibernate Test提供了对...

Global site tag (gtag.js) - Google Analytics