vue 应用测试(一) --- 介绍

vue 应用测试(一) ---介绍

    • 前端测试简介
    • 组件测试
    • Jest 测试框架简介
      • 其他测试框架
    • 第一个测试
      • 避免误报
      • 如何组织测试代码
    • 组件挂载
      • Vue2 组件挂载的方式
      • Vue3 的挂载方式
      • vue-test-utils
      • 挂载选项
    • 如何调试测试用例
    • 参考
    • 小结

前端测试简介

软件测试:检查软件是否按照预期工作的过程。

测试分类:

从是否需要手动测试来分:

  • 手动测试:需要人工操作,比如点击按钮,输入文字等。
  • 自动测试:写代码测试其他代码,不需要人亲自手动测试每一个功能。

前端测试,从测试的范围来分:

  1. 端到端测试

测试整个应用,从用户角度出发,浏览器自动测试整个应用是否按照预期工作。是自动执行的手动测试,加快手动测试的速度。

  • 优点:测试全面,测试结果可靠。
  • 缺点:① 测试速度慢 ② 调试困难 ③ 可能成为 flakey 测试 ④ 编写测试代码的成本高。

flakey 测试:即使程序没有问题,测试也会失败。

  1. 单元测试

对应用的小部分进行的测试。比如测试一个函数,一个组件等。

  • 优点:① 测试速度快 ② 调试方便 ③ 编写测试代码的成本低 ④ 提供文档功能,可通过测试用例了解代码的行为 ⑤ 稳定,有助于重构。
  • 缺点:测试范围小,测试结果不可靠。

重构:不改变代码的功能,但是改变代码的结构,目的是为了提高代码质量。

  1. 快照测试

快照测试会给运行中的应用程序拍一张图片,并将其与以前保存的图片进行比较。如果图像不同,则测试失败。这种测试方法对确保应用程序代码变更后是否仍然可以正确渲染很有帮助。

按照是否测试实现来分类:

  1. 白盒测试

测试代码的实现,比如测试一个函数,需要知道函数的实现细节和依赖关系,然后编写测试代码。往往是开发人员编写的测试代码。

编写白盒测试,为了隔离复杂的依赖,通常会使用各种模拟手段,来模拟依赖的行为,比如模拟网络请求、用户操作和 pinia 等。

不仅测试做了什么,还要测试怎么做到的。

白盒测试用例往往比较脆弱,一旦代码实现发生变化,测试用例很可能就会失败。

  1. 黑盒测试

不关心代码的实现,只关心代码的输入和输出,比如测试一个函数,只需要知道函数的输入和输出,然后编写测试代码。往往是测试人员编写的测试代码。

要少用模拟或者恰当的模拟,因为你不知道内部的依赖关系,也难以模拟,二来越多的模拟测试越不可靠。

测试做了什么,而不是测试怎么做的。

黑盒测试用例往往比较稳定,一旦代码实现发生变化,测试用例不会失败。

各种测试在前端测试中的占比

为何没有集成测试?

前端的集成测试,难以定义、编写和调试,通常认为端到端测试就是集成测试。

什么时候不需要自动化测试?

自动化测试的目的是为了节省时间和精力,长期开发规模较大的项目自动化测试才会带来巨大的收益。如果项目只是一个小项目,或者是一个短期项目,那么自动化测试可能会带来负担,即编写测试代码会比直接编写应用代码更花时间。

实际上,在我的工作中,前端进行自动化测试的团队都很少,大部分都是手动测试。

不必追求 100% 的测试覆盖率

除非一个 bug 导致了严重的后果,比如损失几百万元,否则不必追求 100% 的测试覆盖率。因为测试代码也是需要维护的,测试代码的维护成本也是需要考虑的。

组件测试

组件有很多属性,决定测试哪些属性很重要,能帮助编写高效的测试代码。

如何决定测试哪些属性?

