QCon 演讲火热征集中,快来分享技术实践与洞见! 了解详情
写点什么

学会这些 JS 小技巧,提升编码幸福度

  • 2019-09-05
  • 本文字数:5647 字

    阅读完需:约 19 分钟

学会这些JS小技巧,提升编码幸福度

本文主要介绍一些 JS 中用到的小技巧,可以在日常 Coding 中提升幸福度~

类型强制转换

string 强制转换为数字

可以用*1来转化为数字(实际上是调用.valueOf方法)


然后使用Number.isNaN来判断是否为NaN,或者使用 a !== a 来判断是否为NaN,因为 NaN !== NaN


'32' * 1            // 32'ds' * 1            // NaNnull * 1            // 0undefined * 1    // NaN1  * { valueOf: ()=>'3' }        // 3
复制代码


常用:也可以使用+来转化字符串为数字


+ '123'            // 123+ 'ds'               // NaN+ ''                    // 0+ null              // 0+ undefined    // NaN+ { valueOf: ()=>'3' }    // 3
复制代码

object 强制转化为 string

可以使用 字符串+Object 的方式来转化对象为字符串(实际上是调用 .toString() 方法)


'the Math object:' + Math                // "the Math object:[object Math]"'the JSON object:' + JSON              // "the JSON object:[object JSON]"
复制代码


当然也可以覆盖对象的toStringvalueOf方法来自定义对象的类型转换:


2  * { valueOf: ()=>'3' }                // 6'J' + { toString: ()=>'S' }                // "JS"
复制代码


《Effective JavaScript》P11:当+用在连接字符串时,当一个对象既有toString方法又有valueOf方法时候,JS 通过盲目使用valueOf方法来解决这种含糊。


对象通过valueOf方法强制转换为数字,通过toString方法强制转换为字符串


'' + {toString:()=>'S',valueOf:()=>'J'}                // J
复制代码

使用 Boolean 过滤数组中的所有假值

我们知道 JS 中有一些假值:falsenull0""undefinedNaN,怎样把数组中的假值快速过滤呢,可以使用 Boolean 构造函数来进行一次转换


const compact = arr => arr.filter(Boolean)compact([0, 1, false, 2, '', 3, 'a', 'e' * 23, NaN, 's', 34])             // [ 1, 2, 3, 'a', 's', 34 ]
复制代码

双位运算符 ~~

可以使用双位操作符来替代 Math.floor( )。双否定位操作符的优势在于它执行相同的操作运行速度更快。


Math.floor(4.9) === 4      //true// 简写为:~~4.9 === 4      //true
复制代码


不过要注意,对整数来说 ~~ 运算结果与 Math.floor( ) 运算结果相同,而对于负数来说不相同:


~~4.5            // 4Math.floor(4.5)        // 4~~-4.5        // -4Math.floor(-4.5)        // -5
复制代码

短路运算符

我们知道逻辑与&&与逻辑或||是短路运算符,短路运算符就是从左到右的运算中前者满足要求,就不再执行后者了;


可以理解为:


  • &&为取假运算,从左到右依次判断,如果遇到一个假值,就返回假值,以后不再执行,否则返回最后一个真值。

  • ||为取真运算,从左到右依次判断,如果遇到一个真值,就返回真值,以后不再执行,否则返回最后一个假值。


let param1 = expr1 && expr2let param2 = expr1 || expr2
复制代码



因此可以用来做很多有意思的事,比如给变量赋初值:


let variable1let variable2 = variable1  || 'foo'    
复制代码


如果 variable1 是真值就直接返回了,后面短路就不会被返回了,如果为假值,则会返回后面的foo


也可以用来进行简单的判断,取代冗长的if语句:


let variable = param && param.prop
复制代码


如果param如果为真值则返回param.prop属性,否则返回param这个假值,这样在某些地方防止paramundefined的时候还取其属性造成报错。

取整 | 0

对一个数字| 0可以取整,负数也同样适用,num | 0


1.3 | 0         // 1-1.9 | 0        // -1
复制代码

判断奇偶数 & 1

对一个数字& 1可以判断奇偶数,负数也同样适用,num & 1


const num=3;!!(num & 1)          // true!!(num % 2)          // true
复制代码

函数

函数默认值

func = (l, m = 3, n = 4 ) => (l * m * n);func(2)             //output: 24
复制代码


注意,传入参数为undefined或者不传入的时候会使用默认参数,但是传入null还是会覆盖默认参数。

强制参数

默认情况下,如果不向函数参数传值,那么 JS 会将函数参数设置为undefined。其它一些语言则会发出警告或错误。要执行参数分配,可以使用if语句抛出未定义的错误,或者可以利用强制参数


