参加公司技术嘉年华第一季(前端、服务端)的间隙,陈导问了我一个问题:{}+[] 和 []+{}两个表达式的值分别是什么?根据我的理解我觉得结果应该都是"[object Object]",但是结果却并不是这样子的({}+[] = 0  []+{} = '[object Object]'),我就想这不科学呀,一下子颠覆了我所学过的js知识了。于是我决定搞明白这究竟是是什么与原因导致的?

    通过查找资料,发现之所以结果不一样,是因为js解释器把{}当做了代码块语法解析,那么{}+[]就变成+[],这样的话很容易就解释了上面两个表达式的结果了~
    为什么{}在右边就没有被解释成代码块语法呢?这就要讲到+运算符的用法,+可以做数字的加法和字符串的链接,只要有一个是字符串,另外一个就会数据类型转换成字符串,然后进行拼接操作;都为对象的时候,这种情况下会通过valueOf获取对象的值,如果原始值还是对象的话,会通过toString()方法,转化成字符串。
    下面解释一下{}+[]和[]+{}的执行过程:
    {}+[],由+运算符的语法可以知道,会从左向右解析,遇到{}被认为是代码块(语法)被忽略,继续就变成了+[],此时+就不是加法运算符,也不是字符串连接符了,变成了转换数字的运算符了,数组通过valueOf方法获取值还是数组,执行toString方法后变成''空字符串,然后空字符串数据类型转化成数字,就变成了0,也解释了变种{}-[]=-0
    []+{},从左向右解析,[]被转化成''空字符串,{}通过valueOf方法获取原始值还是对象(注意不能通过{}.valurOf()获取,此时{}会被解析成代码块),通过toString方法结果是'[object Object]',接下来解释简单的字符串连接问题了,所以结果是'[object Object]'。注:对象的toString和valueOf都不一样,并且可以自定义。
    扩展:由上面的例子,进而讨论一下JS中数据类型转换的问题
    JS的取值特别灵活,会根据上下文语境,对数据进行隐式的转换(当然也可以通过显示的方法,对数据进行类型转换)。一般情况下,这种数据类型转换多用在运算符运算的时候.js的数据分成两大类,原始值和对象,原始值有:null undefined string number boolean,其他都是对象。js引擎提供三种数据类型转换操作:ToPrimitive(),ToNumber(),ToString()。
    toPrimitive()是将值转化成原始值,有两个参数,第一个需要转换的数据,第二个参数是prefferedType(number或者string),处理的顺序是:首先如果是原始值直接返回原始值,其次执行valueOf方法,如果返回原始值直接返回,如果不是,执行toString方法,转化成字符串。Date对象第二个参数是string(如果第二个参数是string的话,函数的处理顺序应该把toString方法提到第二步),其他对象都是number。
第一个参数 是否是原始值
valueOf
toString
备注
new Obeject()
Object
'[object Object]'
 
new Number(1)
1 '1' 隐式转换不是对象,可用Object.prototype.isPrototypeOf检测
new Blooean(1)
true 'true' 隐式转换不是对象,可用Object.prototype.isPrototypeOf检测
new Date()
1395985229639
‘Fri Mar 28 2014 13:40:29 GMT+0800 (中国标准时间)  ’
 
new Function('console.log(1);')
function anonymous() { console.log(1); }
'function anonymous() { console.log(1); }'
 
...
...
...
...
 
    toNumber将值转化成数字。原始值转换程数字如下表(对象参照toPrimitive ):
原始值 结果
null
+0
undefined
NAN
123
123
'123'
123
true/false
1/+0

toString将值转化成字符串,原始值转化成字符串如下表(对象参照toPrimitive ):

原始值 结果
null
'null'
undefined
'undefined'
123
'123'
'123a' '123a'
true/false
'true'/'false'

上面是对javascript数据类型转换的一点理解,在这里记录一下。

 
参考资料:
《javascript权威指南(第六版)》

