隐式转换的细节:别再这样if判断对象有没有该属性了!

今天在网上看到一个关于 ((2==true) + 1) 输出多少的问题,我直接就回答2了,然鹅。。。

js 复制代码
((2==true) + 1)  // 1

why??2==true不是1吗,1+1不是2吗。前者的思考是我对if语法判断的误解,因为以前if判断某个对象是否存在一个属性的时候,只要他有值,那就为真,然后就把2当做true了。后面查阅资料后发现,if判断会将判断条件强转为布尔值,要是以此来判断某个对象是否存在一个属性,那这里面就有很多坑了。

js转布尔值有下面四种情况:

  1. 数字→布尔,0和NaN结果是false 其他都是true
  2. 字符串→布尔,空字符串为false,其他都为true,包括空格字符串" "和"0"
  3. null和undefined都会转为false
  4. 引用数据类型都会转为true,包括空数组[]和空对象{}

测试案例:

实际开发中再这样写if代码判断直接得挨开发组长骂了

结论:推荐的判断对象属性的方法还是使用obj.hasOwnProperty(propertyName),这个方法不会顺着原型链查找属性,如果需要查找可以用in操作符


回到最初的那个问题,(2==true)跟if(obj.a)是否有差别呢,答案是,大大的差别。

if判断是直接转布尔类型判断,而双等号的隐式转换规则就多了。

  1. 先看两边有没有NaN,有的话直接返回false
  2. 再看两边有没有布尔值,有就布尔值转数字(true为1,false为0)
  3. 最后看两边有没有字符串又分三种情况。如果对方是对象,则转为字符串(数组会直接转成对应序列字符串,对象会转成'[object Object]');如果对方是数字,字符串转数字(也跟第二点有关);如果对方是字符串,直接比较

这时看最初的那个问题,不就明朗起来了~

((2==true)+1)的隐式转换步骤:

  1. true转为1
  2. 2==true返回false
  3. false+1也进行隐式转换,false转为0
  4. 0+1=1

拓展问题:[ ] == ![ ] 输出什么?

没错,输出的居然是true!!![ ]转字符串是"" 转布尔值是true,这是两个极端。

  1. 先进行布尔转换, ([ ] == ![ ]) ==> ([ ] == false) ==> ([ ] == 0)
  2. 空数组转字符串, ( "" == 0 ) 开始看得出结果了
  3. 字符串转数字, 0 == 0

参考资料:

  1. blog.csdn.net/ximing02071...
  2. www.php.cn/faq/397193....
  3. blog.csdn.net/Leslie_hxd/...
相关推荐
一枚前端小姐姐几秒前
低代码平台表单设计系统架构分析(实战一)
前端·低代码·架构
HelloReader9 分钟前
Tauri 1.0 升级到 Tauri 2.0从“能跑”到“跑得稳”的迁移实战指南(含移动端准备、配置重构、插件化 API、权限系统)
前端
Always_Passion10 分钟前
FE视角下的Referrer全面解析
javascript·面试
JunjunZ23 分钟前
uniapp 文件预览:从文件流到多格式预览的完整实现
前端·uni-app
_Eleven25 分钟前
React 19 深度解析:Actions 与 use API 源码揭秘
前端
七牛云行业应用27 分钟前
大模型接入踩坑录:被 Unexpected end of JSON 折磨三天,我重写了SSE流解析
javascript·人工智能·代码规范
_AaronWong1 小时前
Vue3+Element Plus 通用表格组件封装与使用实践
前端·javascript·vue.js
前端西瓜哥1 小时前
图形编辑器开发:文字排版如何实现自动换行?
前端
代码煮茶1 小时前
JS 异步编程实战 | 从回调地狱到 Promise/Async/Await(附代码 + 面试题)
javascript·面试
全栈老石1 小时前
手写一个无限画布 #3:如何在Canvas 层上建立事件体系
前端·javascript·canvas