优雅的JavaScript代码

目录

  • 常见的JavaScript 使用场景及技巧
  • 总结

1 条件短路

primary

//如果相爱,就在一起
if (love) {
  goTogether();
}

senior

//如果相爱,就在一起
love && goTogether()

常用于判断条件符合,便执行下一步操作。

2 空值合并运算符

primary

let foo;
if (foo === null || foo === undefined) {
  foo = 'default string';
}
console.log(foo);
// expected output: "default string"

senior

const foo = null ?? 'default string';
console.log(foo);
// expected output: "default string"

const bar = 0
console.log(bar||100) //100
cosole.log(bar ?? 100) //0

空值合并操作符(??)是一个逻辑操作符,当左侧的操作数为 null 或者 undefined 时,返回其右侧操作数,否则返回左侧操作数。

3 数组去重

primary

/** ES5 */
function unique(arr){
    var newArr = []
    for(var i = 0; i < arr.length; i++){
        if(newArr.indexOf(arr[i]) == -1){
            newArr.push(arr[i])
        }
    }
    return newArr
}    

senior

/** ES6 */
let unique = (arr) => [...new Set(arr)]

推荐ES6的写法。

4 模板字符串

primary

var message = 'Hello ' + name + ', it\'s ' + time + ' now'

senior

const message = `Hello ${name}, it's ${time} now`

推荐ES6的写法。

5 解构

primary

var data = { name: 'cat', age: 1 }
var name = data.name,
    age = data.age

senior

const data = {name:'cat', age:1} 
const {name, age} = data 

推荐ES6的写法。

6 数组求和

primary

var foo = [1, 2, 3, 4, 5];

//不优雅
function sum(arr){
    var x = 0;
    for(var i = 0; i < arr.length; i++){
        x += arr[i];
    }
    return x;
}
sum(foo) // => 15

senior

//优雅
foo.reduce((a, b) => a + b) // => 15

推荐reduce的写法。

7 函数式编程

primary

//命令式
let feds = [];
for(let i = 0; i < companies.length; i++){
    feds.push(companies[i].fe)
}

senior

//声明式
let feds = companies.map(item => item.fe);

命令式是指通过指令让计算机进行一系列操作,这样做的缺点繁杂,因为是你要先声明一个实例数组feds,再进行for循环遍历,遍历里面再进行if逻辑判断,再push到feds数组里。相反,声明式写法则相对优雅很多,如何进行计数器迭代,返回的数组如何收集,这些细节都隐藏了起来,它指明的是做什么,而不是怎么做。这种声明式写法也是函数式编程的一个很大的优点。

8 对象解构来模拟命名参数

如果你需要将一系列可选项作为参数传入函数,那么你也许倾向于使用了一个对象(Object)来定义配置(Config)。

primary

// 函数定义
function doSomething(config) {
    const foo = config.foo !== undefined ? config.foo : 'Hi';
    const bar = config.bar !== undefined ? config.bar : 'Yo!';
    const baz = config.baz !== undefined ? config.baz : 13;
    // ...
}
// 调用
doSomething({ foo: 'Hello', bar: 'Hey!', baz: 42 });

senior

function doSomething({ foo = 'Hi', bar = 'Yo!', baz = 13 }) {
 // ...
}

如果你想让这个参数是可选的,可以这样:

function doSomething({ foo = 'Hi', bar = 'Yo!', baz = 13 } = {}) {
 // ...
}

9 if else判断

假设规定a和b有如下关系:

a = 5     b = 3

a = 10   b = 6

a = 12   b = 9

a = 15   b = 4

其余情况下,b的值始终等于0


primary

let b = 0; 
if(a == 5){ 
  b = 3; 
} else if(a == 10){ 
  b = 6; 
} else if(a == 12){ 
  b = 9; 
} else if(a == 15){ 
  b = 4; 
} else { 
  b = 0; 
} 

senior

let b = 0; 
switch(a){ 
    case 5 : 
      b = 3; 
      break; 
    case 10 : 
      b = 6; 
      break; 
    case 12 : 
      b = 9; 
      break; 
    case 15 : 
      b = 4; 
      break; 
    default : 
      b = 0; 
}

senior+

let b = (a==5 && 3) || (a==10 && 6) || (a==12 && 9) || (a==15 && 4) || 0;

senior++

let b={'5':1,'10':2,'12':3,'15':4}[a] || 0; 

平时做业务经常遇到判断,这种写法可拓展到自己的平时写业务代码上面。

10 函数劫持

在一个函数运行之前把它劫持下来,添加我们想要的功能。

// 函数定义
function A () {
    console.log("我是原函数");
}
/**
 * 
 * @param {*要劫持的函数所在的对象} obj 
 * @param {*计划要劫持的函数名} method 
 * @param {*回调函数} fun 
 */
const hijack = (obj, method, fun) => {
    let orig = obj[method];//保存原函数
    obj[method] = fun(orig);//将重写的函数作为回调函数的返回值赋给原函数变量
}
hijack(window,'A',function(orig){
    return function (){
        //做任何你想函数A执行时候你想做的事情
        console.log("我劫持了函数A");
        orig.call(this);
    }
})
A();

函数劫持也是常见的钩子函数的原理之一。

11 柯里化

柯里化,英语:Currying,是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术。

假设有一个curry的函数:

// 函数定义
function add(a, b) {
    return a + b;
}

let add1 = add.curry(1);
console.log(add1(5)); // 6
console.log(add1(2)); // 3

怎么实现这样一个curry的函数?它的重点是要返回一个函数,这个函数有一些闭包的变量记录了创建时的默认参数,然后执行这个返回函数的时候,把新传进来的参数和默认参数拼一下变成完整参数列表去调原本的函数,所以有了以下代码:

Function.prototype.curry = function() {
    let defaultArgs = arguments;
    let that = this;
    return function(){
        return that.apply(this, 
          defulatArgs.concat(arguments));
    }
}

但是由于参数不是一个数组,没有concat函数,所以需要把伪数组转成一个数组,可以用Array.prototype.slice:

Function.prototype.curry = function() {
    let slice = Array.prototype.slice;
    let defaultArgs = slice.call(arguments);
    let that = this;
    return function() {
        return that.apply(this, 
            defaultArgs.concat(slice.call(argumens)));    
        }
    };

总结

平时工作中大多数时间都在忙着实现业务,笔者觉得更应该多总结一些技巧和更好的写法,不断鞭策自己,提高自己的技术能力。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容