ES6

学习部分重要ES6知识点的总结。。长,可以先码后看

环境搭建

  • 在项目目录下新建目录src(承载es6语法)和dist(承载编译之后的es5语法)
  • 在项目中npm init -y初始化
  • 安装babel-cli babel-preset-es2015
  • 在根目录下新建.babelrc文件
{
  "presets":[
    "es2015"
  ],
  "plugins":[]
}
  • 在src中写es6语法后,运行babel src/index.js -o dist/index.js

新的声明方式

  • var 全局变量
  • let 局部变量
  • const 静态变量

变量的解构赋值

  • 数组的解构赋值
let [a,b,c] = [1,2,3];
let [a,[b,c],d] = [1,[2,3],4];

let [foo=true] = [];
console.log(foo) // true

let [a,b="World"] = ["Hello"]
console.log(a+b) // HelloWorld

let [a,b="World"] = ['Hello',undefined]
console.log(a+b) //HelloWorld

let [a,b="World"] = ['Hello',null]
console.log(a+b) // Hellonull
  • 对象的解构赋值
// 不用按次序,但是要名相同
let {foo,bar} = {bar:'World',foo:'Hello'}
console.log(foo+bar) //HelloWorld

// 如果解构前,已经定义了变量,需要在解构的语句外面加一个圆括号
let foo;
({foo}={foo:'Hello1'})
console.log(foo)
  • 字符串的解构
const [a,b,c,d,e]="Hello"
console.log(a)
console.log(b)
console.log(c)
console.log(d)
console.log(e)

扩展运算符和rest运算符

  • 对象扩展运算符(…)
// 运用1:对于一个方法传入参数不确定
function f(...arg) {
  console.log(arg[0])
  console.log(arg[1])
  console.log(arg[2])
}
f(1,2) // 1,2,undefined

// 运用2:解决数组赋值只是堆栈引用问题
let arr1=['www','baidu','com'];
//let arr2=arr1;
let arr2=[...arr1];
console.log(arr2);
arr2.push('hh');
console.log(arr2); //["www", "baidu", "com", "hh"]
console.log(arr1); //["www", "baidu", "com"]
  • rest运算符
function f(first,...arg){
    console.log(arg.length);
}
f(0,1,2,3,4,5,6,7); //7

function f1(first,...arg){
    for(let val of arg){  // for of循环可以避免开拓内存空间,增加运行效率
        console.log(val);
    }
}

f1(0,1,2,3,4,5,6,7); // 1,2,3,4,5,6,7

字符串模版

let s1='hhh';
let blog = `非常高兴你能看到这篇文章,${s1}`;
document.write(blog);

//支持html标签
let s2='hhh';
let blog = `<b>非常高兴你能看到这篇文章</b>,${s1}`;
document.write(blog);

//对运算的支持
let a=1;
let b=2;
let result=`${a+b}`;
document.write(result); //3

字符串查找

let s1='hh';
let blog = '非常高兴你能看到这篇文章,我是你的老hh';
document.write(blog.includes(s1));

//判断开头有没有 startsWith
//判断结尾有没有 endsWith

复制字符串

document.write('hhh|'.repeat(3));

ES6数字操作

  • 二进制和八进制
let er = 0b010101 //21
let ba = 0o666 //438
  • 数字判断和转换
// 数字验证
let a = 11/4;
console.log(Number.isFinite(a)) // true

//NaN验证 NaN是特殊的非数字
console.log(Number.isNaN(NaN)) //true

// 判断是否为整数
console.log(Number.isInteger(123.4)) //false

// 整数转换、浮点数转换
let a = '9.18'
console.log(Number.parseInt(a))  //9
console.log(Number.parseFloat(a)) //9.18

// 整数取值范围操作
let a = Math.pow(2,53)-1 // 最大的整数2^53-1

// 最大安全整数
console.log(Number.MAX_SAFE_INTEGER)

// 最小安全整数
console.log(Number.MIN_SAFE_INTEGER)

// 安全整数判断
let a = Math.pow(2,53)-1;
console.log(Number.isSafeInterger(a)) //false

ES6新增的数组知识

  • JSON数组格式转换
let  json = {
    '0': 0,
    '1': 'hh',
    '2': 'hello',
    length:3
}

//将json转化为数组
let arr = Array.from(json)

//将一堆文本或者变量转换成数组
let arr = Array.of(3,'4',5,6)
  • find()方法

在find方法中我们需要传入一个匿名函数,函数需要传入三个参数:
* value:表示当前查找的值。
* index:表示当前查找的数组索引。
* arr:表示当前数组。

在函数中如果找到符合条件的数组元素就进行return,并停止查找。如果找不到会显示undefined

