JavaScript-ES6语法

ES 的全称是 ECMAScript,它是由 ECMA 国际标准化组织制定的一项脚本语言的标准化规范。现在我们说的ES6,实际上是一个泛指,泛指 ES2015 及后续的版本。

JavaScript语言本身也有如下一些令人不满意的地方,ES6出现就是为了解决这些问题:

  • 变量提升特性增加了程序运行时的不可预测性
  • 语法过于松散,实现相同的功能,不同的人可能会写出不同的代码

1 - var、let、const

1.1 let:为了替代var关键字

ES6中新增了用于声明变量的关键字let,它的出现主要是为了替代var关键字的。

  1. 在大括号中使用let关键字声明的变量具有块级作用域,变量只在所处的块级有效
if (true) { 
   var a = 10; // 使用 var
}
console.log(a) // 10
// 如果是使用 var,即使是在{}外面也能正常访问 a,这显然是不合理的

if (true) { 
   let a = 10; // 使用 let
}
console.log(a) // 报错:a is not defined

利用let声明的变量会绑定在这个块级作用域,不会受外界的影响

var tmp = 123;
if (true) {
    let tmp; 
    tmp = 'abc';
    console.log(tmp); // 'abc'
} // 外面的tmp和{}里面的tmp没有任何关系
console.log(tmp); // 123
  1. 使用let关键字声明的变量没有变量提升,这样就可以防止循环变量变成全局变量
for (var i = 0; i < 2; i++) {} // 使用 var
console.log(i); // 2  i变成了全局变量

for (let i = 0; i < 2; i++) {} // 使用 let
console.log(i); // 报错:i is not defined
  1. 变量必须先声明再使用
console.log(a);
var a = 20; // 使用 var
// 打印:undefined

console.log(a); // 报错:Cannot access 'a' before initialization
let a = 20; // 使用 let

1.2 经典面试题

下面代码打印什么?为什么这么打印?

//使用 var
var arr = [];
for (var i = 0; i < 2; i++) {
   arr[i] = function () { // 数组中存放函数
       console.log(i); 
   }
} 
arr[0](); // 2
arr[1](); // 2

当执行数组里面的函数时,就是执行console.log(i);就是打印i,由于内层作用域没有i,根据变量查找原则,就往外层作用域查找i,由于使用var关键字定义不产生块级作用域,所以外层作用域就是全局作用域,全局作用域中有i=2,所以上面打印2。

此题的关键点在于变量i是全局的,函数执行时输出的都是全局作用域下的i值。

//使用 let
let arr = [];
for (let i = 0; i < 2; i++) {
   arr[i] = function () { // 数组中存放函数
       console.log(i); 
   }
}
arr[0](); // 0
arr[1](); // 1

当执行数组里面的函数时,就是console.log(i);就是打印i,由于内层作用域没有i,根据变量查找原则,就往外层作用域查找i,由于使用let关键字定义会产生块级作用域,所以外层作用域就是for循环作用域,每次循环都会产生一个块级作用域,每个块级作用域中的变量都是不同的(分别为i=0,i=1),所以打印 0 1 。

此题的关键点在于每次循环都会产生一个块级作用域,每个块级作用域中的变量都是不同的,函数执行时输出的是循环产生的块级作用域下的i值。

1.3 const:常量

声明常量,常量就是值或者内存地址不能变化的量。

  1. 具有块级作用域
if (true) { 
    const a = 10;
}
console.log(a) // a is not defined
  1. 使用const关键字声明也没有变量提升

  2. 声明常量时必须赋值,赋值后,值不能修改

const b; // 报错:Missing initializer in const declaration
//对于基本数据类型,值不能修改
const PI = 3.14;
PI = 100; // Assignment to constant variable. 

//对于复杂数据类型,指针指向的内容不能修改
const ary = [100, 200];
ary[0] = 'a';
ary[1] = 'b';
console.log(ary); // ['a', 'b']; 
ary = ['a', 'b']; // 报错:Assignment to constant variable.

