完全掌握vue全家桶单元测试 : 4.断言常用方法

断言常用方法

断言常用方法

断言最核心的就是 expect 和后面的 toXXX ,我们在前一章看过最简单的toBe 语法,随着更多的数据结构出现,vitest 根据 JS 不同的数据结构类型,有不同的断言方法,下面我们来详细学习了解下。

万能 toBe

toBe 可用于断言基元是否相等或对象共享相同的引用。 它相当于调用 expect(Object.is(3, 3)).toBe(true)

ts 复制代码
  it('test toBe ', () => {
    const stock = {
      type: 'apples',
      count: 13
    }

    expect(stock.type).toBe('apples')
    expect(stock.count).toBe(13)
    const refStock = stock
    expect(stock).toBe(refStock)
  })

not

使用 not 将否定该断言。 例如,此代码断言 1 值不等于 2。 如果相等,断言将抛出错误,测试将失败。

ts 复制代码
  it('test not toBe ', () => {
    expect(1).not.toBe(2)
  })

基本数据类型断言

数字比较大小

  • toBeGreaterThan 大于预期结果
  • toBeLessThan 小于预期结果
  • toBeGreaterThanOrEqual 大于等于预期结果
  • toBeLessThanOrEqual 小于等于预期结果
ts 复制代码
  it('test 数字 ', () => {
   expect(10 + 10).toBe(20)
   // not
   expect(10 + 10).not.toBe(30)
   // >
   expect(3).toBeGreaterThan(2)
   // <
   expect(3).toBeLessThan(4)
   expect(3 < 4).toBe(true)

   // >=
   expect(3).toBeGreaterThanOrEqual(3)
   expect(3).toBeGreaterThanOrEqual(2)
   expect(3 >= 3).toBe(true)
   expect(3 >= 2).toBe(true)
   // <=
   expect(3).toBeLessThanOrEqual(3)
   expect(3).toBeLessThanOrEqual(4)
   expect(3 <= 3).toBe(true)
   expect(3 <= 4).toBe(true)
 })

浮点数 toBeCloseTo

浮点数不能直接用 expect(0.2 + 0.1).toBe(0.3),因为精度问题,会导致失败,必须得用toBeCloseTo

ts 复制代码
 it.fails('test toBeCloseTo', () => {
   expect(0.2 + 0.1).toBe(0.3) // 0.2 + 0.1 is 0.30000000000000004
 })
 it('test toBeCloseTo', () => {
   expect(0.2 + 0.1).not.toBe(0.3) // 0.2 + 0.1 is 0.30000000000000004
   expect(0.2 + 0.1).toBeCloseTo(0.3)
 })

toBeDefined、toBeUndefined

  • toBeDefined 断言值不等于 undefined
  • toBeUndefined 断言值 is 等于 undefined
ts 复制代码
 it('test undefined', () => {
   // undefined
   expect(undefined).toBe(undefined) // toBe 替代方式
   expect(undefined).not.toBeDefined()
   expect(undefined).toBeUndefined()
   expect('').toBeDefined()
 })

toBeTruthy、toBeFalsy

toBeTruthy断言值在转换为布尔值时为 true。如果你不关心值,只想知道它可以转换为true,也就是我们所说的真值,例如1、{},而不仅仅是 true

ts 复制代码
 it('test Boolean ', () => {
   // boolean true
   expect(!!2).toBe(true) // toBe 替代方式
   expect(true).toBeTruthy()
   expect(1).toBeTruthy()
   expect({}).toBeTruthy()
   expect([]).toBeTruthy()

   // boolean false
   expect(!!'').toBe(false) // toBe 替代方式
   expect(0).toBeFalsy()
   expect('').toBeFalsy()
   expect(null).toBeFalsy()
   expect(undefined).toBeFalsy()
   expect(NaN).toBeFalsy()
   expect(false).toBeFalsy()
 })

toBeNull

toBeNull 断言某些内容是否为 null

ts 复制代码
 it('test null', () => {
   expect(null === null).toBe(true) // toBe 替代方式
   expect(null).toBeNull()
 })

看到这里,你会发现基本数据类型大部分都能用 toBe 去替代,

引用类型断言

toEqual、toStrictEqual