let arr=[1,2,3,4,5,6,7,8,9];
console.log(arr.find(function(value,index,arr){
    return value > 5;
}))
// 6
  • fill()方法

fill()也是一个实例方法,它的作用是把数组进行填充,它接收三个参数,第一个参数是填充的变量,第二个是开始填充的位置,第三个是填充结束的位置。

let arr=[0,1,2,3,4,5,6,7,8,9];
arr.fill('hh',2,5);
console.log(arr); //[0, 1, "hh", "hh", "hh", 5, 6, 7, 8, 9]
  • 数组的遍历
// for...of循环
for(let item of arr){
  console.log(item)
}

//for...of输出数组索引
for(let index of arr.keys()){
  console.log(index)  //0,1,2
}

//同时输出内容和索引
for(let [index,item] of arr.entries()){
  console.log(index + item)
}
  • entries()实例方法

entries()实例方式生成的是Iterator形式的数组,那这种形式的好处就是可以让我们在需要时用next()手动跳转到下一个值。

let arr=['hello','world','hh']
let list=arr.entries();
console.log(list.next().value);
console.log(list.next().value);
console.log(list.next().value);

ES6中箭头函数和扩展

  • 默认值 函数多个参数,可以给其中的参数设置默认值,传参时不必要传
  • 主动抛出错误 使用 throw new Error(”)
  • 严谨模式 在函数体头部 ‘use strict’
  • 获得需要传递的参数个数 function.length
  • 箭头函数 () => {} ,与普通函数区别是this指向外部环境

ES6中的函数和数组补漏

  • 对象的函数解构
let json = {
    a:'hello',
    b:'world'
}
function fun({a,b}){
    console.log(a,b);
}
fun(json);
  • 数组的函数解构
let arr = ['hello','world','hh'];
function fun(a,b,c){
    console.log(a,b,c);
}
fun(...arr);
  • in的用法
//对象判断
let obj={
    a:'hello',
    b:'world'
}
console.log('a' in obj);  //true

//数组判断
//这里的0指的是数组下标位置是否为空
let arr=[,,,,,];
console.log(0 in arr); //false

let arr1=['jspang','技术胖'];
console.log(0 in arr1);  // true
  • 数组的遍历方法
// forEach
let arr=['hello','world','hh'];
arr.forEach((val,index)=>console.log(index,val));

//filter
arr.filter(x=>console.log(x));

// some
arr.some(x=>console.log(x));

// map
arr.map(x=>console.log(x));
// console.log(arr.map(x=>'web'));
  • 数组转换字符串
//join() 在数组中加入间隔
let arr=['hello','world','hh'];
console.log(arr.join('|'));

// toString() 转换时用,隔开
console.log(arr.toString())

ES6中对象

  • 对象赋值

ES6允许把声明的变量直接赋值给对象

let name="hello";
let skill= 'web';
var obj= {name,skill};
console.log(obj);  //Object {name: "hello", skill: "web"}
  • 对象Key值构建

对于不知道key名的,可以用[ ] 的形式,进行对象的构建。

let key='skill';
var obj={
    [key]:'web'
}
console.log(obj.skill);
  • 自定义对象方法
var obj={
    add:function(a,b){
        return a+b;
    }
}
console.log(obj.add(1,2));  //3
  • Object.is()对象比较
var obj1 = {name:'hello'};
var obj2 = {name:'hello'};
console.log(obj1.name === obj2.name);//true
console.log(Object.is(obj1.name,obj2.name)); //true

// 区分=== 和 is方法,===为同值相等,is()为严格相等
console.log(+0 === -0);  //true
console.log(NaN === NaN ); //false
console.log(Object.is(+0,-0)); //false
console.log(Object.is(NaN,NaN)); //true
  • Object.assign()合并对象
var a={a:'hello'};
var b={b:'world'};
var c={c:'web'};

let d=Object.assign(a,b,c)
console.log(d); //{a: "hello", b: "world", c: "web"}

Symbol在对象中的作用

Symbol的意思是全局标记

  • 声明Symbol
var a = new String;
var b = new Number;
var c = new Boolean;
var d = new Array;
var e = new Object;
var f= Symbol();
console.log(typeof(f)); //symbol
  • Symbol的打印
var g = Symbol('hello');
console.log(g); //Symbol(hello)
console.log(g.toString());
  • Symbol在对象中的应用

用Symbol构建对象的Key,并调用和赋值。

var hello = Symbol();
var obj={
    [hello]:'Hello'
}
console.log(obj[hello]);
obj[hello]='web';
console.log(obj[hello]);
  • Symbol对象元素的保护作用

在对象中有很多值,但是循环输出时,并不希望全部输出,那我们就可以使用Symbol进行保护

