为什么要做单元测试

  作为一个前端工程师,我是很想去谢单元测试的,因为每天的需求很多,还要去编写测试代码,感觉时间都不够用了。

  不过最近开发了一个比较复杂的项目,让我感觉一旦项目大了、复杂了,而且还是多人维护一个应用,编写测试代码,

  还是很有必要的。毕竟这样做完之后,后边的维护会轻松很多。

  单元测试

  测试代码的最小单元,一个函数就是一个单元

  测试工具

  主要用到的测试工具是 jest 和 enzyme

  jest 、 enzyme 介绍

  jest 是 facebook 发布的一个开源的,基于 jasmine 框架的 javascript 单元测试工具它提供了很多有效的功能,

  包含了内置的测试环境的 DOM API 来操作 Dom、断言库、mock 等功能,并且 jest 是 react 的官网上面推荐

  的工具。

  enzyme 是 Airbnb 开源的 react 测试类库,提供了一套简洁强大的 api ,可以通过 jq 的风格进行 dom 处理,

  还可以引用一些其他的 dom 操作库。

  react 单元测试的搭建

  react 项目是基于 webpack + babel 来进行构建的。

  安装 npm 包

  首先需要添加一些 jest + enzyme 的 npm 包

yarn add jest enzyme babel-jest regenerator-runtime react-test-renderer

  

  enzyme 需要安装对应的 react 的 adapter 版本

yarn add enzyme-adapter-react-16

  

  

  配置文件

  1、package.json

"scripts": {
"test": "jest --colors --coverage"
},   

  添加颜色和现实覆盖率

  .babelrc 文件

  

"env": { // 单元测试下配置
"test": {
"presets": [
"react",
"react-optimize",
[
"env",
{
"modules": "commonjs", // 主要是添加这个
"debug": true,
"useBuiltIns": false,
"targets": {
"browsers": "defaults"
}
}
],
"stage-0"
]
}
}

  

  因为测试环境的在 babel 中的配置 dev 为 test。

  所以这里需要特殊配置,modules 的形式是 commonjs

  添加jest配置文件jest.config

module.exports = {
roots: ['<rootDir>/__test__/', '<rootDir>/src/'], // 测试的目录
modulePaths: ['<rootDir>'],
coveragePathIgnorePatterns: ['/node_modules/', '/tests/setup.js'], //忽略统计覆盖率的文件
// bail: true,
// testRegex: '', // 后缀名称
// testPathIgnorePatterns: 'est', // 忽略路径
// moduleNameMapper: { // 与测试无关的资源文件同意mock 掉,这样在import 的时候就不会真的引入这些文件
// '^import?': '<rootDir>/build/jestImportMock.js',
// '\\.(css|less|gif|jpg|jpeg|png)$': '<rootDir>/build/jestStyleMock.js',
// },
};

  

  

  jest 详解

  

  1、简单测试结构

function sum(a,b){
return a+b;
} it('adds 1 + 2 to equal 3', () => {
expect(sum(1, 2)).toBe(3);
});

  

  2、异步的测试结构

async function sum (a,b){
return a+b;
} it('adds 1 + 2 to equal 3',async () => {
const total = await sum(1,2)
expect(total).toBe(3);
});

  

  3、Mock Function 结构

function forEach(items, callback) {
for (let index = 0; index < items.length; index++) {
callback(items[index]);
}
} const mockCallback = jest.fn(x => 42 + x);
forEach([0, 1], mockCallback); // 可以 mockCallback.mock 访问 mockCallback 状态

  

  1、jest.fn 用来创建一个 mock function

  2、mockCallback.mock 可以访问 mock function 的状态

  mock 的属性:

  calls : 调用的参数数组,保存了每次调用的入参

  calls.length : 调用次数

  calls[0][0] : 第一次调用的时候的第一个参数

  results:调用的结果数组

  results[0].value : 第一次调用的返回结果

  

  执行 mock function

const myMock = jest.fn();
console.log(myMock());
// > undefined myMock
.mockReturnValueOnce(10)
.mockReturnValueOnce('x')
.mockReturnValue(true); console.log(myMock(), myMock(), myMock(), myMock());
//10 x true

  1、用 mockRetrunValueOnce(value) 来预先设置函数的返回值

  2、mymock() 返回预先设置的值

  异步请求的数据模拟

let fetch = {
get:() =>{}
}
jest.mock(fetch)
fetch.get.mockResolvedValue({data:{},status:0})
fetch.get(url,options)
// {data:{},status:0}

  

  

  4、常用全局方法

  describe(message,()=>{})  : 创建块

  it(message,()=>{}) : 测试单元

   

  5、ecpect 常用方法

  expect(value) : 注入测试单元

   expect.extend({}) : 拓展 expect 方法

