31 01 2020

1.变量声明const和let

在ES6之前,我们都是用var关键字声明变量。无论声明在何处,都会被视为声明在函数的最顶部(不在函数内即在全局作用域的最顶部)。这就是函数变量提升例如:

function aa(){
    if(flag){
        var test = 'hello man';    
    } else {        
        console.log(test);    
    }
}

以上的代码实际上是:

function aa(){    
    var test; //变量提升函数最顶部    
    if(flag){        
        test = 'hello man';    
    } else {        
        //此处访问test值为undefined        
        console.log(test);    
    }    
    //此处访问test值为undefined
}

所以不用关心flag是否为 true or false。实际上,无论如何 test 都会被创建声明。

接下来ES6主角登场:

我们通常用 letconst 来声明,let 表示变量const 表示常量letconst 都是块级作用域。怎么理解这个块级作用域?

  • 在一个函数内部

  • 在一个代码块内部

说白了只要在{}花括号内的代码块即可以认为 letconst 的作用域。

看以下代码:

function aa(){    
    if(flag){        
        let test = 'hello man';    
    } else {        
        //test 在此处访问不到        
        console.log(test);    
    }
}

let 的作用域是在它所在当前代码块,不会被提升到当前函数的最顶部。


总结例子:

var flag = false;
function func5(){    
    if(flag){        
        var test = "hello word"    
    }else{        
        console.log(test); //test已创建,不会报错,输出undefined    
    }
}
func5();
function func6(){    
    if(flag){        
        let test = "hello word"    
    }else{        
        console.log(test); //报错,test is no defined    
    }
}
func6();


再来说说const

const 声明的变量必须提供一个值,而且会被认为是常量,意思就是它的值被设置完成后就不能再修改了。

const name = 'lux';
name = 'joe';  //再次赋值时会报错

还有,如果 const 的是一个对象,对象所包含的值是可以被修改的。抽象一点儿说,就是对象所指向的地址不能改变,而变量成员是可以修改的。

const student = {name: 'cc'}
//没毛病
student.name = 'yy';
//如果这样子就会报错了
student = {name: 'yy'}


总结例子:

const name = "test"; //定义常量,不可重新赋值
name = 'test1'; //报错 error Assignment to constant variableconst 
student = {name: 'test'};
student.name = "test1"; //常量对象,对象成员可以修改
student = {name: 'test1'}; // 报错error Assignment to constant variable 常量对象本身不能修改


说说TDZ(暂时性死区)

{    
    console.log(value); //报错    
    let value = 'lala';
}

我们都知道,JS引擎扫描代码时,如果发现变量声明,用 var 声明变量时会将声明提升到函数或全局作用域的顶部。但是 let 或者 const,会将声明关进一个小黑屋也是TDZ(暂时性死区),只有执行到变量声明这句语句时,变量才会从小黑屋被放出来,才能安全使用这个变量。


说一道面试题

var funcs = [];
for(var i=0; i<10; i++){    
    funcs.push(function(){        
        console.log(i);    
    })
}
funcs.forEach(function(func) {    
    func();
})

这样的面试题是大家很常见,很多同学一看就知道输出十次10

但是如果我们想依次输出0到9呢?

有两种解决方法,直接看一下代码:

// ES5知识,我们可以利用“立即调用函数”解决这个问题
var funcs = [];
for(var i=0; i<10; i++){    
    funcs.push(
        (function(value) {            
            return function(){                
                console.log(value);            
            }        
        })(i)
    )
}
funcs.forEach(function(func){    
    func();
})


再来看看es6怎么处理的

const funcs = [];
for(let i=0; i< 10; i++){    
    funcs.push(functin(){        
        console.log(i);    
    })
}
funcs.forEach(func => func())

达到相同的效果,ES6 简洁的解决方案是不是更让你心动!!!


数组、对象遍历

const obj = {    
    'key': 'value',    
    'name': 'vae',    
    'age': 24
}
for(let i in obj){    
    console.log(`${i}:${obj[i]}`); //输出 key:value  name:vae  age:24
}
const arr = ['value', 'vae', 24];
for(let in of arr){    
    console.log(i); //输出 value zwb 24
}