1.4 var、let、const的区别

  • 使用 var 声明的变量,其作用域为该语句所在的函数内,且存在变量提升现象
  • 使用 let 声明的变量,其作用域为该语句所在的代码块内,不存在变量提升
  • 使用 const 声明的是常量,在后面出现的代码中不能再修改该常量的值

总结:其实let、const就相当于swift中的var、let

2 - 解构赋值

ES6中允许从数组或者对象中提取值,按照对应位置,对变量进行赋值。

  1. 数组解构允许我们按照一一对应的关系从数组中提取值,然后将值赋值给变量。
let [a, b, c, d, e] = [1, 2, 3]; //等号左边不代表数组,代表结构,里面是变量,等号右边是数组
console.log(a) //1
console.log(b) //2
console.log(c) //3
console.log(d) //undefined
console.log(e) //undefined
// 如果解构不成功,变量的值为undefined
  1. 对象解构允许我们使用变量的名字匹配对象的属性,如果匹配成功,将对象属性的值赋值给变量。
 let person = { name: 'zhangsan', age: 20 }; 
 let {name, age} = person;  //等号左边不代表对象,代表结构,里面是变量,等号右边是对象
 console.log(name); // 'zhangsan' 
 console.log(age);  // 20

 let {name: myName, age: myAge} = person; // myName myAge 属于别名
 console.log(myName); // 'zhangsan' 
 console.log(myAge);  // 20

3 - 箭头函数

ES6中新增的快速定义函数的方式,用来简化函数定义语法。

() => {}  // ()代表是函数,=>必须要的符号,指向哪一个代码块,{}函数体
const fn = () => {} // 代表把一个箭头函数赋值给fn
  1. 函数体中只有一句代码,且代码的执行结果就是返回值,可以省略大括号
//传统写法:命名函数
function sum(num1, num2) { 
     return num1 + num2; 
 }
//es6写法:箭头函数
const sum = (num1, num2) => num1 + num2; 
  1. 如果形参只有一个,可以省略小括号
//传统写法:命名函数
function fn (v) {
     return v;
 } 
//es6写法:箭头函数
const fn = v => v;
  1. 箭头函数的this

箭头函数不绑定this关键字(也就是箭头函数没有自己的this),箭头函数中的this,指向的是函数定义位置的上下文this

const obj = { name: '张三'} 
function fn () { 
   console.log(this);// 打印:obj
   return () => {
       console.log(this); // 打印:obj
   } 
 } 
const resFn = fn.call(obj); 
resFn();

面试题:下面代码打印什么?为什么这么打印?

var age = 100;
var obj = {
    age: 20,
    say: () => {
        alert(this.age)
    }
}
obj.say(); // 打印:100

箭头函数this指向的是函数定义位置的上下文this,而对象是没有作用域的,所以箭头函数虽然在对象中被定义,但是this指向的是全局作用域window,window中有age=100,所以打印100。

  1. 箭头函数的优点

箭头函数的优点在于解决了this执行环境所造成的一些问题。比如:解决了匿名函数this指向的问题,包括setTimeout和setInterval中使用this所造成的问题。

看完下面例子就明白了。

var obj = {
    birth: 1990,
    getAge: function() { // 这个是对象方法,所以里面的 this 指向对象
    console.log(this); // 打印:obj
    var b = this.birth;
    var fn = function() { // 这个是普通函数,所以里面的 this 指向window
      console.log(this); // 打印:window
      console.log(this.birth); // window没有birth,所以打印:undefined
      return new Date().getFullYear() - this.birth; 
    };
    return fn();
  }
};
console.log(obj.getAge()); // 打印:NaN

上面最后打印NaN,这显然不是我们想要的结果,代码修改如下:

