ES6常用语法简介import export
let与var用法区别
//var var a = [];for (var i = 0; i < 10; i++) { a[i] = function () { console.log(i); };}a[6](); // 10------------------------------//letvar a = [];for (let i = 0; i < 10; i++) { a[i] = function () { console.log(i); };}a[6](); // 6------------------------------//如果不用 let 实现类似功能function iteratorFactory(i){ var onclick = function(e){ console.log(i) } return onclick;}var clickBoxs = document.querySelectorAll('.clickBox')for (var i = 0; i < clickBoxs.length; i++){ clickBoxs[i].onclick = iteratorFactory(i)}
class, extends, super
class Animal { constructor(){ this.type = 'animal' } says(say){ console.log(this.type + ' says ' + say) }}let animal = new Animal()animal.says('hello') //animal says hello//继承class Cat extends Animal { constructor(){ super() this.type = 'cat' }}let cat = new Cat()cat.says('hello') //cat says hello
上面代码首先用class
定义了一个“类”,可以看到里面有一个constructor
方法,这就是构造方法,而this
关键字则代表实例对象。简单地说,constructor
内定义的方法和属性是实例对象自己的,而constructor
外定义的方法和属性则是所有实例对象可以共享的。
Class
之间可以通过extends
关键字实现继承,这比ES5
的通过修改原型链实现继承,要清晰和方便很多。
super
关键字,它指代父类的实例(即父类的this
对象)。子类必须在constructor
方法中调用super
方法,否则新建实例时会报错。
this
对象,而是继承父类的this
对象,然后对其进行加工。如果不调用super
方法,子类就得不到this
对象。 ES6
的继承机制,实质是先创造父类的实例对象this
(所以必须先调用super
方法),然后再用子类的构造函数修改this
。
箭头函数 arrow function
function(i){ return i + 1; } //ES5(i) => i + 1 //ES6
如果方程比较复杂,则需要用{}
把代码包起来:
//es5function(x, y) { x++; y--; return x + y;}//es6(x, y) => {x++; y--; return x+y}
除了看上去更简洁以外,arrow function
还有一项超级无敌的功能!
this
对象一直是一个令人头痛的问题,在对象方法中使用this
,必须非常小心。例如: //错误代码class Animal { constructor(){ this.type = 'animal' } says(say){ setTimeout(function(){ console.log(this.type + ' says ' + say) }, 1000) }} var animal = new Animal() animal.says('hi') //undefined says hi
运行上面的代码会报错,这是因为setTimeout
中的this
指向的是全局对象。所以为了让它能够正确的运行,传统的解决方法有两种:
1.第一种是将this
传给self
,再用self
来指代this
says(say){ var self = this; setTimeout(function(){ console.log(self.type + ' says ' + say)}, 1000)
2.第二种方法是用bind(this)
,即
says(say){ setTimeout(function(){ console.log(this.type + ' says ' + say) }.bind(this), 1000)}
但现在我们有了箭头函数,就不需要这么麻烦了:
class Animal { constructor(){ this.type = 'animal' } says(say){ setTimeout( () => { console.log(this.type + ' says ' + say) }, 1000) }} var animal = new Animal() animal.says('hi') //animal says hi
当我们使用箭头函数时,函数体内的this
对象,就是定义时
所在的对象,而不是使用时所在的对象。
没有自己的this
,它的this是继承外面的
,因此内部的this就是外层代码块的this。 模板字符串 template string
//不用模板字符串 写法$("#result").append( "There are " + basket.count + " " + "items in your basket, " + "" + basket.onSale + " are on sale!");//使用模板字符串写法$("#result").append(` There are ${basket.count} items in your basket, ${basket.onSale} are on sale!`);
用反引号(`)来标识起始,用
${}
来引用变量,而且所有的空格和缩进
都会被保留在输出之中(==这个需要注意==)
解构 destructuring
ES6
允许按照一定模式,从数组和对象中提取值,对变量进行赋值,这被称为解构(Destructuring
)。
let cat = 'ken'let dog = 'lili'let zoo = {cat: cat, dog: dog}console.log(zoo) //Object {cat: "ken", dog: "lili"}//使用es6解构let cat = 'ken'let dog = 'lili'let zoo = {cat, dog}console.log(zoo) //Object {cat: "ken", dog: "lili"}//反过来可以这么写:let dog = {type: 'animal', many: 2}let { type, many} = dogconsole.log(type, many) //animal 2
默认值 default
function animal(type){ type = type || 'cat' console.log(type)}animal()//ES6function animal(type = 'cat'){ console.log(type)}animal()
展开操作符 rest arguments (...)
function animals(...types){ console.log(types)}animals('cat', 'dog', 'fish') //["cat", "dog", "fish"]
import export
传统的写法CommonJS(服务器端)和AMD(浏览器端,如require.js)
AMD写法
//content.jsdefine('content.js', function(){ return 'A cat';})//index.jsrequire(['./content.js'], function(animal){ console.log(animal); //A cat})
CommonJS
//index.jsvar animal = require('./content.js')//content.jsmodule.exports = 'A cat'
ES6的写法
//index.jsimport animal from './content'//content.jsexport default 'A cat'
ES6 module的其他高级用法
//content.jsexport default 'A cat' export function say(){ return 'Hello!'} export const type = 'dog' //index.jsimport { say, type } from './content' let says = say()console.log(`The ${type} says ${says}`) //The dog says Hello
这里输入的时候要注意:
如果还希望输入content.js中输出的大括号
里面的变量名
,必须与被导入模块(content.js)对外接口的名称相同
。默认值
(default), 可以写在大括号外面。
//index.jsimport animal, { say, type } from './content' let says = say()console.log(`The ${type} says ${says} to ${animal}`) //The dog says Hello to A cat
修改变量名
此时我们不喜欢type这个变量名,因为它有可能重名,所以我们需要修改一下它的变量名。在es6中可以用as
实现一键换名。
//index.jsimport animal, { say, type as animalType } from './content' let says = say()console.log(`The ${animalType} says ${says} to ${animal}`) //The dog says Hello to A cat
模块的整体加载
除了指定加载某个输出值,还可以使用整体加载,即用星号(*
)指定一个对象,所有输出值都加载在这个对象上面。
//index.jsimport animal, * as content from './content' let says = content.say()console.log(`The ${content.type} says ${says} to ${animal}`) //The dog says Hello to A cat
通常星号
*
结合as
一起使用比较合适。