2.字符串

先聊聊模板字符串

ES6模板字符简直是开发者的福音啊,解决了 ES5 在字符串功能上的痛点。

第一个用途,基本的字符串格式化。将表达式嵌入字符串中进行拼接。用${}来界定。

//ES5
var name = 'lux';
console.log('hello' + name);
//ES6
const name = 'lux';
console.log(`hello ${name}`);

第二个用途,在ES5时我们通过反斜杠(\)来做多行字符串或者字符串一行行拼接。ES6反引号(``)直接搞定。

//ES5
var msg = "Hi \man!"
//ES6
const msg = `Himan!`

对于字符串 ES6+ 当然也提供了很多厉害也很有意思的方法,说几个常用的。  

// includes 判断是否包含直接返回布尔值
const str = 'hahay';
console.log(str.includes('y')); // true
// repeat 返回重复n次字符串
console.log(str.repeat(3)); // 'hehehe'
// startsWith 和 endsWith 判断是否以给定文本开始或结束
const str = 'hello world!';
console.log(str.startsWith('hello')); // true;
console.log(str.endsWith('!')); // true
// padStart 和 padEnd 填充字符串, 应用场景:时分秒
SetInterval(() => {    
const now = new Date();    
const hours = now.getHours().toString();    
const minutes = now.getMinutes().toString();    
const seconds = now.getSeconds().toString();    
console.log(`${hours}:${minutes}:${seconds}`
); // 1:2:3    
console.log(`${hours.padStart(2, 0)}:${minutes.padStart(2, 0)}:${seconds.padStart(2, 0):}`);    // 01:02:03  不够2位在前面补0    console.log(`${hours.padEnd(2, 0)}:${minutes.padEnd(2, 0)}:${seconds.padEnd(2, 0):}`);    // 10:10:30  不够2位在后面补0}, 1000)


3.函数

函数默认参数

在ES5我们给函数定义参数默认值是怎么样?

function action(num){
    num = num || 200;
    return num;
}

但细心观察的同学们肯定会发现,num传入为0的时候就是false,但是我们实际的需求就是要拿到num = 0,此时num = 200 明显与我们的实际想要的效果明显不一样

ES6为参数提供了默认值。在定义函数时便初始化了这个参数,以便在参数没有被传递进去时使用。

function action(num = 200){
    console.log(num);
}


箭头函数

ES6很有意思的一部分就是函数的快捷写法。也就是箭头函数。

箭头函数最直观的三个特点。

  • 不需要 function 关键字来创建函数

  • 省略 return 关键字

  • 继承当前上下文的 this 关键字

[1, 2, 3].map(x => x+1);

等同于

[1, 2, 3].map(function(x){
    return x+1;
}).bind(this))

说个小细节。

当你的函数有且仅有一个参数的时候,是可以省略掉括号的。当你函数返回有且仅有一个表达式的时候可以省略{} 和 return;例如:

var people = name => 'hello' + name;

作为参考

var people = (name, age) => {
    const fullname = 'hello' + name;
    return fullName;
}

请使用ES6重构以下代码

var calculate = function(x, y, z) {
    if(typeof x != 'number') {x = 0}
    if(typeof y != 'number') {y = 6}
    
    var dwt = x % y;
    var result;
    
    if(dwt == z) { result = true };
    if(dwt !=z) { result = false };
    
    return result;
}

重构后

const calculate = (x, y, z) => {
    x = typeof x !== 'number' ? 0 : x;
    y = typeof y !== 'number' ? 6 : y;
    return x % y === z;
}


4.拓展的对象功能

对象初始化简写

ES5我们对于对象都是以键值对的形式书写,是有可能出现键值对重名的。例如:

function people(name, age) {
    return {
        name: name,
        age: age
    }
}

键值对重名,ES6可以简写如下:

function people(name, age) {
    return {
        name, 
        age
    }
}

ES6 同样改进了为对象字面量方法赋值的语法。ES5为对象添加方法

const people = {
    name: 'lux',
    getName: function(){
        console.log(this.name);
    }
}

ES6通过省略冒号与 function 关键字,将这个语法变得更简洁

const people = {
    name: 'lux',
    getName() {
        console.log(this.name);
    }
}


ES6 对象提供了 Object.assign()这个方法来实现浅复制。

Object.assign() 可以把任意多个源对象自身可枚举的属性拷贝给目标对象,然后返回目标对象。第一参数即为目标对象。在实际项目中,我们为了不改变源对象。一般会把目标对象传为{}

const objA = {name: 'cc', age: 18};
const objB = {address: 'beijing'};
const objC = {};
const obj = Object.assign(objC, objA, objB);

//objA objB objC obj 分别输出
console.log(objA); // {name: 'cc', age: 18}
console.log(objB); // {address: 'beijing'}
console.log(objC); // {name: 'cc', age: 18, address: 'beijing'}
console.log(obj);  // {name: 'cc', age: 18, address: 'beijing'}


5.更方便的数据访问--解构

数组和对象是JS中最常用也是最重要表示形式。为了简化提取信息,ES6新增了解构,这是将一个数据结构分解为更小的部分的过程

ES5我们提取对象中的信息形式如下:

const people = {
    name: 'lux',
    age: 20
}
const name = people.name;
const age = people.age;

是不是觉得很熟悉,没错,在ES6之前我们就是这样获取对象信息的,一个一个获取。现在,解构能让我们从对象或者数组里取出数据存为变量,例如

// 对象
const people = {
    name: 'lux',
    age: 20
}
const { name, age} = people;

//数组
const color = ['red', 'blue'];
const [first, second] = color;
console.log(first); // red
console.log(second); // blue

6.Spread Operator 展开运算符

ES6中另外一个好玩的特性就是Spread Operator 也是三个点儿...接下来就展示一下它的用途。

组装对象或者数组

// 数组
const color = ['red', 'yellow'];
const colorfull = [...color, 'green', 'pink'];
console.log(colorful); // ['red', 'yellow', 'green', 'pink']

// 对象
const alp = {first: 'a', second: 'b'};
const alphabets = { ...alp, third: 'c'};
console.log(alphabets); //{ first: "a", second: "b", third: "c" }

有时候我们想获取数组或者对象除了前几项或者除了某几项的其他项

// 数组
const number = [1, 2, 3, 4, 5];
const [fitst, ...rest] = number;
console.log(rest); // [2, 3, 4, 5]

// 对象
const user = {
    username: 'lux',
    gender: 'female',
    age: 19,
    address: 'peking'
}
const { username, ...rest } = user;
console.log(rest); // { gender: 'female', age: 19, address: 'peking' }

对于 Object 而言,还可以用于组合成新的 Object 。(ES2017 stage-2 proposal) 当然如果有重复的属性名,右边覆盖左边

const first = {
    a: 1,
    b: 2,
    c: 6
}
const second = {
    c: 3,
    d: 4
}
const total = { ...first, ...second }
console.log(total); // { a:1, b:2, c:3, d:4 }

7.import 和 export

import导入模块、export导出模块

// 全部导入
import people from './people'

// 有一种特殊情况,即允许你将整个模块当做单一对象进行导入
// 该模块的所有导出都会作为对象的属性存在
import * as people from "./people"
console.log(people.name);
console.log(people.age);
console.log(people.getName());

// 导入部分
import { name, age } from './people'

// 导出默认, 有且只有一个默认
export default App

// 部分导出
export class App extend Component {};

导入的时候有没有大括号的区别:

1、当用 export default people 导出时, 就用 import people 导入 (不带大括号)

2、一个文件里,有且只能有一个 export default ,但可以有多个 export

3、当用 export name 时, 就用 import { name } 导入 (带大括号)

4、当一个文件里,既有一个 export default people , 又有多个 export name 或者 export age 时,导入就用 import people, { name, age }

5、当一个文件里出现n多个 export 导出很多模块,导入时处理一个一个导入, 也可以用 import * as people


8. Promise

在promise之前代码过多的回调或者嵌套,可读性差、耦合度高、扩展性低。通过Promise机制,扁平化的代码机构,大大提高了代码可读性;用同步编程的方式来编写异步代码,保存线性的代码逻辑,极大的降低了代码耦合性而提高了程序的可扩展性。

说白了就是用同步的方式去写异步代码。

发起异步请求

fetch('/api/todos')
  .then(res => res.json())
  .then(data => ({ data }))
  .catch(err => ({ err }))

今天看到一篇关于面试题的很有意思。

setTimeout(function(){
    console.log(1);
}, 0);

// 异步执行 resolve 对应 then, reject 对应catch
new Promise(function executor(resolve, reject){
    console.log(2);
    for(var i=0; i<1000; i++){ //只会执行 resolve 和 reject 中的一个,执行一次
        i==9999 ? resolve():reject();
    }
    console.log(3);
}).then(() => console.log(4))
.catch(() => console.log(6))
console.log(5);

// 所以输出顺序是 2 3 5 6 1


9.Generators

// 使用迭代器, 实现 for...of
const arr = [1, 2, 3, 4, 5];
const it = arr.values();  //生成一个迭代器
console.log(it); // Array Iterator
// let curr = it.next(); // 每一次next, 就遍历下一个
while(!curr.done){ // 当遍历结束后,done 变为true
    console.log(curr.value); // 获取迭代的值
    curr = it.next();
}

// 输出 1 2 3 4 5

生成器( generator)是能返回一个迭代器的函数。生成器函数也是一种函数,最直观的表现就是比普通的function多了个星号*,在其函数体内可以使用yield关键字,有意思的是函数会在每个yield后暂停。

这里生活中有一个比较形象的例子。咱们到银行办理业务时候都得向大厅的机器取一张排队号。你拿到你的排队号,机器并不会自动为你再出下一张票。也就是说取票机“暂停”住了,直到下一个人再次唤起才会继续吐票。

OK。说说迭代器。当你调用一个generator时,它将返回一个迭代器对象。这个迭代器对象拥有一个叫做next的方法来帮助你重启generator函数并得到下一个值。next方法不仅返回值,它返回的对象具有两个属性:done和value。value是你获得的值,done用来表明你的generator是否已经停止提供值。继续用刚刚取票的例子,每张排队号就是这里的value,打印票的纸是否用完就这是这里的done。

// 生成器
function *createIterator(){
    yield 1;
    yield 2;
    yield 3;
}

// 生成器能像正规函数那样被调用,但会返回一个迭代器
let iterator = createIterator();

console.log(iterator.next().value); // 1
console.log(iterator.next().value); // 2
console.log(iterator.next().value); // 3


function *interrogate(){
    const name = yield "what is your name";
    const color = yield "what is your favoriate color";
    return `${name}'s favoriate color is ${color}`;
}
const it = interrogate();
console.log(it);
console.log(it.next().value); // what is your name
console.log(it.next('Ethan').value); // what is your favoriate color 传值给上一个迭代器 name=Ethan
console.log(it.next('green').value); // Ethan's favoriate color is green 传值给上一个迭代器 color=green

那生成器和迭代器又有什么用处呢?

围绕着生成器的许多兴奋点都与异步编程直接相关。异步调用对于我们来说是很困难的事,我们的函数并不会等待异步调用完再执行,你可能会想到用回调函数,(当然还有其他方案比如Promise比如Async/await)。

生成器可以让我们的代码进行等待。就不用嵌套的回调函数。使用generator可以确保当异步调用在我们的generator函数运行一下行代码之前完成时暂停函数的执行。

那么问题来了,咱们也不能手动一直调用next()方法,你需要一个能够调用生成器并启动迭代器的方法。就像这样子的

function run(taskDef){ // taskDef 即一个生成器函数
    // 创建迭代器,让它在别处可用
    let task = taskDef();
    // 启动任务
    let result = task.next();
    // 递归使用函数来保持对next() 的调用
    function step(){
        if(!result.done){
            result = task.next();
            step();
        }
    }
    step();
}

生成器与迭代器最有趣、最令人激动的方面,或许就是可创建外观清晰的异步操作代码。你不必到处使用回调函数,而是可以建立貌似同步的代码,但实际上却使用 yield 来等待异步操作结束。



原文链接地址:http://www.jianshu.com/p/287e0bb867ae