从0开始学习JavaScript--JavaScript 单元测试

JavaScript单元测试是保障代码质量和可维护性的关键步骤之一。通过编写和运行单元测试,开发者可以确保代码在不断迭代的过程中依然具有正确的行为。本文将深入探讨JavaScript单元测试的核心概念、工具使用和最佳实践,并通过丰富的示例代码演示其实际应用。

1. 单元测试的基本概念

1.1 什么是单元测试?

单元测试是对代码中最小可测试单元进行验证的过程。这个最小单元通常是函数、方法或类中的一个功能。单元测试旨在保证每个单元的功能是正确的,当进行修改时,能够快速检测到潜在的问题。

1.2 为什么需要单元测试?

  • 保证代码质量: 单元测试可以捕捉潜在的错误,确保每个功能单元都按照预期工作。

  • 提高可维护性: 单元测试作为代码文档的一部分,帮助开发者理解和维护代码。

  • 支持重构: 在重构代码时,单元测试可以确保修改不会破坏现有的功能。

2. 单元测试工具

2.1 Jest

Jest是一个由Facebook开发的JavaScript测试框架,具有简单易用、高度集成、性能优越等特点。

# 安装 Jest
npm install --save-dev jest
// package.json
{
  "scripts": {
    "test": "jest"
  }
}

2.2 Mocha

Mocha是一个灵活的JavaScript测试框架,可以在浏览器和Node.js环境中运行。

# 安装 Mocha
npm install --save-dev mocha
// package.json
{
  "scripts": {
    "test": "mocha"
  }
}

3. 编写和运行测试用例

3.1 Jest 示例

// math.js
function add(a, b) {
  return a + b;
}

module.exports = { add };
// math.test.js
const { add } = require('./math');

test('adds 1 + 2 to equal 3', () => {
  expect(add(1, 2)).toBe(3);
});
# 运行 Jest 测试
npm test

3.2 Mocha 示例

// math.js
function add(a, b) {
  return a + b;
}

module.exports = { add };
// math.test.js
const { add } = require('./math');
const assert = require('assert');

describe('Math', () => {
  it('should return 3 when adding 1 and 2', () => {
    assert.strictEqual(add(1, 2), 3);
  });
});
# 运行 Mocha 测试
npm test

4. 常用的断言库

4.1 Jest 断言

Jest内置了强大的断言库,其中最常用的是expect

// Jest 示例
test('adds 1 + 2 to equal 3', () => {
  expect(add(1, 2)).toBe(3);
  expect(add(1, 2)).toEqual(3);
  expect(add(1, 2)).not.toBeFalsy();
});

4.2 Mocha 断言

Mocha并不内置断言库,通常结合使用Node.js内置的assert库或其他第三方库,如chai

// Mocha 示例(使用 assert)
const assert = require('assert');

describe('Math', () => {
  it('should return 3 when adding 1 and 2', () => {
    assert.strictEqual(add(1, 2), 3);
    assert.notStrictEqual(add(1, 2), 4);
  });
});
// Mocha 示例(使用 chai)
const { expect } = require('chai');

describe('Math', () => {
  it('should return 3 when adding 1 and 2', () => {
    expect(add(1, 2)).to.equal(3);
    expect(add(1, 2)).to.not.equal(4);
  });
});

5. 测试异步代码

5.1 Jest 异步测试

Jest提供了多种处理异步代码的方式,例如使用async/await、`.

then().catch()`等。

// Jest 异步测试示例
test('async test', async () => {
  const result = await asyncFunction();
  expect(result).toBe('resolved value');
});

5.2 Mocha 异步测试

Mocha同样支持异步测试,可以使用done回调、async/await等方式。

// Mocha 异步测试示例
it('async test', (done) => {
  asyncFunction().then((result) => {
    expect(result).to.equal('resolved value');
    done();
  });
});

6. Mocking 和 Spying

6.1 Jest Mocking 和 Spying

Jest提供了强大的Mocking和Spying功能,可以方便地模拟函数的行为。

// Jest Mocking 和 Spying 示例
const mockFn = jest.fn();
mockFn.mockReturnValue(42);
mockFn.mockResolvedValue(42);

test('mocking and spying', () => {
  mockFn(1, 2, 3);
  expect(mockFn).toHaveBeenCalledWith(1, 2, 3);
  expect(mockFn).toHaveBeenCalledTimes(1);
  expect(mockFn()).toBe(42);
});

6.2 Mocha Mocking 和 Spying

在Mocha中,通常结合使用sinon库进行Mocking和Spying。

// Mocha Mocking 和 Spying 示例
const sinon = require('sinon');

