热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[JavaScript]多数前端工程师都没注意到的一个关于console.log()的坑

[JavaScript]多数前端工程师都没注意到的一个关于console.log()的坑请阅读以下代码并猜测结果:functiontest(){le

[Javascript] 多数前端工程师都没注意到的一个关于console.log()的坑
请阅读以下代码并猜测结果:

      function test() {
        let obj = {}, arr=[]
          for(var i = 0; i <4; i++) { obj.first = i obj.secOnd= i+1 obj.three = i+2 arr.push(obj) }
        console.log(arr)
      }
      test()

输出结果为:
这里写图片描述
为什么输出结果不是以下这个?
这里写图片描述

我要说的坑不是这个,因为相信这个引用类型的赋值问题很多伙伴都能轻松解决。以下提供两种解决方案:
(方法一),将let obj={}移到for循环中,每次循环创建一个新对象:

      function test() {
        let arr=[]
          for(var i = 0; i <4; i++) { let obj = {}
            obj.first = i
            obj.secOnd= i+1
            obj.three = i+2
            arr.push(obj)
        }
        console.log(arr)
      }
      test()

能得到预期结果:
这里写图片描述

(方法二):使用ES6新特性Object.assign,轻松实现深复制,这样的好处是不用每次都创建一个新对象:

      function test() {
        let obj= {}, arr=[]
          for(var i = 0; i <4; i++) { obj.first = i obj.secOnd= i+1 obj.three = i+2 arr.push(Object.assign({}, obj))
        }
        console.log(arr)
      }
      test()

同样能得到预期结果:
这里写图片描述

问题都解决了,那么我说的坑是什么呢?
接下来我们将解决了以上问题的代码作出以下修改,将console.log移到循环中,如下:

      function test() {
        let obj= {}, arr=[]
          for(var i = 0; i <4; i++) { obj.first = i obj.secOnd= i+1 obj.three = i+2 arr.push(Object.assign({}, obj))
            console.log(arr)
        }
      }
      test()

请先预测一下结果是什么再看以下答案。
运行结果为:

这里写图片描述
每次循环输出的都是最后的值,是不是有点惊讶?我们再来看一点更惊讶的,在输出数组前数组一下数组的长度console.log(arr.length),如下:

      function test() {
        let obj= {}, arr=[]
          for(var i = 0; i <4; i++) { obj.first = i obj.secOnd= i+1 obj.three = i+2 arr.push(Object.assign({}, obj))
            console.log(arr.length)
            console.log(arr)
        }
      }
      test()

同样,先预测运行结果再看以下答案。
运行结果为:
这里写图片描述
第一次循环输出数组的长度是1,看起来也是只有一个对象,这很符合预期;但是为什么展开后,里边会有四个对象?第二次循环数组长度是2,看起来有两个对象,这符合预期,但是点开后为什么里边也是有四个对象?这就是我说的坑!!!

原因:
使用console.log()输出引用类型值时,console.log的实际执行会推迟,相当于“惰性”求值,遇上数组、对象这样的引用类型就出上面的问题了,也就是我们点击箭头打开的时候,才会输出真正的值,而这时候循环早就已经执行完了,所以输出的会是循环执行完后的结果!

测试
我们可以将arr转成非引用类型的数据格式,来证明我们的想法,如下:

      function test() {
        let obj= {}, arr=[]
          for(var i = 0; i <4; i++) { obj.first = i obj.secOnd= i+1 obj.three = i+2 arr.push(Object.assign({}, obj))
            console.log(arr.length)
            console.log(JSON.parse(JSON.stringify(arr)))
        }
      }
      test()

这里写图片描述
显然,结果已经符合预期,猜测正确!!

完结~ 午休一会~~~


推荐阅读
  • JavaScript概述1.JavaScript定义JavaScript是Netscape公司开发的一种基于对象和事件驱动的脚本语言。它是弱类型语言,只能由浏览器解释执行。其中:脚本语言:解释运行( ... [详细]
  • 本文介绍了如何使用JSONObiect和Gson相关方法实现json数据与kotlin对象的相互转换。首先解释了JSON的概念和数据格式,然后详细介绍了相关API,包括JSONObject和Gson的使用方法。接着讲解了如何将json格式的字符串转换为kotlin对象或List,以及如何将kotlin对象转换为json字符串。最后提到了使用Map封装json对象的特殊情况。文章还对JSON和XML进行了比较,指出了JSON的优势和缺点。 ... [详细]
  • javascript如何判断值是否为undefined
    这篇文章主要介绍“javascript如何判断值是否为undefined”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“ja ... [详细]
  • Whyusingstringsaskeysofarray,consoleisshowingthatarraywithoutthesedeclaredvaluesand ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • Itwasworkingcorrectly,butyesterdayitstartedgiving401.IhavetriedwithGooglecontactsAPI ... [详细]
  • 用JavaScript实现的太空人手表
    用JavaScript实现的太空人手表-JS写的太空人手表,没有用canvas、svg。主要用几个大的函数来动态显示时间、天气这些。天气的获取用到了AJAX请求。代码中有详细的注释 ... [详细]
  • javascript函数中参数传递问题示例探讨-相信每一位刚接触javascript的同学在函数参数传递上都会很疑惑,原因无他,那就是它的语法太怪异了,你定义一个函数例如funct ... [详细]
  • JavaScript实现在页面间传值的方法-本文实例讲述了JavaScript实现在页面间传值的方法。分享给大家供大家参考。具体如下:问题如下:在a.html页面中,的 ... [详细]
  • 本文讨论了在openwrt-17.01版本中,mt7628设备上初始化启动时eth0的mac地址总是随机生成的问题。每次随机生成的eth0的mac地址都会写到/sys/class/net/eth0/address目录下,而openwrt-17.01原版的SDK会根据随机生成的eth0的mac地址再生成eth0.1、eth0.2等,生成后的mac地址会保存在/etc/config/network下。 ... [详细]
  • IhaveawebapplicationthatusesanActiveXCOMcomponent,forexample:我有一个使用ActiveXCOM组件的Web应用程 ... [详细]
  • 表单提交前的最后验证:通常在表单提交前,我们必须确认用户是否都把必须填选的做了,如果没有,就不能被提交到服务器,这里我们用到表单的formname.submit()看演示,其实这个对于我们修炼道 ... [详细]
  • get()方法用于拦截某个属性的读取操作,如果没有该属性的时候会报一个undefined的,如果结果get处理会返回对应的错误信息varperson{name:张三 ... [详细]
  • JavaScript - let和var区别
    前提ES5只有函数作用域和全局作用域,var属于ES5。let属于ES6,新增块级作用域。目的是可以写更安全的代码。Theletstatementdeclaresablocks ... [详细]
  • 但有时候,需要当某事件触发时,我们先做一些操作,然后再跳转,这时,就要用JAVASCRIPT来实现这一跳转功能。下面是具体的做法:一:跳转到新页面,并且是在新窗口中打开时:复制代码代码如下:fu ... [详细]
author-avatar
死了才能爱_403
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有