不要再写indexOf>-1了,太LOW啊🤔

JavaScript提供了很多原生好用的操作符,在开发过程中如果能合理的使用肯定能达到事半功倍的效果。我们的目标是能少些一个字符,绝不多写 😁

背景

在日常开发中,我们经常会遇到需要检查字符串中是否包含某个子串的情况。通常的做法是使用indexOf()方法,并与-1比较,但这种方法在可读性方面并非最佳。下面我们来看看,如何利用原生操作符使代码更加简洁、易读。

按位非操作符(~)

使用~操作符

rust 复制代码
if (~str.indexOf("subStr")) {
  // 子字符串存在
} else {
  // 子字符串不存在
}

原理分析~操作符

为什么使用~操作符后,就不用写>-1了呢?我们来看看其背后的隐式转换和~操作符原理

  • 不存在时得到的结果是: ~-1

step1. 转成32位的二进制:11111111111111111111111111111111

注:在二进制中**-1**表示为所有位都是1的二进制数

step2. 按位取反:00000000000000000000000000000000

  • 存在时得到的结果,假设是3

step1. 转成32位的二进制00000000000000000000000000000011

step2. 按位取反11111111111111111111111111111100

step3. 二进制到十进制的转换

取反后得到的是一个新的二进制数,但这个二进制数以1开头,表示它是一个负数。在计算机中,负数通常使用补码形式表示,所以我们需要将这个二进制数转换为它的补码对应的十进制数。

补码的计算过程是:

  1. 取反(已经在步骤2里完成了)。
  2. 加1。
markdown 复制代码
11111111111111111111111111111100
+                                 1
-----------------------------------
11111111111111111111111111111101

得到的新二进制数仍然是负数的补码表示,它对应的十进制数是-4。

小结:这个过程也可以简单的通过数学表达式来表示:~N = -N - 1,可以理解为这个操作符是赔了夫人又折兵
梳理

  1. 使用~str.indexOf(xxx)后得到的结果一定是小于等于0的数字
  2. 而if括号内的表单式会将数字隐式转换为布尔值
  3. 因此只有~-1 ==> 0 ==> false,其它情况都是true

非常有趣的隐式转换

为什么加!后,结果不变

less 复制代码
[] == 0      // true
![] == 0      // true

分析:

在[] == 0中,对于复杂类型转化过程是先执行toString再通过Number转成数字,因此结果是Number([].toString())==0

在![] == 0中,![]优先执行将数组转成布尔值再取反返回false,再转成数字,因此结果也是0

为什么"5">15为false,而"5">"15"为true

原因是:两个字符串数字比较的不是数字本身,而是通过charCodeAt获取到的Unicode编码的索引:

非常容易踩坑的引用类型隐式转换

less 复制代码
[] == [] //false(引用地址不一样、无异议)
[] == ![] //true(比较的转化后的数字)

{} == {} //false同上
{} ==!{} //false,哈哈和上面的数组对比结果是相反的

小结:隐式转换有时候容易让人产生误解,但也是有迹可循,把握住要点、就不会出错,如果大家有遇到什么有意思的隐式转换,欢迎留言讨论。。。

总结

充分了解掌握操作符和隐式转换的行为有助于我们编写更可靠、更易于维护 的代码。但在实际开发中,应合理的使用隐式转换,并在可能引起混淆的地方采用显式类型转换,以提高代码的清晰度和稳定性。

另外,合理的使用操作符可以帮助我们编写更简洁的代码,还可以提高代码的可读性 ( 如+操作符快捷转换数字、??控制合并操作符对处理数字0非常有用)。

相关推荐
Hi_kenyon8 分钟前
理解vue中的ref
前端·javascript·vue.js
jin1233221 小时前
基于React Native鸿蒙跨平台地址管理是许多电商、外卖、物流等应用的重要功能模块,实现了地址的添加、编辑、删除和设置默认等功能
javascript·react native·react.js·ecmascript·harmonyos
2501_920931701 小时前
React Native鸿蒙跨平台医疗健康类的血压记录,包括收缩压、舒张压、心率、日期、时间、备注和状态
javascript·react native·react.js·ecmascript·harmonyos
落霞的思绪2 小时前
配置React和React-dom为CDN引入
前端·react.js·前端框架
Hacker_Z&Q2 小时前
CSS 笔记2 (属性)
前端·css·笔记
Anastasiozzzz2 小时前
LeetCode Hot100 295. 数据流的中位数 MedianFinder
java·服务器·前端
橙露2 小时前
React Hooks 深度解析:从基础使用到自定义 Hooks 的封装技巧
javascript·react.js·ecmascript
Exquisite.2 小时前
Nginx
服务器·前端·nginx
2501_920931702 小时前
React Native鸿蒙跨平台使用useState管理健康记录和过滤状态,支持多种健康数据类型(血压、体重等)并实现按类型过滤功能
javascript·react native·react.js·ecmascript·harmonyos
打小就很皮...3 小时前
dnd-kit 实现表格拖拽排序
前端·react.js·表格拖拽·dnd-kit