组件的输入和输出(有人叫组件契约或者组件接口)可帮助决定测试哪些属性。

从开发人员使用组件但又不了解组件具体实现的角度来编写测试,好的组件单元测试应该始终可触发一个输入,并断言一个输出。

常见的组件输入:

  1. 用户操作,比如点击按钮,输入文字等;
  2. props;
  3. 组件事件;
  4. vuex store 中的数据;
  5. inject 注入的数据。

常见的输出:

  1. 触发的事件;
  2. 外部调用的方法,即公有方法;
  3. 渲染结果。

Jest 测试框架简介

测试文件:以 .spec.js 或者 .test.js 结尾。

Jest 在查找项目中测试文件时使用默认的 glob 匹配模式。对于 non-glob 模式而言,这意味着 Jest 匹配tests目录中的.js 和.jsx 文件,以及扩展名为 .spec.js 或 .test.js 的所有文件。

globs 是文件匹配模式。Jest 使用 Node glob 模块匹配文件。你可以在如下链接页面的 glob primer 部分中阅读到更多关于 globs 的内容,glob-primer。

其他测试框架

vitest

peeky

第一个测试

已经存在一个使用 vue-cli 创建的项目,希望添加测试。

  1. 安装 vue 测试插件:
vue add @vue/cli-plugin-unit-jest

使用 vue-cli 创建的项目,可以使用 vue add 命令安装插件,会自动配置测试环境。

  1. 编写 HelloWorld.vue 组件:

安装完毕会自动配置测试环境,并创建了一个测试 HelloWorld.vue 的用例,但是项目里没有 HelloWorld.vue,在 tests/unit 就近新建一个。

<script>export default {name: 'HelloWorld',props: {msg: {type: String,default: '',},},data() {return {}},}
</script><template><div>{{ msg }}</div>
</template>

然后引入组件:

import HelloWorld from './HelloWorld.vue'
  1. 运行测试

执行 npm run test:unit ,测试环境是否配置成功。

可用性(sanity)测试

搭建测试系统的第一步是编写一个简单的测试来检查系统是否配置正确。这被称为可用性(sanity)测试

在排查复杂问题或者配置环境时,可用性测试应该成为第一个测试用例,因为它能检查环境是否配置正确。

就近放置测试文件

将单元测试放置在尽可能接近被测代码的位置,会更容易被其他开发人员找到。

避免误报

测试中,需要避免误报。测试之所以通过,是因为源代码正常工作,而不是因为编写始终能通过的测试。

常见的误报测试是使用异步代码