var obj = {
    birth: 1990,
    getAge: function() { // 这个是对象方法,所以里面的 this 指向对象
    var that = this; // 将指向对象的 this 保存下来
    var b = this.birth;
    var fn = function() { // 这个是普通函数,所以里面的 this 指向window
      return new Date().getFullYear() - that.birth; 
    };
    return fn();
  }
};
console.log(obj.getAge()); // 打印:31

上面代码,使用var that = this;就可以正确打印了,但是使用var that = this;很麻烦,使用箭头函数就不用使用var that = this;了,代码如下:

var obj = {
    birth: 1990,
    getAge: function() { // 这个是对象方法,所以里面的 this 指向对象
    console.log(this); // 打印:obj
    var b = this.birth;
    // 箭头函数中的this,指向的是函数定义位置的上下文this,也就是obj对象
    var fn = () => new Date().getFullYear() - this.birth;
    return fn();
  }
};
console.log(obj.getAge()); // 打印:31

4 - 剩余参数

剩余参数语法允许我们将一个不定数量的参数表示为一个数组,这种方式很方便的去声明不知道参数情况下的一个函数。

如下,first是第一个参数,args是其他参数数组,…代表解构,就是把数组中的元素拿出来。

function sum (first, ...args) {
   console.log(first); // 10
   console.log(args);  // [20, 30] 
}
sum(10, 20, 30) 

剩余参数和数组解构配合使用。

let students = ['wangwu', 'zhangsan', 'lisi'];
let [s1, ...s2] = students; 
console.log(s1);  // 'wangwu' 
console.log(s2);  // ['zhangsan', 'lisi']

5 - ES6 的内置对象扩展方法

5.1 Array 的扩展方法

① 展开运算符:...

展开运算符可以将数组或者对象转为用逗号分隔的参数序列。

let ary = [1, 2, 3];
console.log(...ary); // 1 2 3
console.log([...ary]); // [1, 2, 3]
// 虽然console.log(...ary);结果打印是1 2 3,但是将…ary放到数组里面就会自动变成用逗号分隔的参数序列。
  1. 展开运算符可以应用于合并数组
// 方法一 
let ary1 = [1, 2, 3];
let ary2 = [3, 4, 5];
let ary3 = [...ary1, ...ary2];
console.log(...ary1); // 1 2 3
console.log(ary3); // [1, 2, 3, 3, 4, 5]
// 方法二 
ary1.push(...ary2);
console.log(ary1); // [1, 2, 3, 3, 4, 5]
  1. 展开运算符可以将伪数组转换为真正的数组

将伪数组转换成真正的数组之后就可以调用数组的方法了。

let oDivs = document.getElementsByTagName('div'); //返回值就是div的伪数组
oDivs = [...oDivs]; //转换成数组

② 构造函数方法:Array.from()

除了展开运算符,Array.from()也可以将伪数组或可遍历对象转换成真正的数组。

  1. 如果是伪数组
var oDivs = document.getElementsByTagName('div'); 
console.log(oDivs)  //伪数组
var ary = Array.from(oDivs);
console.log(ary);   //数组

打印结果如下:

  1. 如果是可遍历对象
let arrayLike = { //对象
    '0': 'a', // ① 属性名要写对应的索引
    '1': 'b',
    '2': 'c',
    'length': 3 // ② 要指定长度
}; 
//转成数组
let arr2 = Array.from(arrayLike);
console.log(arr2); // ['a', 'b', 'c']

方法还可以接受第二个参数,作用类似于数组的map方法,用来对每个元素进行处理,将处理后的值放入返回的数组

let arrayLike = { //对象
   '0': 1,
   '1': 2,
   'length': 2
}
let newAry = Array.from(arrayLike, item => item *2);
console.log(newAry); // [2, 4]

注意:如果是对象,那么属性名要写对应的索引,并且一定要指定长度。

③ 实例方法:find()

用于找出第一个符合条件的数组成员,如果没有找到返回undefined,find()的参数是个箭头函数。