const spy = sinon.spy();
const mock = sinon.mock().returns(42);

it('mocking and spying', () => {
  spy(1, 2, 3);
  sinon.assert.calledWith(spy, 1, 2, 3);
  mock();
  sinon.assert.calledOnce(mock);
  sinon.assert.returned(mock(), 42);
});

7. 持续集成与覆盖率检查

7.1 持续集成

集成CI/CD(Continuous Integration/Continuous Deployment)工具,如Travis CI、Jenkins,可以在每次代码提交时运行测试,确保代码的稳定性。

7.2 代码覆盖率检查

代码覆盖率工具,如istanbulnyc,可以帮助开发者评估测试覆盖的程度,确保每个代码路径都得到了测试。

# 安装 nyc
npm install --save-dev nyc
// package.json
{
  "scripts": {
    "test": "nyc mocha"
  }
}

8. 测试最佳实践

8.1 编写独立、可重复的测试

测试应该是独立的,不依赖于其他测试的执行结果。同时,测试应该是可重复的,不论运行多少次,结果都应该保持一致。

8.2 频繁运行测试

在开发过程中,频繁运行测试可以及时发现和修复问题,保持代码的稳定性。

8.3 测试覆盖率不是唯一标准

虽然高测试覆盖率通常是好的,但并不是唯一的标准。有时候,某些复杂的代码路径可能很难覆盖到,这时需要权衡测试的成本和效益。

总结

JavaScript单元测试是确保代码质量和可维护性的关键步骤。通过使用Jest、Mocha等测试框架,结合断言库和Mocking工具,开发者可以编写独立、可重复的测试,捕捉潜在的错误,并确保每个功能单元都按照预期工作。在持续集成和代码覆盖率检查的支持下,可以构建出更加健壮和可维护的代码。在实际应用中,需要根据项目需求和团队实际情况,选择合适的工具和策略,以确保测试的效果最大化。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/209290.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

PgSQL技术内幕 • statement_timeout做的那些事

PgSQL技术内幕 • statement_timeout做的那些事 statement_timeout是Postgres种的一个配置参数,用于指定SQL语句执行的超时时间,当超时时就取消该SQL的执行,并返回错误信息。这个参数通常用于控制运行时间较长的查询,避免影响数据…

STM32CubeIDE(CUBE-MX hal库)----蓝牙模块HC-05(详细配置)

系列文章目录 STM32CubeIDE(CUBE-MX hal库)----初尝点亮小灯 STM32CubeIDE(CUBE-MX hal库)----按键控制 STM32CubeIDE(CUBE-MX hal库)----串口通信 STM32CubeIDE(CUBE-MX hal库)----定时器 文章目录 系列文章目录前言一、蓝牙配置二、CUBE-MX可视化配置三、蓝牙APP调试助手四、…

微信小程序 地图撒点