expect.extend({
toBeBetween(received, arg1, arg2) {
const min = arg1;
const max = arg2;
if (received > min && received < max) {
return {
message: '',
pass: true,
};
}
return {
message: `received is between in ${min} - ${max}`,
pass: true,
};
},
}); expect(99).toBeBetween(1,100)

  

  expect.anything()  : 返回除了 null 和 undefined 意外的任何类型数据

  expect.any(Number/String) : 任意 数字/字符串

   expect.assertions(num) : 确定在块中调用几次异步

  toBe(value) : 等于

  toHaveBeenCalled() : 确定函数调用

  toHaveBeenCalledTime(number) : 确定被调用的次数

  toHaveBeenCalledWith(arg1,arg2) :取保调用的函数的值

  toHaveBeenNthCalledWith(num,arg2) : 确保第几个参数

  toHaveRetruned() :确保有返回值

  toHave[:Nth]RetrunedWith([:num,]arg) : 确保某一个有返回值

  toContain(item) : 数组中是否包含 

   toMatch(regex) :正则表达式

  

  enzyme

  enzyme 一共有四个大的 api ( shallow 、mount 、render 、selectors )。

   通过这四个 api 可以对 react 组件进行操作。

   我们可以通过类似于 jquery 的操作来进行 test 。

  

  

react 单元测试 (jest+enzyme)的更多相关文章

  1. react: typescript jest && enzyme

    Install Jest 1.install jest dependencies jest @types/jest ts-jest -D 2.jest.config.js module.exports ...

  2. 搭建 Jest+ Enzyme 测试环境

    1.为什么要使用单元测试工具? 因为代码之间的相互调用关系,又希望测试过程单元相互独立,又能正常运行,这就需要我们对被测函数的依赖函数和环境进行mock,在测试数据输入.测试执行和测试结果检查方面存在 ...

  3. react typescript jest config (一)

    1. initialize project create a folder project Now we'll turn this folder into an npm package. npm in ...

  4. React单元测试——十八般兵器齐上阵,环境构建篇

    一个完整.优秀的项目往往离不开单元测试的环节,就 github 上的主流前端项目而言,基本都有相应的单元测试模块. 就 React 的项目来说,一套完整的单元测试能在在后续迭代更新中回归错误时候给与警 ...

  5. 如何为我的VUE项目编写高效的单元测试--Jest

    Unit Testing(单元测试)--Jest 一个完整的测试程序通常由几种不同的测试组合而成,比如end to end(E2E)测试,有时还包括整体测试.简要测试和单元测试.这里要介绍的是Vue中 ...

  6. jest enzyme unit test react

    1. 测试类型 单元测试:指的是以原件的单元为单位,对软件进行测试.单元可以是一个函数,也可以是一个模块或一个组件,基本特征就是只要输入不变,必定返回同样的输出.一个软件越容易些单元测试,就表明它的模 ...

  7. Jest+Enzyme React js/typescript测试环境配置案例

    本文案例github:https://github.com/axel10/react-jest-typescript-demo 配置jest的react测试环境时我们可以参考官方的配置教程: http ...

  8. [React] Use Jest's Snapshot Testing Feature

    Often when testing, you use the actual result to create your assertion and have to manually update i ...

  9. 单元测试jest部署

    引入jest需安装的基础插件: 基础插件 @babel/core 编译工具核心模块包 @babel/preset-env 编译工具,支持es2015特性的编译打包工具包 babel-jest 对.js ...

随机推荐

  1. 9i 和 11 g 区别

    9i 和 11 g 区别 9i 不支持 2/4/8G,只支持 2000/4000/8000 M的

  2. 获取Spring容器中的Bean协助调试

    在使用Spring进行开发时,有时调bug真的是很伤脑筋的一件事,我们可以通过自定义一个监听器来获取Spring容器中的Bean实例来协助我们调试. 第一步:编写自定义监听器 /** * 监听serv ...

  3. 树莓派+百度api实现人脸识别

    title: 树莓派+百度api实现人脸识别 tags: 树莓派 date: 2018-5-31 20:06:00 --- 树莓派对接百度api 我以前玩安卓的时候一直用的讯飞的平台和api,对于百度 ...

  4. Tcl之Math

    expr is for Tcl to do math operations. It takes all of its arguments ("2 + 2" for example) ...

  5. Xcode 6 Beta 高速官方下载地址

    推荐迅雷下载: http://adcdownload.apple.com//wwdc_2014/xcode_6_beta_ie8g3n/xcode_6_beta.dmg

  6. arx 插入图片

    #include <ShLwApi.h> #pragma comment(lib, "ShLwApi.lib") //插入影像图 Acad::ErrorStatus i ...

  7. java动态代理实现与原理详细分析(【转载】By--- Gonjan )

    [转载]By---Gonjan  关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式     ...

  8. Java基础(五)--内部类

    内部类简单来说就是把一个类的定义放到另一个类的定义内部 内部类分为:成员内部类.局部内部类.匿名内部类.静态内部类 成员内部类:最常见的内部类 public class Outter { privat ...

  9. 06C语言运算符

    C语言运算符 算术运算符 运算符 描述 + 把两个操作数相加 - 从第一个操作数中减去第二个操作数 * 把两个操作数相乘 / 分子除以分母 % 取模运算符,整除后的余数 ++ 自增运算符,整数值增加 ...

  10. A2. JVM 类加载机制

    [概述] 虚拟机把描述类的数据从 Class 文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的 Java 类型,这就是虚拟机的类加载机制. 与那些在编译时需要进行连接 ...