let ary = [{
   id: 1,
   name: '张三'
}, { 
   id: 2,
   name: '李四'
}]; 
//找数组里面符合条件的值,当数组中元素id等于2的查找出来。注意,只会匹配第一个
let target = ary.find((item, index) => item.id == 2); 
console.log(target); // 打印的是李四对象

④ 实例方法:findIndex()

用于找出第一个符合条件的数组成员的位置,如果没有找到返回-1,findIndex()的参数是个箭头函数。

let ary = [1, 5, 10, 15];
let index = ary.findIndex((value, index) => value > 9); 
console.log(index); // 2

⑤ 实例方法:includes()

判断某个数组是否包含给定的值,返回布尔值。

[1, 2, 3].includes(2) // true 
[1, 2, 3].includes(4) // false

5.2 String 的扩展方法

① 模板字符串

ES6新增的模板字符串,使用反引号定义

let name = `zhangsan`; // 使用反引号定义
  1. 模板字符串中可以解析变量,格式是:${变量名}
let name = 'zhangsan'; 
let sayHello = `hello,my name is ${name}`;
console.log(sayHello); // 打印:hello, my name is zhangsan
  1. 在模板字符串中可以调用函数,格式是:${函数调用}
const sayHello = function () { 
  return '哈哈哈哈 追不到我吧 我就是这么强大';
}; 
let greet = `${sayHello()} 哈哈哈哈`;
console.log(greet); // 哈哈哈哈 追不到我吧 我就是这么强大 哈哈哈哈
  1. 模板字符串中可以换行
let result = { 
   name: 'zhangsan', 
   age: 20,
   sex: '男' 
} 
let html = ` <div>
   <span>${result.name}</span>
   <span>${result.age}</span>
   <span>${result.sex}</span>
</div> `;
console.log(html);
// 模板字符串可以换行,所以可以把代码写的比较美观,打印如下:
/*
    <div>
      <span>zhangsan</span>
      <span>20</span>
      <span>男</span>
    </div>
*/

② 实例方法:startsWith() 和 endsWith()

  • startsWith():表示参数字符串是否在原字符串的头部,返回布尔值
  • endsWith():表示参数字符串是否在原字符串的尾部,返回布尔值
let str = 'Hello world!';
str.startsWith('Hello') // true 
str.endsWith('!')       // true

③ 实例方法:repeat()

repeat方法表示将原字符串重复n次,返回一个新字符串

'x'.repeat(3)     // "xxx" 
'hello'.repeat(2) // "hellohello"

5.3 Set

ES6 提供了新的数据结构 Set,它类似于数组,但是成员的值都是唯一的,没有重复的值

使用举例:搜索历史关键字(没有重复的值)

  1. Set本身是一个构造函数,用来生成 Set 数据结构
const s = new Set();
  1. Set函数可以接受一个数组作为参数,用来初始化
const set = new Set([1, 2, 3, 4, 4]); // {1, 2, 3, 4}
console.log(set.size);  // 4  set.size 可以获取set存储了多少值
  1. 使用set可以做数组去重,如下:
const s3 = new Set(["a","a","b","b"]);
console.log(s3.size) //2
const ary = [...s3]; //将set通过展开运算符转成数组
console.log(ary)     //["a","b"]

① 实例方法

  • add(value):添加某个值,返回 Set 结构本身
  • delete(value):删除某个值,表示删除是否成功,返回一个布尔值
  • has(value):该值是否为 Set 的成员,返回一个布尔值
  • clear():清除所有成员,没有返回值
const s = new Set();
s.add(1).add(2).add(3); // 向set结构中添加值,返回Set结构本身
s.delete(2)             // 删除set结构中的2值,返回布尔值 
s.has(1)                // set结构中是否有1这个值,返回布尔值 
s.clear()               // 清除set结构中的所有值,没返回值
//注意:删除的是元素的值,不是代表的索引

② 遍历

Set 结构的实例与数组一样,也拥有forEach方法,用于对每个成员执行某种操作,没有返回值。

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

推荐阅读更多精彩内容