1. 微信小程序 地图撒点 1.1 说明 首先使用微信小程序自带标签,并且设置好宽高让地图显示,用longitude和latitude表示中心点。   (1)show-location 显示带有方向的当前定位点,本项目不需要不添加。   (2&#xff…

组合(回溯+剪枝、图解)

77. 组合 - 力扣(LeetCode) 题目描述 给定两个整数 n 和 k,返回范围 [1, n] 中所有可能的 k 个数的组合。 你可以按 任何顺序 返回答案。 样例输入 示例 1: 输入:n 4, k 2 输出: [[2,4],[3,4],[2,3],…

【算法】单调栈题单——字典序最小⭐(一种类型的模板题)

文章目录 题目列表316. 去除重复字母⭐⭐⭐⭐⭐(类型题模板:单调栈,字典序最小)221021天池-03. 整理书架(保留数量为 limit 的字典序最小)402. 移掉 K 位数字(最多删除 k 次 前导零的处理&…

mysql主从复制-redis集群扩容缩容、缓存优化(缓存更新策略、穿透,击穿,雪崩)、mysql主从搭建、django实现读写分离

基于Docker实现读写分离 1 redis集群扩容缩容 1.1 集群扩容 1.2 集群缩容 2 缓存优化 2.1 缓存更新策略 2.2 穿透,击穿,雪崩 3 mysql主从搭建 4 django实现读写分离 1 redis集群扩容缩容 1.1 集群扩容 # 6台机器,3个节点集群# 8台机器&am…

hbase thrift2 jar包冲突导致启动失败问题排查记录

1、启动命令 ${HBASE_HOME}/bin/hbase-daemon.sh start thrift2 2、异常情况 hbase-root-thrift2-hdfs-test07.yingzi.com.out异常日志: Exception in thread "main" java.lang.AbstractMethodError: org.apache.hadoop.metrics2.sink.timeline.Hadoo…

TextToSpeech类学习和简单封装

TextToSpeech类简单学习封装 前言一、TTS是什么?二、TextToSpeech简单使用1.官方介绍2.简单使用 三、TextToSpeech简单封装总结 前言 业务涉及到对接TTS相关,所以简单学习下如何使用。 一、TTS是什么? TextToSpeech简称为TTS,即…

[网鼎杯 2020 青龙组]singal 1

前言 在主函数中找到了一个vm的译码器,译码器主要是解释传入的opcode,然后对我们输入的字符操作,这里我们发现他是单字节比较的,方法很多可以使用单字节映射,也可以是使用符号化执行,当然也可以硬着头皮去…

软件测试计划书

测试计划书 1.测试参考文档和测试提交文档 2.测试进度计划 3.测试资源 4.系统风险、优先级 5.测试策略 6.缺陷管理 7.测试停止标准 软件开发全文档下载进入主页。

Linux部署elasticsearch集群

文章目录 一、集群规划二、安装前准备(所有节点操作)创建数据目录修改系统配置文件/etc/sysctl.conf创建用户组设置limits.conf 三、初始化配置(在节点1上操作)下载安装包解压安装包修改jvm.options文件下配置的所占内存修改集群配置文件elasticsearch.yml将安装包传到另外两个…

JavaFramework JDK Version Test

测试JDK8 JDK17编译包 当前环境JDK8 CASE 1: /*** * author ZengWenFeng* email 117791303QQ.com* mobile 13805029595* date 2023-08-07*/ package zwf;import a.T; import ce.pub.util.GUID;/*** 测试高版本JDK编译JAR,低版本错误** author ZengWenF…

电梯导航的小练习

目录 css代码 html代码 js代码 完整代码 效果图 需求&#xff1a;点击某个模块&#xff0c;显示对应内容 css代码 <style>*{padding: 0;margin: 0;list-style: none;}ul{display: flex;justify-content: center;position: fixed;top: 0;left: 20%;}ul>li{text-…

对换数组的维度numpy.transpose()

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 对换数组的维度 numpy.transpose() 请以下代码执行print(np.transpose(a))后输出的结果是&#xff1f; import numpy as np a np.array([[0, 1], [2, 3]]) b np.array([[0, 1], [2, 3], […

Tomcat 漏洞修复

1、去掉请求响应中Server信息 修复方法&#xff1a; 在Tomcat的配置文件的Connector中增加 server" " &#xff0c;server 的值可以改成你任意想返回的值。

Gee教程5.中间件

鉴权认证、日志记录等这些保障和支持系统业务属于全系统的业务&#xff0c;和具体的系统业务没有关联&#xff0c;对于系统中的很多业务都适用。 因此&#xff0c;在业务开发过程中&#xff0c;为了更好的梳理系统架构&#xff0c;可以将上述描述所涉及的一些通用业务单独抽离…

蓝桥杯第198题 人物相关性分析 C++ 模拟 字符串 双指针

题目 思路和解题方法 程序首先定义了一个函数check&#xff0c;用于判断一个字符是否为字母。接下来&#xff0c;程序读取输入的整数k和一行字符串str。定义了两个空的向量a和b&#xff0c;用于存储满足条件的子串的起始位置。使用for循环遍历字符串str的每个字符&#xff0c;检…

Python--使用布林线设计均值回归策略

在本教程中,我们将探讨均值回归的概念以及如何使用 Python 中的布林线设计交易策略。均值回归是一种流行的交易策略,它基于这样的假设:随着时间的推移,资产价格往往会恢复到历史平均水平。布林线 (Bollinger Bands) 由约翰布林格 (John Bollinger) 开发,是一种技术分析工具…

喜讯 | Circulation(IF:37.8)ChIP-seq+RNA-seq助力解析USP28在糖尿病性心脏病的调控机制

2023年11月23日&#xff0c;国际知名期刊Circulation&#xff08;IF:37.8&&#xff09;在线发表了武汉大学人民医院心内科唐其柱教授团队题为 ” USP28 Serves as a Key Suppressor of Mitochondrial Morphofunctional Defects and Cardiac Dysfunction in the Diabetic He…

OSI七层模型与TCP/IP四层模型

一、OSI七层模型简述 OSI 模型的七层是什么&#xff1f;在 OSI 模型中如何进行通信&#xff1f;OSI 模型有哪些替代方案&#xff1f; TCP/IP 模型关于专有协议和模型的说明 二、七层模型详解&#xff08;DNS、CDN、OSI&#xff09; 状态码DNS nslookup命令 CDN whois命令 …
最新文章