toEqual 断言实际值是否等于接收到的值,或者如果它是一个对象,则是否具有相同的结构(递归比较它们)。如果对象某个属性是 undefined 时,会自动忽略该属性。我们可以通过以下示例看到 toEqualtoBe 之间的区别:

ts 复制代码
  it('test toEqual', () => {
    const stockBill = {
      type: 'apples',
      count: 13
    }
    const stockMary = {
      type: 'apples',
      count: 13
    }
    const stockBill2 = {
      type: 'apples',
      count: 13,
      name: undefined
    }
    const stockMary2 = {
      type: 'apples',
      count: 13
    }
    expect(stockBill).toEqual(stockMary)
    expect(stockBill).not.toBe(stockMary)

    expect(stockBill2).toEqual(stockMary2)
    expect(stockBill2).not.toBe(stockMary2)
  })

toStrictEqual .toEqual 的区别:

  • 检查具有 undefined 属性的键。 例如 使用 .toStrictEqual 时, {a: undefined, b: 2}{b: 2} 不匹配。
  • 检查数组稀疏性。 例如 使用 .toStrictEqual 时, [, 1][undefined, 1] 不匹配。
  • 检查对象类型是否相等。 例如 具有字段 ab 的类实例不等于具有字段 ab 的文字对象。
ts 复制代码
 it('test toStrictEqual', () => {
   const stockBill = {
     type: 'apples',
     count: 13,
     name: undefined
   }
   const stockMary = {
     type: 'apples',
     count: 13
   }
   class Stock {
     type: any
     constructor(type: any) {
       this.type = type
     }
   }
   expect(stockBill).not.toStrictEqual(stockMary)
   expect([1]).not.toStrictEqual([undefined, 1])
   expect(new Stock('apples')).not.toStrictEqual({ type: 'apples' })
 })

toContain

toContain 断言实际值是否在数组中。toContain 还可以检查一个字符串是否是另一个字符串的子串,此断言还可以检查类是否包含在 classList 中,或一个元素是否包含在另一个元素中。

ts 复制代码
 it('test toContain', () => {
   expect(['apple', 'orange']).toContain('orange')
   expect('123abc123').toContain('123abc')
   // const element = document.querySelector('#el')
   // expect(document.querySelector('#wrapper')).toContain(element)
 })

toHaveProperty

toHaveProperty 断言对象是否具有提供的引用 key 处的属性。

ts 复制代码
 it('test toHaveProperty', () => {
   expect({ name: 'xxx', age: 10 }).toHaveProperty('name')
 })

toMatchObject

toMatchObject 断言对象是否匹配另一个对象的部分属性。类似之前数组的toContain,对象会从最外层开始比较,如果最外层就找不到,就会直接失败

ts 复制代码
 it('test toMatchObject', () => {
   expect([{ foo: 'bar' }, { baz: 1 }]).toMatchObject([{ foo: 'bar' }, { baz: 1 }])
   expect({ obj: { name: 'xxx' }, height: 10 }).toMatchObject({ height: 10 })
   expect({ obj: { name: 'xxx' }, height: 10 }).not.toMatchObject({ name: 'xxx' })
 })

Error 断言

捕获错误的一个断言方法,例如在一些抛出错误,表单检验、数据格式错误、try...catch 等场景下会用到

ts 复制代码
  it('test Error ', () => {
    expect(() => {
      JSON.parse('{')
    }).toThrow()
  })

快照断言

快照可以理解成,把对象的结构或者基础数据类型转换成字符串,然后做一个拍照存档的概念,一般用于记录,其实就是一个偷懒的行为。例如我有一个盒子,里面有一个苹果,我拍照记录了,下次如果盒子里面放了梨,那就代表盒子被别人动过了,用例就会失败了

toMatchInlineSnapshot

toMatchInlineSnapshot 用于行内快照断言,它适合小范围,少量的数据结构存储

ts 复制代码
  it('test toMatchInlineSnapshot', () => {
    const data = { foo: new Set(['bar', 'snapshot']) }
    expect(data).toMatchInlineSnapshot(`
      {
        "foo": Set {
          "bar",
          "snapshot",
        },
      }
    `)
    expect(22).toMatchInlineSnapshot('22')
    expect(true).toMatchInlineSnapshot('true')
    expect([1, 2, 3]).toMatchInlineSnapshot(`[
  1,
  2,
  3,
]`)
    expect({ name: 'xxx' }).toMatchInlineSnapshot(`{
  "name": "xxx",
}`)
  })

