一、写在前面
这篇文章的代码使用karma,mocha,chai,sinon-chai配合Vue的实例属性进行单元测试
二、全局的组件的坑
由于我的g-icon是全局注册的,所以使用g-input组件时的时候g-icon是直接用的,所以测试时有关icon的代码永远是错的。
把g-icon局部注册的组件
三、在测试中触发点击事件
模拟我在app.vue里使用g-input组件
<g-input v-model="message"></g-input>
使用new event 和 dispatch 模拟事件在组件上触发,虽然这个事件和我们实际的事件不一样,但名字一样就够了,测试回调函数自带的参数
it("支持事件", () => { ["change", "input", "focus", "blur"].forEach(eventName => { vm = new Constructor({}).$mount(); const callback = sinon.fake(); vm.$on(eventName, callback); let event = new Event(eventName); Object.defineProperty(event, "target", { value: { value: "hi" }, enumerable: true }); let inputElement = vm.$el.querySelector("input"); inputElement.dispatchEvent(event); expect(callback).to.have.been.calledWith("hi"); }); });
测试这个组件事件触发时,回调的参数,由于自定义事件没有target,我们需要自己写上去
value: { value: "hi" }第一个value是defineProperty的
四、Vue的版本
坑来自于下面一段代码
it("接受gutter", function(done) { Vue.component("g-row", Row); Vue.component("g-col", Col); const div = document.createElement("div"); document.body.appendChild(div); div.innerHTML = ` <g-row gutter="20"> <g-col></g-col> <g-col></g-col> </g-row>`; const vm = new Vue({ el: div }); setTimeout(() => { const row = vm.$el.querySelector(".row"); expect(getComputedStyle(row).marginRight).to.eq("-10px"); expect(getComputedStyle(row).marginLeft).to.eq("-10px"); const cols = vm.$el.querySelectorAll(".col"); expect(getComputedStyle(cols[0]).paddingRight).to.eq("10px"); expect(getComputedStyle(cols[1]).paddingLeft).to.eq("10px"); done(); vm.$el.remove(); vm.$destroy(); }, 0); });
我使用直接在el上写入template代码,所以我默认的import Vue from "vue"(runtimeonly版本)无法编译这个代码,import Vue from "../node_modules/vue/dist/vue.esm.js"使用上面引入即可
在没有template选项是,el不替换
五、异步测试
还是这个代码,先看以下测试两个组件关系
it("接受gutter", function(done) { Vue.component("g-row", Row); Vue.component("g-col", Col); const div = document.createElement("div"); document.body.appendChild(div); div.innerHTML = ` <g-row gutter="20"> <g-col></g-col> <g-col></g-col> </g-row>`; const vm = new Vue({ el: div }); setTimeout(() => { const row = vm.$el.querySelector(".row"); expect(getComputedStyle(row).marginRight).to.eq("-10px"); expect(getComputedStyle(row).marginLeft).to.eq("-10px"); const cols = vm.$el.querySelectorAll(".col"); expect(getComputedStyle(cols[0]).paddingRight).to.eq("10px"); expect(getComputedStyle(cols[1]).paddingLeft).to.eq("10px"); done(); vm.$el.remove(); vm.$destroy(); }, 0); });
先说为什么需要seTimeout
从created和mounted钩子说起,createElement和appendChild在js代码是同步的,两个钩子分别在这两段代码后执行,钩子异步执行的。
由于我们在g-row组件中有mounted钩子,所以我们必须得进行异步检测,否则我们在new Vue之后立马进行测试,钩子还没执行完。
mocha异步测试
mocha默认不执行异步,加入done参数,调用done()就可以
六、垃圾回收
每一个测试完成之后,都要写下面两条代码
vm.$el.remove(); vm.$destroy();
有两个作用:
- 销毁在页面中的数据
- 销毁在内存的数据
虽然js是单线程,但是还有一个dom线程
var div = document. getElementById('xxx') div.onclick = function() { ///code } setTimeout(function(){ div. remove() }, 3000)
现在我们讨论,什么时候div上的函数被回收
函数被全局变量div上的onlick引用了
div.remove()只是在页面删掉了,没有被内存删掉
var div = document. getElementById('xxx') div.onclick = function() { ///code } setTimeout(function(){ div = mull }, 3000)
这个函数并没有被删除,函数是写在dom上的,div变量只是引用了dom对象
var div = document. getElementById('xxx') div.onclick = function() { ///code } setTimeout(function(){ var div2 = document. getElementById('xxx') }, 3000)
div= null和div.remove同时做就可以了,分别从内存和dom上删除了
ie有bug,即使这样都删不了,div.onlick = null 可以
Vue,单元测试
RTX 5090要首发 性能要翻倍!三星展示GDDR7显存
三星在GTC上展示了专为下一代游戏GPU设计的GDDR7内存。
首次推出的GDDR7内存模块密度为16GB,每个模块容量为2GB。其速度预设为32 Gbps(PAM3),但也可以降至28 Gbps,以提高产量和初始阶段的整体性能和成本效益。
据三星表示,GDDR7内存的能效将提高20%,同时工作电压仅为1.1V,低于标准的1.2V。通过采用更新的封装材料和优化的电路设计,使得在高速运行时的发热量降低,GDDR7的热阻比GDDR6降低了70%。
更新日志
- 三国志8重制版恶名怎么消除 恶名影响与消除方法介绍
- 模拟之声慢刻CD《柏林之声5》2019[原抓WAV+CUE]
- AlexandraSoumm-Parisestunefte(2024)[24Bit-96kHz]FLAC
- 李嘉《国语转调1》[天王唱片][WAV整轨]
- 不是哥们 这都能跑?网友展示用720显卡跑《黑神话》
- 玩家自制《黑神话:悟空》亢金星君3D动画 现代妆容绝美
- 大佬的审美冲击!《GTA6》环境设计师展示最新作品
- 纪晓君.2001-野火·春风【魔岩】【WAV+CUE】
- 汪峰.2005-怒放的生命【创盟音乐】【WAV+CUE】
- 群星.1995-坠入情网【宝丽金】【WAV+CUE】
- 群星《谁杀死了Hi-Fi音乐》涂鸦精品 [WAV+CUE][1G]
- 群星1998《宝丽金最精彩98》香港首版[WAV+CUE][1G]
- 汪峰《也许我可以无视死亡》星文[WAV+CUE][1G]
- 李嘉-1991《国语转调2》[天王唱片][WAV整轨]
- 蔡琴2008《金声回忆录101》6CD[环星唱片][WAV整轨]