//没有保护的写法
var obj={name:'hh',skill:'web',age:18};
for (let item in obj){
    console.log(obj[item]);
}

// 用Symbol进行保护
let obj={name:'hh',skill:'web'};
let age=Symbol();
obj[age]=18;
for (let item in obj){
    console.log(obj[item]); //hh web
}
console.log(obj[age]); // 18

Set和WeakSet数据解构

Set的数据结构是以数组的形式构建的。Set.size可以得到Set值的数量

  • Set声明
let setArr = new Set(['hello','world','hh','hello']);
// Set会去重
console.log(setArr);//Set {"hello", "world", "hh"}
  • Set值的增删查
//add追加。
let setArr = new Set(['hello','world','web']);
setArr.add('前端职场');
console.log(setArr);

//delete删除
setArr.delete('前端职场');
console.log(setArr);

//has查找
console.log(setArr.has('hello'));//true

//clear清空
setArr.clear();
console.log(setArr);//Set(0) {}
  • WeakSet的声明

WeakSet里边的值也是不允许重复的

// 直接在new 的时候就放入值,将报错
let weakObj=new WeakSet();
let obj={a:'hello',b:'world'}
weakObj.add(obj);
console.log(weakObj);

map数据结构

在一些构建工具中是非常喜欢使用map这种数据结构来进行配置的,因为map是一种灵活,简单的适合一对一查找的数据结构。

  • Json和map格式的对比

map的效率和灵活性更好,你可以把它看成一种特殊的键值对,但你的key可以设置成数组,值也可以设置成字符串,让它不规律对应起来。

let json = {
    name:'hello',
    skill:'web'
}
console.log(json.name);

var map=new Map();
map.set(json,'hello');
// map.set('hello',json)
console.log(map);
  • map的增删查
// 增加set
map.set(json,'hello');

// 取值get
console.log(map.get(json));

// 删除delete
map.delete(json);
console.log(map)

// size取长度
console.log(map.size);

// has查找
console.log(map.has('hello'))

// clear清空
map.clear()

用Proxy进行预处理

在运行函数前初始化一些数据,在改变对象值后做一些善后处理。这些都算钩子函数,Proxy的存在就可以让我们给函数加上这样的钩子函数,你也可以理解为在执行方法前预处理一些代码。你可以简单的理解为他是函数或者对象的生命周期。
Proxy的应用可以使函数更加强大,业务逻辑更加清楚,而且在编写自己的框架或者通用组件时非常好用。

  • 声明Proxy

第一个花括号就相当于我们方法的主体,后边的花括号就是Proxy代理处理区域,相当于我们写钩子函数的地方。

// 基本形式
// new Proxy({},{});

var pro = new Proxy({
    add: function (val) {
        return val + 10;
    },
    name: 'I am hh'
}, {
        get:function(target,key,property){
            console.log('come in Get');
            return target[key];
        }
    });

console.log(pro.name); // come in Get, I am hh
  • get属性

get属性是在你得到某对象属性值时预处理的方法,他接受三个参数

target:得到的目标值
key:目标的key值,相当于对象的属性
property:这个不太常用,用法还在研究中,还请大神指教。

  • set属性

set属性是值你要改变Proxy属性值时,进行的预先处理。它接收四个参数。

target:目标值。
key:目标的Key值。
value:要改变的值。
receiver:改变前的原始值。

var pro = new Proxy({
    add: function (val) {
        return val + 10;
    },
    name: 'I am hh'
}, {
        get:function(target,key){
            console.log('come in Get');
            return target[key];
        },
        set:function(target,key,value,receiver){
            console.log(`setting ${key} = ${value}`);
            return target[key] = value;
        }

    });

console.log(pro.name); // come in Get,I am hh
pro.name='zust'; // setting name = zust
console.log(pro.name); // come in Get,zust
  • apply的使用

apply的作用是调用内部的方法,它使用在方法体是一个匿名函数时。

let target = function () {
    return 'I am hh';
};
var handler = {
    apply(target, ctx, args) {
        console.log('do apply');
        return Reflect.apply(...arguments);
    }
}

var pro = new Proxy(target, handler);

console.log(pro()); // do apply,I am hh

promise对象的使用

ES6中的promise的出现给我们很好的解决了回调地狱的问题,在使用ES5的时候,在多层嵌套回调时,写完的代码层次过多,很难进行维护和二次开发,ES6认识到了这点问题,现在promise的使用,完美解决了这个问题。那我们如何理解promise这个单词在ES5中的作用那,你可以想象他是一种承诺,当它成功时执行一些代码,当它失败时执行一些代码。它更符合人类的行为思考习惯,而不在是晦涩难懂的冰冷语言。

  • promise的基本用法

