撸代码时出现了一个奇怪的 Bug 。
主要逻辑是用 axios 获取后端接口返回的状态码,如果成功则将表单值置为空,以便下次添加。
但是当在控制台输出的时候,输出的对象是先经过主函数的 then 中的语句更改过的对象,但是如果读取其中的属性值,却是未经过更改的值。
测试的执行顺序也是输出的 1,2,3 的顺序(先是执行 dispatch 之前的语句,接着执行 action,最后执行 then 中的语句)。
有一点实在想不通:为啥还没运行到赋值语句,直接输出对象会是赋值后的样子。而且只有输出对象时,控制台显示对象的属性值是更改,但直接对象中的属性值时却未改变。
这是主函数:
addBookmark() { console.log(1) // 1, 测试,用于测试执行顺序 console.log(this.newBookmarkInfo) // 获取到的对象是经过过 then 中修改之后的对象 {bookmarkId:0, bookmarkTitle:1, bookmarkUrl:1} console.log(this.newBookmarkInfo.bookmarkTitle) // 获取到的是未经更改的 title: "title" this.$store.dispatch('addBookmark',this.newBookmarkInfo).then(() => { // 调用 Action console.log(this.newBookmarkInfo) // 同样是值更改过的对象 console.log(this.newBookmarkInfo.bookmarkTitle) // 获取到的是未经更改的对象中的 title: "title" // 赋值语句 this.newBookmarkInfo.bookmarkTitle='1' this.newBookmarkInfo.bookmarkUrl='1' console.log(3) // 3 // 对表单状态和弹窗状态的重置 this.$refs.obs.reset() this.dialog = false }).catch(error => { console.log('error') console.log(error.response) }) },
这是 vuex 中的 Action:
addBookmark(undefine, newBookmarkInfo){ console.log(2) // 2 console.log(newBookmarkInfo) // 同样也是被更改过的对象 console.log(newBookmarkInfo.bookmarkTitile) // 获取到的是未经过更改的 title return new Promise((resolve,reject) => { addBookmarkOne(newBookmarkInfo).then(resp => { const { data } = resp resolve(data) }).catch(error => { reject(error) }) }) },
这是运行的控制台截图:
![]() | 1 shintendo 2020-10-29 21:01:37 +08:00 ![]() 简单地说,chrome 在 console.log 输出对象的时候,输出的是对象的引用,而非对象内容的“快照”,你点+号展开的时候才对内容求值 |
![]() | 2 shintendo 2020-10-29 21:03:01 +08:00 ![]() 补充,解决这个问题的一个方法是 console.log(JSON.parse(JSON.stringify(obj))) |
3 wzl20001001 OP @shintendo 懂了,谢谢大佬。一直以为是代码的问题,没往这方面想。 |
5 azcvcza 2020-10-30 09:45:57 +08:00 火狐还是什么浏览器就严谨一点,到了对象该输出的时候才输出。chrome 的话,如果一开始一个{},打印的时候没值,但异步之后有值,那么之前打印的那条也会变成有值 |
8 wzl20001001 OP @SxqSachin 今天晚上搞其他的组件的时候发现问题了,在值传递的时候没有进行对象拷贝,直接把源对象赋值给了变量,导致了加入 vuex 的一直是对源对象的引用。 |