test('sets finished to true after 100ms', () => {runner.start()setTimeout(() => {expect(runner.finished).toBe(true) // 100ms 后 finished 为 true}, 100)
})

避免误报的最好方法是使用 TDD。

红色阶段是编写一个因正确原因而失败的测试。这里的关键词是“因正确原因”,即确定程序失败的边界条件。

测试驱动开发(TDD)是一种在编写源代码之前先编写测试代码的工作流程,即在编写组件代码之前,需要先编写能够确保组件正常运行的测试代码。

“红、绿、重构” 是一种很流行的 TDD 方法。红代表编写一个不能通过的测试,绿代表让测试通过,在测试通过后,通过重构增强代码可读性。

以这样的方式开发应用程序会有如下好处。首先,你只编写测试功能的源代码,从而保持较少的源代码量;其次,它可以使你在编写代码之前先考虑组件设计。

如何组织测试代码

describe 函数用于组织测试代码,describe 用于定义一组测试用例,每个测试用例都是一个 test 函数。

describe 函数将多个单元测试用例定义为一个测试套件。当你在命令行运行测试时,Jest 会格式化输出,以便你了解哪些测试套件通过,哪些测试套件失败。

describe('HelloWorld.vue', () => {it('renders props.msg when passed', () => {const msg = 'new message'const wrapper = shallowMount(HelloWorld, {propsData: {msg},})expect(wrapper.text()).toMatch(msg)})
})

当运行测试时,会在控制台格式化输出 describe 和 test 的一个参数,方便查看测试结果。

一个文件可写多个 describe,describe 可嵌套,

推荐的做法是一个文件只写一个 describe。

否则会降低测试代码的可读性和新加的测试用例的不知道放在哪个 describe 里面。

测试代码和源代码挨近,方便他人查看。

不要嵌套使用 describe,会让测试代码难以理解。

test 表示一个测试用例。

两个参数:

第一个参数是一个 字符串 ,在同一个测试套件中,需要唯一,用于标识测试报告中的测试,用来对你的测试做讲要的说明,方便你的阅读测试报告。

第二个参数是包含测试代码的函数。

it 是它的别名 xit 表示跳过这个测试用例,在跳过某些正在或者不想要测试的用例时特别有用。

test.only 表示只运行这个测试用例,其他测试用例都会被跳过。

三步法(3A法则)写测试用例

  1. 准备测试环境(数据、模拟的函数、模拟模块、挂载组件等,是测试的必要条件)(Arrange),让测试就绪,这里是渲染组件。
  2. 采取行动(Action),执行某些操作,比如用户输入、查找渲染结果等,这里是获取组件的文本内容。
  3. 断言(Assert),判断上述行动是否符合预期,这里是断言组件的文本内容和测试数据是否一致,。

以上三步的代码使用空行分隔,这样可以让测试代码更加清晰可读。

编程的经验法则之一:代码排版反映思路。

代码越是美观合理,说明写下这段代码的时候,思路越是清晰,这样的代码也更容易被其他人理解,反之亦然。

组件挂载

Vue 组件想要渲染到页面上,需要一个挂载的动作,或者说触发组件渲染到页面上的动作叫挂载。

Vue2 组件挂载的方式

  1. 在组件选项中指定 el。

  2. 使用 Vue 构造器动态挂载。

new Vue(componentOptions).$mount(el)

new Vue.extend(componentOptions).$mount(el)

Vue.extend 接收一个组件选项,然后返回一个构造器。

使用 Vue.extend 手动挂载组件,也是 vue2 中实现弹窗的方式,即新建一个和 body 同级的 div,然后把组件挂载到这个 div,从而让组件的渲染结果脱离组件的嵌套关系。

import Vue from 'vue'
const App = {props: {count: Number,},data() {return {msg: 'hello',innerCount: 0}},methods: {add() {this.innerCount += 1},},template: /*html*/ `<div v-if="count % 2 === 0">count:{{count}}. count is even.</div><div v-else>count:{{count}}. count is odd.</div><button @click="add">{{innerCount}}</button>`,
}describe('App', () => {it('挂载App', () => {const Ctor = new Vue.extend(App)const app = new Ctor()app.$mount()})
})

Vue3 的挂载方式

  1. createApp
const app = createApp(App)
app.mount('#app')

使用 createApp 挂载一个弹窗。

<template><div class="modal-container">我是弹窗组件<button type="button" @click="close">点击我关闭</button></div>
</template><script>import {ref,onMounted,reactive,watch,computed,defineComponent} from 'vue'export default defineComponent({name: 'Modal',components: {},setup(props, {emit,attrs,slots}) {onMounted(() => {console.log('onMounted')})function close() {const modal = document.querySelector('.modal')document.body.removeChild(modal)}return {close}},})
</script><style>.modal {position: fixed;left: 50%;top: 50%;transform: translate(-50%, -50%);width: 400px;height: 200px;z-index: 1000;background-color: #ccc;}.modal-container {position: relative;left: 50%;top: 50%;transform: translate(-50%, -50%);width: 100px;height: 100px;background-color: red;}
</style>

在某个组件内执行 mountModal 挂载显示弹窗。

function mountModal() {const div = document.createElement('div')div.className = 'modal'document.body.appendChild(div)const modal = createApp(Modal)modal.mount(div)
}

参考问题

  1. 瞬移组件 Teleport
<template><Teleport to="modal-container"><p>通过to属性指定挂载的DOM</p></Teleport>
</template>

jest 在 jsdom 环境中运行,jsdom 是一个模拟浏览器环境的库,它提供了一些浏览器环境的全局变量,比如 window、document 等。
所以能直接挂载组件。

vue-test-utils

手动挂载组件,代码量较多,vue-test-utils 提供了一些方便的 API 帮我们做这些事情。

mount 方法,该方法在接收一个组件后,会将其挂载并返回一个包含被挂载组件实例(vm)的包装器对象

知道为什么 mount 不直返回 Vue 实例(vm)而是返回包装器?

mount 返回的包装器不仅包含 Vue 实例,还包括一些辅助方法,你可以使用它们来设置 props检查实例属性以及操作实例

常用的包装器方法:

  1. text 方法:返回包装器的文本内容。
  2. html 方法:返回包装器的 HTML 内容。
  3. find 方法:返回包含指定选择器的第一个 DOM 元素的包装器。
  4. exists 方法:返回一个布尔值,指示包装器是否包含一个或多个匹配的元素。
  5. findAll 方法:返回包含指定选择器的所有 DOM 元素的包装器。
  6. setData 方法:设置组件的 data 属性。
  7. setProps 方法:设置组件的 props 属性。
  8. trigger 方法:触发指定的事件。
  9. vm 属性:返回包装器的 Vue 实例。

mount vs shallowMount

另一个挂载方法, shallowMount ,该方法与 mount 方法类似,但是它不会渲染组件的子组件,而是使用 vuecomponent-stub 代替。

它隔离了组件与其子组件的关系,排除了复杂的依赖关系,使得测试更加专注于当前组件。

mount 会渲染子组件,更加贴近真实环境,但是会增加测试的复杂度。

一种不渲染子组件的方 — stubs 模拟子组件。

mount(ParentCom, {stubs: {Child: trueChild2: `<span>我是子组件</span>`}
})

使用 stubs 对象模拟子组件,可以使用布尔值或者字符串模拟子组件。

这些子组件不会真的渲染,但是它们会存在你的包装器中,你可以使用 find 方法找到它们。

挂载选项

挂载时可以传递一些组件选项,比如 propsData、data、mocks 等,这些选项会覆盖组件的默认选项。

Mounting Options

如何调试测试用例

  1. 使用 vscode 扩展

Jest Runner 可以在 vscode 中运行测试用例,方便调试。

推荐使用

  1. 在 chrome 浏览器调试

开启 jest 调试模式,新加一个脚本:

"test:debug": "node --inspect-brk node_modules/.bin/vue-cli-service test:unit",

没成功,可能哪儿没弄对。你可以试试。

  1. 在 vscode 中调试

参考

vue 官方对测试的建议

Jest 单元测试环境搭建

Vue.js unit test cases with vue-test-utils and Jest

Guide to Unit Testing Vue Components

All Vue Content

Vue NYC - Component Tests with Vue.js - Matt O’Connell—技术演讲

An Introduction to testing in Javascript

小结

  • 介绍了前端测试的分类,单元测试能帮助我们编写高质量的代码。
  • 组件测试的要点:给组件输入,测试输出。
  • Jest 测试框架简介。
  • 3A法则写测试用例。
  • vue 组件渲染的三种方式:shallowMount、mount、Teleport.to 和手动挂载。
  • 不渲染子组件的几种方式:stubs 配置、shallowMount。
  • 调试测试用例的三种方式:vscode 插件、chrome、vscode。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[AIGC] Python的Range函数

Python的range()函数是一个内置函数&#xff0c;常常用于编程中生成数列。这个函数可以生成一个整数序列&#xff0c;这个序列通常用在循环中。 文章目录 基本用法详细用法注意事项 基本用法 range()函数的基本形式为 range(stop) —— 这将生成一个从0开始&#xff0c;到stop…

Docker_1.0

1.初识Docker 1.1.什么是Docker 微服务虽然具备各种各样的优势&#xff0c;但服务的拆分通用给部署带来了很大的麻烦。 - 分布式系统中&#xff0c;依赖的组件非常多&#xff0c;不同组件之间部署时往往会产生一些冲突。 - 在数百上千台服务中重复部署&#xff0c;环境不一…

Content type ‘application/x-www-form-urlencoded;charset=UTF-8‘ not supported

Content type application/x-www-form-urlencoded;charsetUTF-8 not supported 问题背景新增页面代码改造 问题背景 这里有一个需求&#xff0c;前端页面需要往后端传参&#xff0c;参数包括主表数据字段以及子表数据字段&#xff0c;由于主表与子表为一对多关系&#xff0c;在…

基于单片机的多功能智能小车设计

第一章 绪论 1.1 课题背景和意义 随着计算机、微电子、信息技术的快速发展,智能化技术的发展速度越来越快,智能化与人们生活的联系也越来越紧密,智能化是未来社会发展的必然趋势。智能小车实际上就是一个可以自由移动的智能机器人,比较适合在人们无法工作的地方工作,也可…

C++ 12 之 指针引用

c12指针引用.cpp #include <iostream>using namespace std;struct students12 {int age; };int main() {students12 stu;students12* p &stu; // 结构体指针students12* &pp p; // 结构体指针起别名pp->age 20;// (*pp).age 22;cout << "…

【CTF Web】CTFShow 探针泄露 Writeup(PHP+探针泄露+信息收集)

探针泄露 10 对于测试用的探针&#xff0c;使用完毕后要及时删除&#xff0c;可能会造成信息泄露 解法 查看网页源代码。 view-source:https://11170dfe-84c7-4fde-b1ca-5d1ec3dd7570.challenge.ctf.show/没有找到有用的信息。 用 dirsearch 扫描。 dirsearch -u https://1…

打造私密的通信工具,极空间搭建免费开源的电子邮件管理程序『Cypht』

打造私密的通信工具&#xff0c;极空间搭建免费开源的电子邮件管理程序『Cypht』 哈喽小伙伴门好&#xff0c;我是Stark-C~ 说起电子邮件大家都不陌生&#xff0c;哪怕是在当前微信或者QQ已经非常普遍的今天&#xff0c;电子邮件在我们很多人的工作中都充当了重要的通信工具。…

【TB作品】基于STM32单片机的实验室器材管理登记二维码系统

这个单片机代码实现了一个实验室管理系统&#xff0c;该系统的主要功能包括记录和管理ID信息、日期和时间、以及显示这些信息到OLED屏幕上。以下是对代码主要功能的分析&#xff1a; 全局变量定义 定义了多个全局变量来存储系统状态、页面、密码、ID列表等信息。time 结构体用…

专题六——模拟

目录 一替换所有的问号 二提莫攻击 三N字形变换 四外观数列 五数青蛙 一替换所有的问号 oj链接&#xff1a;替换所有的问号 思路&#xff1a;简单模拟&#xff1b;注意i0和in是处理越界问题就行&#xff01;&#xff01; class Solution { public:string modifyString…

docker容器基本原理简介

一、docker容器实例运行的在linux上是一个进程 1&#xff09;、我们通过docker run 通过镜像运行启动的在linux上其实是一个进程&#xff0c;例如我们通过命令运行一个redis&#xff1a; docker run -d --name myredis redis2&#xff09;、可以看到首先我们本地还没有redis镜…

【LeetCode最详尽解答】11-盛最多水的容器 Container-With-Most-Water

欢迎收藏Star我的Machine Learning Blog:https://github.com/purepisces/Wenqing-Machine_Learning_Blog。如果收藏star, 有问题可以随时与我交流, 谢谢大家&#xff01; 链接&#xff1a; 11-盛最多水的容器 直觉 这个问题可以通过可视化图表来理解和解决。 通过图形化这个…

「动态规划」如何求乘积最大子数组?

152. 乘积最大子数组https://leetcode.cn/problems/maximum-product-subarray/description/ 给你一个整数数组nums&#xff0c;请你找出数组中乘积最大的非空连续子数组&#xff08;该子数组中至少包含一个数字&#xff09;&#xff0c;并返回该子数组所对应的乘积。测试用例的…

【数据结构】初识集合深入剖析顺序表(Arraylist)

【数据结构】初识集合&深入剖析顺序表&#xff08;Arraylist&#xff09; 集合体系结构集合的遍历迭代器增强for遍历lambda表达式 List接口中的增删查改List的5种遍历ArrayList详解ArrayList的创建ArrayList的增删查改ArrayList的遍历ArrayList的底层原理 &#x1f680;所属…

【全栈实战】大模型自学:从入门到实战打怪升级,20W字总结(一)

&#x1f60a;你好&#xff0c;我是小航&#xff0c;一个正在变秃、变强的文艺倾年。 &#x1f514;本栏讲解【全栈实战】大模型自学&#xff1a;从入门到实战打怪升级。 &#x1f514;专栏持续更新&#xff0c;适合人群&#xff1a;本科生、研究生、大模型爱好者&#xff0c;期…

JVM-GC-什么是垃圾

JVM-GC-什么是垃圾 前言 所谓垃圾其实是指&#xff0c;内存中没用的数据&#xff1b;没有任何引用指向这块内存&#xff0c;或者没有任何指针指向这块内存。没有的数据应该被清除&#xff0c;垃圾的处理其实是内存管理问题。 JVM虽然不直接遵循冯诺依曼计算机体系架构&#…

基于flask的网站如何使用https加密通信-问题记录

文章目录 项目场景&#xff1a;问题1问题描述原因分析解决步骤解决方案 问题2问题描述原因分析解决方案 参考文章 项目场景&#xff1a; 项目场景&#xff1a;基于flask的网站使用https加密通信一文中遇到的问题记录 问题1 问题描述 使用下面的命令生成自签名的SSL/TLS证书和…

Docker镜像技术剖析

目录 1、概述1.1 什么是镜像&#xff1f;1.2 联合文件系统UnionFS1.3 bootfs和rootfs1.4 镜像结构1.5 镜像的主要技术特点1.5.1 镜像分层技术1.5.2 写时复制(copy-on-write)策略1.5.3 内容寻址存储(content-addressable storage)机制1.5.4 联合挂载(union mount)技术 2.机制原理…

用PHP来调用API给自己定制一个“每日新闻”

头条新闻汇聚了互联网上的时事动态&#xff0c;提供最新新闻动态、网络热门话题和视频更新等&#xff0c;覆盖社会、政治、体育、经济、娱乐、科技等多个领域&#xff0c;并不断刷新内容。企业应用这一接口后&#xff0c;可以快速吸引更多的用户访问自己的平台。即使是非新闻类…

有趣的傅里叶变换与小波变换对比(Python)

不严谨的说&#xff0c;时域和频域分析就是在不同的空间看待问题的&#xff0c;不同空间所对应的原子(基函数)是不同的。你想一下时域空间的基函数是什么&#xff1f;频域空间的基函数是什么&#xff1f;一般的时-频联合域空间的基函数是什么&#xff1f;小波域空间的基函数是什…

摄影师在人工智能竞赛中与机器较量并获胜

摄影师在人工智能竞赛中与机器较量并获胜 自从生成式人工智能出现以来&#xff0c;由来已久的人机大战显然呈现出一边倒的态势。但是有一位摄影师&#xff0c;一心想证明用人眼拍摄的照片是有道理的&#xff0c;他向算法驱动的竞争对手发起了挑战&#xff0c;并取得了胜利。 迈…