promise执行多步操作非常好用,那我们就来模仿一个多步操作的过程,那就以吃饭为例吧。要想在家吃顿饭,是要经过三个步骤的。

1.洗菜做饭。
2.坐下来吃饭。
3.收拾桌子洗碗。
这个过程是有一定的顺序的,你必须保证上一步完成,才能顺利进行下一步。我们可以在脑海里先想想这样一个简单的过程在ES5写起来就要有多层的嵌套。那我们现在用promise来实现。

let state=1;

function step1(resolve,reject){
    console.log('1.开始-洗菜做饭');
    if(state==1){
        resolve('洗菜做饭--完成');
    }else{
        reject('洗菜做饭--出错');
    }
}


function step2(resolve,reject){
    console.log('2.开始-坐下来吃饭');
    if(state==1){
        resolve('坐下来吃饭--完成');
    }else{
        reject('坐下来吃饭--出错');
    }
}


function step3(resolve,reject){
    console.log('3.开始-收拾桌子洗完');
     if(state==1){
        resolve('收拾桌子洗完--完成');
    }else{
        reject('收拾桌子洗完--出错');
    }
}

new Promise(step1).then(function(val){
    console.log(val);
    return new Promise(step2);

}).then(function(val){
     console.log(val);
    return new Promise(step3);
}).then(function(val){
    console.log(val);
    return val;
});

class类的使用

我们在ES5中经常使用方法或者对象去模拟类的使用,虽然可以实现功能,但是代码并不优雅,ES6为我们提供了类的使用。需要注意的是我们在写类的时候和ES5中的对象和构造函数要区分开来,不要学混了。

  • 类的声明
class coder{
    name(val){
        console.log(val);
    }
}
  • 类的使用

实例化类,并使用类中的方法

class Coder{
    name(val){
        console.log(val);
    }
}

let hh= new Coder;
hh.name('hh');
  • 类的多方法声明

注意的是两个方法中间不要写逗号了,还有这里的this指类本身,还有要注意return 的用法

class Coder{
    name(val){
        console.log(val);
        return val;
    }
    skill(val){
        console.log(this.name('hh')+':'+'Skill:'+val);
    }
}

let hh= new Coder;
hh.name('hh');
hh.skill('web');
  • 类的传参

在类的参数传递中我们用constructor( )进行传参。传递参数后可以直接使用this.xxx进行调用。

class Coder{
    name(val){
        console.log(val);
        return val;
    }
    skill(val){
        console.log(this.name('hh')+':'+'Skill:'+val);
    }

    constructor(a,b){
        this.a=a;
        this.b=b;
    }

    add(){
        return this.a+this.b;
    }
}

let hh=new Coder(1,2);
console.log(hh.add());

我们用constructor来约定了传递参数,然后用作了一个add方法,把参数相加。这和以前我们的传递方法有些不一样,所以需要小伙伴们多注意下。

  • class的继承
class htmler extends Coder{

}

let hh=new htmler;
hh.name('hh');

声明一个htmler的新类并继承Coder类,htmler新类里边为空,这时候我们实例化新类,并调用里边的name方法。结果也是可以调用到的。

模块化操作

在ES5中我们要进行模块华操作需要引入第三方类库,随着前后端分离,前端的业务日渐复杂,ES6为我们增加了模块话操作。模块化操作主要包括两个方面。

export :负责进行模块化,也是模块的输出。
import : 负责把模块引,也是模块的引入操作。

  • export的用法

export可以让我们把变量,函数,对象进行模块话,提供外部调用接口,让外部进行引用。先来看个最简单的例子,把一个变量模块化。我们新建一个temp.js文件,然后在文件中输出一个模块变量。

// temp.js
export var a = 'hh';

// index.js
import {a} from './temp.js';
console.log(a);
  • 多变量的输出

这里声明了3个变量,需要把这3个变量都进行模块化输出,这时候我们给他们包装成对象就可以了

var a ='hello';
var b ='hh';
var c = 'web';

export {a,b,c}
  • 函数的模块化输出
export function add(a,b){
    return a+b;
}
  • as的用法

有些时候我们并不想暴露模块里边的变量名称,而给模块起一个更语义话的名称,这时候我们就可以使用as来操作

var a ='hello';
var b ='hh';
var c = 'web';

export {
    x as a,
    y as b,
    z as c
}
  • export default的使用

加上default相当是一个默认的入口。在一个文件里export default只能有一个。我们来对比一下export和export default的区别

1.export

export var a ='hh';

export function add(a,b){
    return a+b;
}

对应的导入方式

import {a,add} form './temp';//也可以分开写

2.export default

export default var a='jspang';

对应的引入方式

import str from './temp';

ES6的模块化不能直接在浏览器中预览,必须要使用Babel进行编译之后正常看到结果。