{}+[] = ? 和 []+{} = ? 浅谈JS数据类型转换的更多相关文章

  1. 浅谈JS数据遍历的几种方式

    遍历对象(数组)是我们日常撸码的必不可少的部分,如何从性能上优化代码,提高运行效率?下文为你揭开真像: 第一种:普通的for循环 for(j = 0; j < arr.length; j++) ...

  2. 浅谈JS中的!=、== 、!==、===的用法和区别 JS中Null与Undefined的区别 读取XML文件 获取路径的方式 C#中Cookie,Session,Application的用法与区别? c#反射 抽象工厂

    浅谈JS中的!=.== .!==.===的用法和区别   var num = 1;     var str = '1';     var test = 1;     test == num  //tr ...

  3. 浅谈JS之AJAX

    0x00:什么是Ajax? Ajax是Asynchronous Javascript And Xml 的缩写(异步javascript及xml),Ajax是使用javascript在浏览器后台操作HT ...

  4. 浅谈JS中的闭包

    浅谈JS中的闭包 在介绍闭包之前,我先介绍点JS的基础知识,下面的基础知识会充分的帮助你理解闭包.那么接下来先看下变量的作用域. 变量的作用域 变量共有两种,一种为全局变量,一种为局部变量.那么全局变 ...

  5. 浅谈 js 正则字面量 与 new RegExp 执行效率

    原文:浅谈 js 正则字面量 与 new RegExp 执行效率 前几天谈了正则匹配 js 字符串的问题:<js 正则学习小记之匹配字符串> 和 <js 正则学习小记之匹配字符串优化 ...

  6. 浅谈 js 字符串之神奇的转义

    原文:浅谈 js 字符串之神奇的转义 字符串在js里是非常常用的,但是你真的了解它么?翻阅<MDN String>就可以了解它的常见用法了,开门见山的就让你了解了字符串是怎么回事. 'st ...

  7. 浅谈js拖拽

    本文来自网易云社区 作者:刘凌阳 前言 本文依据半年前本人的分享<浅谈js拖拽>撰写,算是一篇迟到的文章. 基本思路 虽然现在关于拖拽的组件库到处都是,HTML5也把拖放纳入了标准.但考虑 ...

  8. 浅谈JS中 var let const 变量声明

    浅谈JS中 var let const 变量声明 用var来声明变量会出现的问题: 1. 允许重复的变量声明:导致数据被覆盖 2. 变量提升:怪异的数据访问.闭包问题 3. 全局变量挂载到全局对象:全 ...

  9. JS 数据类型转换

    JS 数据类型转换 方法主要有三种 转换函数.强制类型转换.利用js变量弱类型转换. 1. 转换函数: js提供了parseInt()和parseFloat()两个转换函数.前者把值转换成整数,后者把 ...

随机推荐

  1. ViewPager顶部标题控件PagerSlidingTabStrip

    最近搞一个项目,要求做一个和网易新闻顶部菜单的滑动效果,如图: 顶部标题中下面有个红色的矩形小条,左右滑动时会跟随手势动态滑动,效果很绚丽,唉,特效啊! 自己搞了一上午无果,还是是github上找大神 ...

  2. [RxJS] Handling Multiple Streams with Merge

    You often need to handle multiple user interactions set to different streams. This lesson shows hows ...

  3. UIImagePickerController从拍照、图库、相册获取图片

    iOS 获取图片有三种方法: 1. 直接调用摄像头拍照 2. 从相册中选择 3. 从图库中选择 UIImagePickerController 是系统提供的用来获取图片和视频的接口: 用UIImage ...

  4. actionBar兼容2.1及以上版本的做法 .

    正在准备一个项目,需要尊重android design的同时还要做到很好的兼容低版本,于是就先从actionBar开始吧. 1,新建一个android工程startActionBar,minSdkVe ...

  5. TableView数据源方法的执行顺序

    UITableView显示数据的过程 1.调用一次tableView:numberOfRowsInSection:方法确定行数 2.调用多次tableView:heightForRowAtIndexP ...

  6. CodeSmith使用总结--读取一个表试试

    我感觉CodeSmith对于我的最大用途是不用我手动创建每个表的Model.BLL和DAL了,那些繁琐的工作真的让我很无语. CodeSmith要读取数据库中的表就要先连接数据库. 新建一个数据库连接 ...

  7. 转载:android——eclipse如何去除Ctrl+shift+R组合键查找到的.class文件

    转载自:http://blog.csdn.net/virgilli/article/details/22500409 AS里面的build文件下一堆的.class 文件,当你要定位资源文件的时候,有些 ...

  8. discuzx3.2伪静态

    首先,我们要新建一个名为.htaccess的文件,文件名为空白,这点很重要.很多人无法新建这个文件,在这里教大家如何新建没有名字的文件. 新建一个TXT文本,名字先默认.然后打开这个文本,然后把我们的 ...

  9. C# 标签打印示例 1

    初次写博客,有哪些不足的地方,还请多多指点,给予建议,谢谢! 如若想要源码,请留言.        本实例是在Webservice 中通过excel做模板来打印标签.具体需求是:一个订单一页纸打印4行 ...

  10. IOS 图片模糊处理 ------ 直接代码 复制出去就可用 值得标记

    1. UIImage *imag = [UIImage imageNamed:@"img"]; /* --------------------使用 coreImg  ------- ...