mandatory = ( ) => {  throw new Error('Missing parameter!');}foo = (bar = mandatory( )) => {            // 这里如果不传入参数,就会执行manadatory函数报出错误  return bar;}
复制代码

隐式返回值

返回值是我们通常用来返回函数最终结果的关键字。只有一个语句的箭头函数,可以隐式返回结果(函数必须省略大括号{ },以便省略返回关键字)。


要返回多行语句(例如对象文本),需要使用( )而不是{ }来包裹函数体。这样可以确保代码以单个语句的形式进行求值。


function calcCircumference(diameter) {  return Math.PI * diameter}// 简写为:calcCircumference = diameter => (  Math.PI * diameter;)
复制代码

惰性载入函数

在某个场景下我们的函数中有判断语句,这个判断依据在整个项目运行期间一般不会变化,所以判断分支在整个项目运行期间只会运行某个特定分支,那么就可以考虑惰性载入函数


function foo(){    if(a !== b){        console.log('aaa')    }else{        console.log('bbb')    }}
// 优化后function foo(){ if(a != b){ foo = function(){ console.log('aaa') } }else{ foo = function(){ console.log('bbb') } } return foo();}
复制代码


那么第一次运行之后就会覆写这个方法,下一次再运行的时候就不会执行判断了。当然现在只有一个判断,如果判断很多,分支比较复杂,那么节约的资源还是可观的。

一次性函数

跟上面的惰性载入函数同理,可以在函数体里覆写当前函数,那么可以创建一个一次性的函数,重新赋值之前的代码相当于只运行了一次,适用于运行一些只需要执行一次的初始化代码


var sca = function() {    console.log('msg')    sca = function() {        console.log('foo')    }}sca()        // msgsca()        // foosca()        // foo
复制代码

代码复用

Object [key]

虽然将 foo.bar 写成 foo ['bar'] 是一种常见的做法,但是这种做法构成了编写可重用代码的基础。许多框架使用了这种方法,比如 element 的表单验证


请考虑下面这个验证函数的简化示例:


function validate(values) {  if(!values.first)    return false;  if(!values.last)    return false;  return true;}console.log(validate({first:'Bruce',last:'Wayne'})); // true
复制代码


上面的函数完美的完成验证工作。但是当有很多表单,则需要应用验证,此时会有不同的字段和规则。如果可以构建一个在运行时配置的通用验证函数,会是一个好选择。


// object validation rulesconst schema = {  first: {    required:true  },  last: {    required:true  }}
// universal validation functionconst validate = (schema, values) => { for(field in schema) { if(schema[field].required) { if(!values[field]) { return false; } } } return true;}console.log(validate(schema, {first:'Bruce'})); // falseconsole.log(validate(schema, {first:'Bruce',last:'Wayne'})); // true
复制代码


现在有了这个验证函数,我们就可以在所有窗体中重用,而无需为每个窗体编写自定义验证函数。

数字字面量

不同进制表示法

ES6 中新增了不同进制的书写格式,在后台传参的时候要注意这一点。


29            // 10进制035            // 8进制29      原来的方式0o35            // 8进制29      ES6的方式0x1d            // 16进制290b11101            // 2进制29
复制代码

精确到指定位数的小数

将数字四舍五入到指定的小数位数。使用 Math.round() 和模板字面量将数字四舍五入为指定的小数位数。 省略第二个参数 decimals ,数字将被四舍五入到一个整数。


const round = (n, decimals = 0) => Number(`${Math.round(`${n}e${decimals}`)}e-${decimals}`)round(1.345, 2)         // 1.35round(1.345, 1)         // 1.3
复制代码

数字补 0 操作

感谢网友 @JserWang @vczhan 提供 这个小技巧


有时候比如显示时间的时候有时候会需要把一位数字显示成两位,这时候就需要补 0 操作,可以使用slice和 string 的padStart方法


const addZero1 = (num, len = 2) => (`0${num}`).slice(-len)const addZero2 = (num, len = 2) => (`${num}`).padStart( len   , '0')addZero1(3) // 03 addZero2(32,4)  // 0032
复制代码

数组

reduce 方法同时实现 map 和 filter

假设现在有一个数列,你希望更新它的每一项(map 的功能)然后筛选出一部分(filter 的功能)。如果是先使用 map 然后 filter 的话,你需要遍历这个数组两次。


在下面的代码中,我们将数列中的值翻倍,然后挑选出那些大于 50 的数。


const numbers = [10, 20, 30, 40];const doubledOver50 = numbers.reduce((finalList, num) => {  num = num * 2;  if (num > 50) {    finalList.push(num);  }  return finalList;}, []);doubledOver50;            // [60, 80]
复制代码

统计数组中相同项的个数

很多时候,你希望统计数组中重复出现项的个数然后用一个对象表示。那么你可以使用 reduce 方法处理这个数组。


下面的代码将统计每一种车的数目然后把总数用一个对象表示。


var cars = ['BMW','Benz', 'Benz', 'Tesla', 'BMW', 'Toyota'];var carsObj = cars.reduce(function (obj, name) {  obj[name] = obj[name] ? ++obj[name] : 1;  return obj;}, {});carsObj; // => { BMW: 2, Benz: 2, Tesla: 1, Toyota: 1 }
复制代码

使用解构来交换参数数值

有时候你会将函数返回的多个值放在一个数组里。我们可以使用数组解构来获取其中每一个值。


let param1 = 1;let param2 = 2;[param1, param2] = [param2, param1];console.log(param1) // 2console.log(param2) // 1
复制代码


当然我们关于交换数值有不少其他办法:


var temp = a; a = b; b = temp                b = [a, a = b][0]                         a = a + b; b = a - b; a = a - b      
复制代码

接收函数返回的多个结果

在下面的代码中,我们从/post 中获取一个帖子,然后在/comments 中获取相关评论。由于我们使用的是 async/await,函数把返回值放在一个数组中。而我们使用数组解构后就可以把返回值直接赋给相应的变量。


async function getFullPost(){  return await Promise.all([     fetch('/post'),     fetch('/comments')  ]);}const [post, comments] = getFullPost();
复制代码

将数组平铺到指定深度

使用递归,为每个深度级别 depth 递减 1 。 使用 Array.reduce()Array.concat() 来合并元素或数组。 基本情况下,depth 等于 1 停止递归。 省略第二个参数,depth 只能平铺到 1 (单层平铺) 的深度。


const flatten = (arr, depth = 1) =>  depth != 1    ? arr.reduce((a, v) => a.concat(Array.isArray(v) ? flatten(v, depth - 1) : v), [])    : arr.reduce((a, v) => a.concat(v), []);flatten([1, [2], 3, 4]);                         // [1, 2, 3, 4]flatten([1, [2, [3, [4, 5], 6], 7], 8], 2);           // [1, 2, 3, [4, 5], 6, 7, 8]
复制代码

数组的对象解构

数组也可以对象解构,可以方便的获取数组的第 n 个值


const csvFileLine = '1997,John Doe,US,john@doe.com,New York';const { 2: country, 4: state } = csvFileLine.split(',');
country // USstate // New Yourk
复制代码

对象

使用解构删除不必要属性

有时候你不希望保留某些对象属性,也许是因为它们包含敏感信息或仅仅是太大了(just too big)。你可能会枚举整个对象然后删除它们,但实际上只需要简单的将这些无用属性赋值给变量,然后把想要保留的有用部分作为剩余参数就可以了。


下面的代码里,我们希望删除_internal 和 tooBig 参数。我们可以把它们赋值给 internal 和 tooBig 变量,然后在 cleanObject 中存储剩下的属性以备后用。


let {_internal, tooBig, ...cleanObject} = {el1: '1', _internal:"secret", tooBig:{}, el2: '2', el3: '3'};
console.log(cleanObject); // {el1: '1', el2: '2', el3: '3'}
复制代码

在函数参数中解构嵌套对象

在下面的代码中,engine 是对象 car 中嵌套的一个对象。如果我们对 engine 的 vin 属性感兴趣,使用解构赋值可以很轻松地得到它。


var car = {  model: 'bmw 2018',  engine: {    v6: true,    turbo: true,    vin: 12345  }}const modelAndVIN = ({model, engine: {vin}}) => {  console.log(`model: ${model} vin: ${vin}`);}modelAndVIN(car); // => model: bmw 2018  vin: 12345
复制代码

代码复用

Object [key]

虽然将 foo.bar 写成 foo ['bar'] 是一种常见的做法,但是这种做法构成了编写可重用代码的基础。许多框架使用了这种方法,比如 element 的表单验证


请考虑下面这个验证函数的简化示例:


function validate(values) {  if(!values.first)    return false;  if(!values.last)    return false;  return true;}console.log(validate({first:'Bruce',last:'Wayne'})); // true
复制代码


上面的函数完美的完成验证工作。但是当有很多表单,则需要应用验证,此时会有不同的字段和规则。如果可以构建一个在运行时配置的通用验证函数,会是一个好选择。


// object validation rulesconst schema = {  first: {    required:true  },  last: {    required:true  }} // universal validation functionconst validate = (schema, values) => {  for(field in schema) {    if(schema[field].required) {      if(!values[field]) {        return false;      }    }  }  return true;}console.log(validate(schema, {first:'Bruce'})); // falseconsole.log(validate(schema, {first:'Bruce',last:'Wayne'})); // true
复制代码


现在有了这个验证函数,我们就可以在所有窗体中重用,而无需为每个窗体编写自定义验证函数。


网上的帖子大多深浅不一,甚至有些前后矛盾,在下的文章都是学习过程中的总结,如果发现错误,欢迎留言指出~

参考资料


推介阅读:

ES6演示小网站


2019-09-05 13:007471

评论 3 条评论

发布
用户头像
async function getFullPost(){
return await Promise.all([
fetch('/post'),
fetch('/comments')
]);
}
为什么不写成
function getFullPost(){
return Promise.all([
fetch('/post'),
fetch('/comments')
]);
}
展开
2019-09-06 17:02
回复
用户头像
可读性是最重要的,很多团队并不欢迎这些技巧
2019-09-06 09:46
回复
用户头像
你好,请问我可以转载到我的博客上吗?不做商业用途,仅交流学习使用。
2019-09-06 09:27
回复
没有更多了
发现更多内容

Intellij运行Java程序启动等待BUG分享

FunTester

Java IDEA intellij bug

终于拿到了阿里2021年度九大Java技术文档——面试题+文档+白皮书

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

NDK 使用入门

Changing Lin

8月日更

毕业总结

俞嘉彬

架构实战营

爆款阿里P5到P7晋升之路,九大源码文档助我超神果然努力幸运并存

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

对数字化的审视思维

boshi

数字化转型

不愧是阿里内部“SpringCloudAlibaba学习笔记”这细节讲解,神了

Java 程序员 架构 面试 微服务

Java实现基于朴素贝叶斯的情感词分析

码农参上

Java 贝叶斯公式 文本情感分析 8月日更

rosedb 事务实践

roseduan

数据库 事务 存储 KV存储引擎

装机必备的软件里,绝对不能少了这一款 | 剪贴板增强工具

彭宏豪95

效率 工具 windows 实用工具

Swift 仿自如 App 裸眼 3D 效果

fuyoufang

ios swift 8月日更

Vue进阶(三十二):npm install、npm install --save与 npm install --save-dev 区别

No Silver Bullet

Vue 8月日更

阿里大牛最新公开压轴的“Redis深度笔记”,GitHub已标星81.6K

Java~~~

Java redis 架构 面试 架构师

架构实战营毕业总结

Vic

架构实战营

「让我们一起Golang」怎样出让协程资源和设置可用CPU核心数

Regan Yue

高并发 协程 Go 语言 8月日更

【Dubbo3.0技术专题】总体技术体系介绍及技术指南(序章)

洛神灬殇

dubbo RPC 8月日更 Dubbo3

前方高能!Alibaba最新出版的JDK源码剖析手册(究极奥义版)开源

Java~~~

Java 架构 jdk 面试 架构师

架构实战营毕设

Vic

架构实战营

摸鱼小技巧之IDEA调试篇一

4ye

Java debug 后端 IDEA 8月日更

数据库知识整理

十二万伏特皮卡丘

手撸二叉树之二叉树的中序遍历

HelloWorld杰少

数据结构与算法 8月日更

如何做好高并发系统设计

架构精进之路

高并发 8月日更

性能测试框架中实时QPS取样器实现

FunTester

性能测试 测试框架 QPS FunTester 取样器

网络安全证书合集系列

网络安全学海

网络安全 证书 信息安全 渗透测试 安全漏洞

一夜标星过百万,轻松霸榜GitHub的手绘图解HTTP笔记,竟来源阿里

Java~~~

Java 架构 面试 网络 HTTP

【Flutter 专题】67 图解基本约束 Box (二)

阿策小和尚

Flutter 小菜 0 基础学习 Flutter Android 小菜鸟 8月日更

B 端产品经理如何快速成长?

蒋川

数据产品经理 b端产品经理

Alibaba内部首发“面试百宝书+超全算法面试手册”PDF版下载

Java~~~

Java 架构 面试 微服务 JVM

spring 循环依赖与三级缓存

Rubble

springboot 8月日更

京东T7架构师用470页就把微服务架构原理与开发实战文档讲完了

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

被boss直聘转发过多而“封杀”的2021年全套java高级面试题有多牛

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

学会这些JS小技巧,提升编码幸福度_语言 & 开发_HERlocked93_InfoQ精选文章