为什么consolelog在修改之前就输出修改后的值?
本文讲解JavaScript中修改数组中的数据与console.log问题。
背景
今天写DP,console.log
打印二维数组,发现无论是修改前打印还是修改后打印都是一样的。
1 | arr = [[1,2,3,4,5],[6,7,8,9,10],[11,12,13,14,15],[16,17,18,19,20]]; |
为什么会出现这种现象?如果我非要看修改前的,怎么看?最后有例子
预备知识
console.log
对于六种基本类型是直接输出,对于引用类型是打印引用的快照。
且当有多层时,不会展开多层,而是保存第二层的引用,当你点击展开时才去读取内容。
分析
chrome devtools
会默认预览下一层。
注意“加载、展开、预览”的区别
加载:从内存中加载数据
展开:单击按钮展开数据
预览:未展开时看到的数据
初始是不展开任何层,但预览第一层。
当你点第一个展开键(三角形)时,加载第一层的内容,展开的是第一层,预览第二层(无第二层则只展开第一层)。但因为 chrome devtools
有缓存,即使展开看到修改后的内容,但预览不会变。
点击第二个展开键,加载第二层,展开第二层,预览第三层(没有则只展开第二层)。多层同理。
若想看到未修改的所有内容
若想看到未修改的所有内容,则要在修改语句执行之前就展开后续层级,这时候 chrome devtools
就会输出未修改的内容,可以用 setTImeout
验证,在 setTImeout
的延时函数执行前展开后续层级。
如果是修改语句执行结束了,再去展开,就是修改后的内容。
关闭再展开并不会重新读取内容。
关闭再展开并不会重新读取内容。当 setTimeout
的延迟函数执行前展开所有,会看到未修改的所有内容,预览和展开都正确。当 setTImeoout
执行完后关闭再展开(此时已修改),输出依然不变,依然是修改前的内容,因为已经加载过了就不重新读取了。
怎么解决
虽然 setTimeoout
可以用,但并不是最优解。毕竟每次都要赶时间去展开所有层级。
JSON化
先将对象(二维数组也是对象)转换成json字符串再解析一次。 JSON.parse(JSON.stringify(xxx))
JSON.stringify()
将JavaScript对象或值转换为json字符串
JSON.paarse()
解析json字符串
深拷贝
对每个层级进行一次深拷贝。但这不是明智的做法。
代码演示
1 | const testAny = () => { |
附录
参考