toMatchSnapshot

toMatchSnapshot 快照断言会生成一个文件,它适合一些大型的,长久不变更的地方,例如对配置文件进行快照或者如果使用一些远程图标库 icon,我们可以对 icon 地址进行快照,这个文件如果变更了,就会出现报错,报错就代表有风险,需要谨慎操作。

ts 复制代码
  it('test toMatchSnapshot', () => {
    const config = { url: 'url', domain: 'domain', analysis: 'analysis alias' }
    expect(config).toMatchSnapshot()
  })

例如上面的用例执行了之后,会生成一个文件 __snapshots__/expect.test.ts.snap,文件上一次快照生成时候的样子

如果这个配置文件某天被某个小可爱不小心修改了,那就会报错,用例就通过不了,需要二次确认

然后在命令行输入 u 二次确认之后就可以通过了

函数断言

  • toHaveBeenCalled 判断函数是否被调用
  • toHaveBeenCalledTimes 判断函数被调用的次数
  • toHaveBeenCalledWith 判断函数被调用的时候传递了什么参数

上面的断言在执行之前,要先使用vi.spyOn对原函数进行调用, 例如下面例子的vi.spyOn(market, 'buy')

ts 复制代码
  it('test function ', () => {
    const market = {
      buy(subject: string, amount: number) {
        // ...
      }
    }
    const buySpy = vi.spyOn(market, 'buy')
    expect(buySpy).not.toHaveBeenCalled()
    market.buy('apples', 10)
    market.buy('apples', 10)
    expect(buySpy).toHaveBeenCalled()
    expect(buySpy).toHaveBeenCalledTimes(2)
    expect(buySpy).toHaveBeenCalledWith('apples', 10)
  })

又或者使用 vi.fn 创建一个假函数,现在大家还不知道 vi.spyOnvi.fn 的含义,但后面我会在 mock 章节专门讲解,大家也可以提前去官网查看 api 预习

ts 复制代码
 const mockFunction = vi.fn();
    mockFunction();
    expect(mockFunction).toHaveBeenCalled()
    expect(mockFunction).toHaveBeenCalledTimes(1)
    expect(mockFunction).toHaveBeenCalledWith()

函数副作用

在编程中,函数副作用是指函数在计算结果之外对外部状态的任何潜在影响,例如日期、时间、请求、浏览器平台相关的 api,说到这,你会发现,好像本章节并没有讲到如何断言,那这些有副作用的函数如何测试呢?后面章节会仔细讲解如何在使用有副作用的外部依赖也能保证函数返回相同的值,然后再根据上面断言的方法就可以了。

课件地址

上面的代码,都放到了 github 上,欢迎点赞收藏,我会持续更新代码和文章

往期文章

完全掌握vue全家桶单元测试 : 1. 为什么需要前端测试

完全掌握vue全家桶单元测试 : 2. 搭建 vitest 环境

完全掌握vue全家桶单元测试 : 3. vitest 用法概览

完全掌握vue全家桶单元测试 : 4.断言常用方法

相关推荐
bysking30 分钟前
【前端-组件】定义行分组的表格表单实现-bysking
前端·react.js
王哲晓1 小时前
第三十章 章节练习商品列表组件封装
前端·javascript·vue.js
fg_4111 小时前
无网络安装ionic和运行
前端·npm
理想不理想v1 小时前
‌Vue 3相比Vue 2的主要改进‌?
前端·javascript·vue.js·面试
酷酷的阿云1 小时前
不用ECharts!从0到1徒手撸一个Vue3柱状图
前端·javascript·vue.js
微信:137971205871 小时前
web端手机录音
前端
齐 飞1 小时前
MongoDB笔记01-概念与安装
前端·数据库·笔记·后端·mongodb
神仙别闹1 小时前
基于tensorflow和flask的本地图片库web图片搜索引擎
前端·flask·tensorflow
aPurpleBerry2 小时前
JS常用数组方法 reduce filter find forEach
javascript
GIS程序媛—椰子2 小时前
【Vue 全家桶】7、Vue UI组件库(更新中)
前端·vue.js