51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 3830|回复: 2
打印 上一主题 下一主题

[原创] 工具推荐:通过代码构造Excel测试数据

[复制链接]
  • TA的每日心情
    擦汗
    3 天前
  • 签到天数: 1027 天

    连续签到: 2 天

    [LV.10]测试总司令

    跳转到指定楼层
    1#
    发表于 2021-9-7 10:20:10 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
    项目背景
      系统有个Excel导入功能,需要通过Excel来构造测试数据,生成测试数据之后,上传Excel并导入验证。
      如果数据验证出错,需要给出错误提示;如果数据校验通过,则执行导入操作。
      我们主要通过Alibaba的Easy Excel框架来做。
      期望的生成Excel数据模板如下:

    生成了人员的工号、姓名、证件类型、身份证、民族等等字段。
      这里我们使用java语言,用到的工具类如下:

      java-faker
      构造测试数据时,需要绞尽脑汁浪费时间,JavaFaker可以释放你的生产力。

      easy-excel
      easy-excel重写了poi对07版Excel的解析,能够原本一个3M的Excel用POI sax依然需要100M左右内存降低到几M,并且再大的Excel不会出现内存溢出。
      03版依赖POI的sax模式,在上层做了模型转换的封装,让使用者更加简单方便。

      lombok
      作用类上,生成所有成员变量的getter/setter方法,作用于成员变量上,生成该成员变量的getter/setter方法,可以设定访问权限及是否懒加载等。

      代码解析
      代码调用链

    我们先从最底层的Model层来说:以一个项目模板为例,构建Model,此时一个Excel的sheet,就对应一个model。
      这里我们需要注意的是 @ExcelIgnore 和 @ExcelProperty注解:
      ·@ExcelIgnore表示忽略该字段
      ·@ExcelProperty表示设置字段名称、第几列
    1. package org.example.excel.model.emp;
    2. import com.alibaba.excel.annotation.ExcelProperty;
    3. import lombok.Data;

    4. //个人信息更新
    5. @Data
    6. public class EmpInformationUpdateModel {

    7.     @ExcelProperty(value = "*F00730-工号", index = 0)
    8.     private String empJobNumber;

    9.     @ExcelProperty(value = "*F00719-姓名", index = 1)
    10.     private String empName;

    11.     @ExcelProperty(value = "*F00721-英文名", index = 2)
    12.     private String foreignName;

    13.     @ExcelProperty(value = "F00741-昵称", index = 3)
    14.     private String nickName;

    15.     @ExcelProperty(value = "*F00370-性别", index = 4)
    16.     private String gender;

    17.     @ExcelProperty(value = "*F00392-国籍", index = 5)
    18.     private String nationality;

    19.     @ExcelProperty(value = "*F00720-证件类型", index = 6)
    20.     private String idCardType;

    21.     @ExcelProperty(value = "*F00717-证件号码", index = 7)
    22.     private String idCardNumber;

    23. }
    复制代码
    Model层和Excel的列是一一对应的,以下面的注解为例子:

    1. @ExcelProperty(value = "*F00730-工号", index = 0)
    复制代码
    第一行是标题头,第一行第一列的位置由0来表示,0,1,2,3,4…依次递增,第一行第一列的标题名称是*F00730-工号。

    Service层
    package org.example.excel.change;

    import com.github.javafaker.Faker;
    import org.example.excel.ExceptionTransferImport;
    import org.example.excel.constant.ChangeConstant;
    import org.example.excel.model.Employee;
    import org.example.excel.model.emp.EmpInformationUpdateModel;
    import org.example.excel.util.IdCardGenerator;
    import java.util.ArrayList;
    import java.util.List;
    import java.util.Locale;
    import java.util.Random;
    import java.util.logging.Logger;
    import static org.example.excel.util.RandomDateUtil.randomDate;

    //个人信息更新
    public class EmpInformationUpdate {

        //日志记录
        static java.util.logging.Logger logger = Logger.getLogger("com.hongtao.flywheel");

        //调用Model层的函数
        public static List<EmpInformationUpdateModel> empInformationUpdateData(List<Employee> empList) {

            //构建一个List来存放数据
            List<EmpInformationUpdateModel> list = new ArrayList<EmpInformationUpdateModel>();

            //使用java-faker框架,来生成随机姓名,身份证
            Faker FAKER = new Faker(Locale.CHINA);
            Faker FAKEREN = new Faker(Locale.ENGLISH);
            IdCardGenerator idCardGenerator = new IdCardGenerator();

            //构建for循环,控制生成测试数据的条数
            for (int i = 0; i < ExceptionTransferImport.CYCLESNUMBER; i++) {

                //调用了Model层,表示生成的单条数据的类型,和Model类一致
                EmpInformationUpdateModel data = new EmpInformationUpdateModel();
                data.setEmpName(empList.get(i).getEmpName());
                data.setEmpJobNumber(empList.get(i).getEmpJobNumber());
                
                //使用java-faker框架,来生成随机姓名,身份证
                data.setForeignName(FAKEREN.name().lastName() + new Random().nextInt(50));
                data.setNickName(FAKEREN.name().lastName() + new Random().nextInt(50));

                //下面都是调用数据字典类数据
                data.setIdCardType(new ChangeConstant().getValue(ChangeConstant.idCardType));
                data.setNationality(new ChangeConstant().getValue(ChangeConstant.nationality));
                data.setNation(new ChangeConstant().getValue(ChangeConstant.nation));
               
          
                //List链表增加一条数据
                list.add(data);
            }
            
            //返回链表
            return list;
        }
    }

    数据字典类解析:配置数据字典,因为在有些数据是固定值,需要随机选择数据,所以用list来存储这些数据字典。

    1. (new ChangeConstant().getValue(ChangeConstant.contractType);
    复制代码
     这句话是为了随机获取一个存在List内的数据字典。
      数据字典也是用list存放,通过getValue方法随机取一个值,如下:
    1.    public static final List idCardType = Arrays.asList("中国护照", "台湾居民来往大陆通行证", "外国护照", "居民身份证", "港澳居民来往内地通行证");
    2.     public static final List gender = Arrays.asList("男", "女");
    3.    
    4.    public String getValue(List list) {
    5.         Random random = new Random();
    6.         int n = random.nextInt(list.size());
    7.         return list.get(n).toString();
    8.     }
    复制代码
    生成随机时间的类
      randomDate("2020-01-01", "2022-01-01")是随机生成在开始时间和结束时间范围内的一个日期值,函数如下:
    1. public static String randomDate(String beginDate,String endDate){
    2.         try {

    3.             SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
    4.             Date start = format.parse(beginDate);
    5.             Date end = format.parse(endDate);

    6.             if(start.getTime() >= end.getTime()){
    7.                 return null;
    8.             }
    9.             long date = random(start.getTime(),end.getTime());

    10.             String dateString=
    11.                     asLocalDate(longToDate(date)).format(DateTimeFormatter.ofPattern("yyyy-MM" +
    12.                     "-dd"));
    13.             return dateString;
    14.         } catch (Exception e) {
    15.             e.printStackTrace();
    16.         }
    17.         return null;
    18.     }
    复制代码
    java-faker的用法
      可以生成随机的姓名、街道、公司、天气、超级英雄、星球大战等等等等。
    1. data.setForeignName(FAKEREN.name().lastName());
    2. data.setLuckyName(FAKER.funnyName().name());
    复制代码
    结合起来
      调用Easy Excel框架,生成Excel的方法,Main函数生成数据:
    1. String updateContractChangeFileName =
    2.                 TestFileUtil.getPath() + "个人信息更新"+".xlsx";
    3.         EasyExcel.write(updateContractChangeFileName,
    4.                 UpdateContractChangeModel.class).sheet("模板").doWrite(UpdateContractChange.updateContractChangeData(empInfomationList));
    复制代码
     结果展示

    总结
    在工作当中,我们经常遇到批量生成测试数据的需求,所以这个时候,我们可以借鉴网上成熟的生成随机数据方法,Java-Faker框架,很方便的构造测试数据。
      再结合业务上的需求,引入了阿里巴巴的Easy Excel框架,站在前人的肩膀上,助力测试业务更好的开展。这样子测试的工作才会越走越顺,测试的效率也能够不断的提高~







    本帖子中包含更多资源

    您需要 登录 才可以下载或查看,没有帐号?(注-册)加入51Testing

    x
    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-10-3 18:20 , Processed in 1.304449 second(s), 24 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表