ES6+
本文是我在学习 JavaScript ES6+
这一部分内容时知识点的记录。
参考资料:MDN JavaScript Docs
ES6 语法汇总:
ES7~ES11 语法汇总:
let、块级作用域、const 常量
let 声明变量
- 使用
let
定义的变量在同一作用域下不可被重复定义:
let b = 1
let b = 2 // 报错 语法错误:标识符“b”已经被声明
// SyntaxError: Identifier 'b' has already been declared
let c = 1
var c = 2 // 报错 语法错误:标识符“c”已经被声明
// SyntaxError: Identifier 'c' has already been declared
- 使用
var
声明过的变量也不可被let
重复声明:
var a = 1
let a = 2 // 报错 语法错误:标识符“a”已经被声明
// SyntaxError: Identifier 'a' has already been declared
- 使用 let 声明的变量在预编译时不会提升,会产生一个暂时性死区:
console.log(a) // 报错,引用错误: 在初始化之前不能访问 'a'
let a = 1
// ReferenceError: Cannot access 'a' before initialization
function test(x = y, y = 2) {
console.log(x, y)
}
test() // 报错,引用错误: 在初始化之前不能访问 'y'
// ReferenceError: Cannot access 'y' before initialization
console.log(typeof a) // 报错,引用错误: 在初始化之前不能访问 'a'
let a
// ReferenceError: Cannot access 'a' before initialization
let
只能在当前的作用域下生效:
if (1) {
let a = 2
}
console.log(a) // 报错
for (let i = 0; i < 10; i++) {
let a = 1
}
console.log(a) // 报错
console.log(i) // 报错
块级作用域
花括号中的是独立的块级作用域:
let a = 1
{
let a = 2
console.log(a) // 2
}
console.log(a) // 1
for
循环中有两个作用域:
for(父级作用域) {
子级作用域
}
for(let i = 0; i < 10; i++) {
let i = 'abc';
console.log(i);
}
// 上面的 for 循环执行后将会打印 10 个 abc
// 因为 for 括号中的 和 花括号中的 是两个不同的作用域
// 花括号作用域是括号作用域的子作用域,括号作用域又是全局作用域的子作用域,这样就形成了一个作用域链:
// 自顶到底: 全局作用域 --> 括号作用域 --> 花括号作用域
const 声明常量
使用 const
声明的是常量,常量是不可被修改的:
// 使用 const 声明一个常量:
const a = 1
console.log(a) // 1
// 常量不可被修改,下面的代码将抛出错误
a = 2 // 报错
// 常量不可被删除
delete a
console.log(a) // 1
模版字符串 & 字符串新增方法
``
反引号定义模版字符串,在模版字符串中可以随意换行。
字符串模板中使用 ${}
的形式添加变量
使用这种方法拼接字符串比使用
+
的方式更简洁美观
var name = 'Bob'
var age = 19
var message = `Hello ${name}, Your age is ${age}. `
console.log(message)
// Hello Bob, Your age is 19.
字符串新增的一些方法:
str.includes()
在字符串中查找指定的字符,返回布尔值
var str = 'Hello World!'
var message = str.includes('llo')
console.log(message)
// true
str.startsWith()
检测字符串是否以指定的字符开头,返回布尔值
var str = 'Hello World!'
var message = str.startsWith('H')
console.log(message)
// true
str.endsWith()
检测字符串是否以指定的字符结尾,返回布尔值
var str = 'Hello World!'
var message = str.endsWith('!')
console.log(message)
// true
str.repeat()
指定字符串被重复多少次,返回重复指定次数的字符串
var str = 'Hello World!'
var message = str.repeat(3)
console.log(message)
// Hello World!Hello World!Hello World!
str.padStart()
往字符串开头填充指定的字符(需指定填充后字符串的长度)
var name = 'Bob'
var age = 19
var message = name.padStart(5, age)
console.log(message)
// 19Bob
str.padEnd()
往字符串末尾填充指定的字符(需指定填充后字符串的长度)
var name = 'Bob'
var age = 19
var message = name.padEnd(5, age)
console.log(message)
// Bob19
解构赋值
一个基本的结构赋值:
// 解构赋值,等号两边的格式必须一致
let [a, b, c] = [1, 2, 3]
console.log(a, b, c) // 1 2 3
// let [a, [b, c]] = [1, [2, 3]]
// console.log(a, b, c) // 1 2 3
解构对象
// 解构对象
// 直接把对象的键提取出来并用花括号包裹起来就可以解构对象
let {
name,
age,
email,
likes: { first: firstLike }
} = {
name: 'Bob',
age: 18,
email: 'Bob@gmail.com',
likes: {
first: 'game',
second: 'study'
}
}
console.log(name, age, email, firstLike) // Bob 18 Bob@gmail.com game
指定别名
// key:name 指定别名,在键后写冒号,冒号后面指定的字符就替代了键名
let {
name: n,
age: g,
email: e
} = {
name: 'Bob',
age: 18,
email: 'Bob@gmail.com'
}
console.log(n, g, e) // Bob 18 Bob@gmail.com
设置默认值
// 设置默认值,仅在值为 undefined 时默认值生效
let [a, b, c = '默认值'] = [1, 2]
console.log(a, b, c) // 1 2 "默认值"
解构函数返回值
// 解构函数的返回值
let people = () => ({ name: 'Bob', age: 18 })
let { name, age } = people() // 使用定义好的变量接收函数返回的值
console.log(name, age) // Bob 18
引入模块时的解构
// 这里的 a b c 对应的是 test 模块中返回的三个值方法
import { a, b, c } from './test'
函数形参使用解构赋值
// 函数形参使用解构赋值
let fun = ({ name, age }) => console.log(name + "'s age is " + age)
fun({
name: 'Bob',
age: 20
}) // Bob's age is 20
this 指向、箭头函数、...运算符、函数参数
this
this
指向的 4
条规则:
- 默认绑定: 默认指向
window
:
function test() {
console.log(this)
}
test()
// Window {parent: Window, opener: null, top: Window, length: 0, frames: Window, …}
- 隐式绑定:被谁调用就指向谁:
function test() {
console.log(this.a)
}
var obj = {
a: 1,
b: 2,
test: test
}
obj.test()
// 1
- 显示绑定:
call(obj, a, b, c)
、apply(obj, [a, b, c])
、bind(obj, a, b, c)
var obj = {
name: 'Bob',
age: 19
}
function sayName() {
console.log(this.name)
}
sayName.call(obj)
// Bob
new
构造函数:
function Cars(name, color) {
this.name = name
this.color = color
this.sayName = function () {
console.log(this.name)
}
}
var car = new Cars('BMW', 'red')
car.sayName()
// BMW
=>
箭头函数 - 箭头函数是由
=>
操作符定义的; - 箭头函数是一个函数表达式;
- 箭头函数的
this
指向定义这个函数时所在的对象,而不是运行函数时所在的对象; - 构造函数不能使用箭头函数;
- 箭头函数中不存在
arguments
,可以使用...
运算符; - 能用到回调函数的地方都能使用箭头函数 一个基本的箭头函数最基本的定义形式:
() => {};
当仅有一个参数或一个返回值时,定义一个箭头函数的语法为:
var 函数名 = 形参 => 返回值;
var a = 'test';
var f = a => a + ' Hello World!';
console.log(f(a)); // test Hello World!
// 等价于
function f(a) {
return a + ' Hello World!';
}
console.log(f(a)); // test Hello World!
当有多个形参或多条函数内容时,定义一个箭头函数的语法为:
var 函数名 = (形参1, 形参2 ...) => {
// 函数体
}
var a = 1;
var b = 2;
var add = (x, y) => {
var result = x + y;
return result;
}
console.log(add(a, b));
// 3
//等价于
function add(x, y) {
var result = x + y;
return result;
}
...
spread/rest 运算符
用于展开或收集数组
箭头函数中没有 arguments
实参列表,但可以使用 ...
操作符来获取实参列表;
展开或重置的基本用法:
// ... 运算符展开或收集数组
// 展开数组
let arr = [1, 2, 3, 4, 5]
console.log(...arr) // 1 2 3 4 5
// 收集数组(主要用于函数的形参中)
function makeArray(...names) {
names.forEach((item) => console.log(item))
}
makeArray('Bob', 'Jack', 'Henry')
// Bob
// Jack
// Henry
...
收集实参列表或数组
// ... 运算符(收集实参列表)
// ... 后默认就是一个数组,传多少个实参都会被放入到这个数组中
var sum = (...args) => {
console.log(args)
console.log(args[0] + args[1])
}
sum(1, 2)
// [1, 2]
// 3
// ... 运算符收集 剩余的所有实参数组 必须是最后一个参数
// 收集剩余参数时,...必须放到形参列表的末尾,这样前面依然可以正常传实参,多余的实参会被收入到该数组中
let fn = (a, b, ...c) => {
console.log(a, b, c)
}
fn(1, 2, 3, 4, 5)
// 1 2 [3, 4, 5]
...
展开实参列表或数组
// ... 运算符(展开实参列表)
function foo(x, y, z) {
console.log(x, y, z)
}
foo(...[1, 2, 3])
// 等价于
foo(1, 2, 3)
// 等价于
foo.apply(null, [1, 2, 3])
// 1 2 3
// 1 2 3
// 1 2 3
// ... 运算符 展开数组
var a = [2, 3, 4]
let b = [1, ...a, 5]
console.log(b)
// [1, 2, 3, 4, 5]
箭头函数与解构赋值结合
// 箭头函数与解构赋值结合
var full = ({ first, last } = {}) => first + '' + last
console.log(full({ first: 3, last: 5 })) // '35'
// 等价于
function full({ first, last } = {}) {
return first + '' + last
}
箭头函数应用于数组排序
// 箭头函数应用于数组排序
var arr = [109, 59, 6, 9, 15, 4651, 13, 52, 1235]
var arr1 = arr.sort((a, b) => a - b)
console.log(arr1)
// [6, 9, 13, 15, 52, 59, 109, 1235, 4651]
// 使用 ...args 收集实参数组
const sortNum = (...args) => args.sort((a, b) => a - b)
console.log(sortNum(109, 59, 6, 9, 15, 4651, 13, 52, 1235))
// [6, 9, 13, 15, 52, 59, 109, 1235, 4651]
函数参数
函数默认参数,可在定义函数形参时指定默认值:
// 形参默认值
// 指定默认值的形参应该放到形参列表的末尾
function people(name, age = 19) {
console.log(name + "'s age is " + age)
}
函数参数默认已经定义了,不能再使用 let
, const
声明:
// 函数参数默认已经定义了,不能再使用 let const 声明
function test(age = 18) {
let age = 20 // 错误,形参 age 已经被定义
}
数组新增
数组新增方法
Array.from(arr)
将类数组转为纯数组,返回一个纯数组
let lis = document.querySelectorAll('ul li')
console.log(lis) // Nodelist(5) [li, li, li, li, li] (类数组)
// Array.from(arr) 将类数组转为纯数组
let liArr = Array.from(lis)
console.log(liArr) //(5) [li, li, li, li, li] (纯数组)
// Array.from() 还可以将所有带有 length 属性的类型都转为数组
let str = 'Hello' // 字符串带有 length 属性,转换字符串
console.log(Array.from(str)) // ["H", "e", "l", "l", "o"]
let obj = {
0: 'apple',
1: 'banana',
2: 'orange',
length: 3
} // 转换类数组
console.log(Array.from(obj)) // ["apple", "banana", "orange"]
Array.of(arg1, arg2[,arg3 ...])
把一组值转为数组,类似于new Array()
// Array.of(arg1, arg2[,arg3 ...]) 将一组值转为数组,类似于 new Array
let ofArr = Array.of('apple', 'banana', 'orange')
console.log(ofArr) // ["apple", "banana", "orange"]
数组实例方法
arr.find((item, index, arr) => {})
遍历数组,在找到第一个对回调函数的条件返回true
的项时,返回该项,找不到则返回undefined
// arr.find((item, index, arr) => {}) 遍历数组,在数组中查找指定的项,返回第一个找到的项,没找到返回 undefined
let nums = [1, 4, 6, 76, 234, 110]
let result = nums.find((item, index, arr) => {
return item > 100
})
console.log(result) // 234
arr.findIndex((item, index, arr) => {})
在数组中查找满足条件的项,返回该项的索引,没找到返回-1
// arr.findIndex((item, index, arr) => {}) 遍历数组,在数组中查找满足条件的项,返回该项的索引,没找到返回 -1
let nums = [1, 4, 6, 76, 234, 110]
let result = nums.findIndex((item, index, arr) => {
return item > 100
})
console.log(result) // 4
arr.fill(填充的值,开始位置,结束位置)
填充数组如果不指定开始位置和结束位置则全部填充
// arr.fill(item, startIndex, endIndex) 填充数组
let emptyArr = new Array(5)
console.log(emptyArr) // [empty × 5]
let newArr = emptyArr.fill('apple', 0, 3)
console.log(newArr) // ["apple", "apple", "apple", empty × 2]
arr.includes(item)
判断数组中是否包含指定的项,返回布尔值
// arr.includes(item) 判断数组中是否包含指定的项,返回布尔值
let colors = ['red', 'green', 'orange']
console.log(colors.includes('green')) // true
console.log(colors.includes('blue')) // false
数组循环相关
ES5
中数组数组实例可以调用的各种循环方法
arr.forEach((item, index, arr) => {})
代替普通for
循环,没有返回值arr.map((item, index, arr) => {})
正常情况下,需要配合return
返回的是一个新的数组,若是没有return
,相当于forEach
。注意:平时只要用
map
,一定要有return
。arr.filter((item, index, arr) => {})
过滤数组,把对回调函数中的条件返回true
的项加入到一个新的数组然后返回arr.some((item,index,arr) => {})
查询数组,只要数组中有一项对回调函数返回true
,some()
方法就返回true
,否则就返回false
arr.every((item, index, arr) => {})
查询数组,必须数组中的所有项都对回调函数返回true
,every()
方法才返回true
,否则就返回false
reduce((prev, cur, index, arr) => {})
递归数组,得到数组中所有的项累计(加减乘除)的值,从左往右reduceRight((prev, cur, index, arr) => {})
递归数组,得到数组中所有的项累计(加减乘除)的值,从右往左
ES6
新增循环数组方法
for...of...
for of
循环默认可以拿到数组中的每一项,而不是数组的每一项的索引
// for of 循环 默认循环数组,拿到数组中的每一项
let arr = ['apple', 'banana', 'orange']
for (item of arr) {
console.log(item)
}
// apple
// banana
// orange
arr.keys()
返回只包含索引的数组Array Iterator {}
,此数组只能被迭代;for (index of arr.keys()) {}
拿到数组中每一项的索引:
// for (index of arr.keys()) {} 拿到数组中每一项的索引
let arr = ['apple', 'banana', 'orange']
for (index of arr.keys()) {
console.log(index)
}
// 0
// 1
// 2
arr.entries()
返回包含索引和值的数组Array Iterator {}
,此数组只能被迭代;for ([index, item] of arr.entries()) {}
拿到数组中每一项的索引和值:
// for ([index, item] of arr.entries()) {} 拿到数组中每一项的索引和值
let arr = ['apple', 'banana', 'orange']
for ([index, item] of arr.entries()) {
console.log(index, item)
}
// 0 "apple"
// 1 "banana"
// 2 "orange"
arr.values()
返回包含值的数组Array Iterator {}
,此数组只能被迭代;for (item of arr.values()) {}
拿到数组中每一项的索引和值:for of
默认循环的就是arr.values()
// for (item of arr.values()) {} 拿到数组中的每一项,for of 默认循环的就是 arr.values()
let arr = ['apple', 'banana', 'orange']
for (item of arr.values()) {
console.log(item)
}
// apple
// banana
// orange
对象新增
对象简洁语法
- 可以将相同名称的属性和值变为只写为一个属性名
- 方法可以直接声明,但不能使用箭头函数,会造成
this
混乱
// 对象简洁语法,可以将相同名称的属性和值只写为一个属性名
// 方法可以直接声明,但不能使用箭头函数,会造成 this 混乱
let name = 'Bob'
let age = 10
let obj = {
// name: name,
name, // 等价于上面 // age: age,
age, // 等价于上面 // sayName: function () { // console.log(this.name) // },
sayName() {
// 等价于上面
console.log(this.name)
}
}
console.log(obj) // {name: "Bob", age: 10, sayName: ƒ }
Object
新增方法
对象构造函数以下方法通过对象构造函数 Object
来调用
Object.is(arg1, arg2)
判断两个参数是否相等(使用这种方法时,NaN
和NaN
是相等的),返回布尔值
//Object.is(arg1, arg2) 判断两个参数是否相等
// 使用这种方法时,NaN 和 NaN 是相等的
// -0 和 +0 不相等
console.log(Object.is(NaN, NaN)) // true
console.log(Object.is(-0, +0)) // false
console.log(Object.is('aaa', 'aaa')) // true
Object.assign({}/[], sourceObj1[,sourceObj2 ...])
合并对象或数组,将多个对象或数组合并到一个指定的对象中,返回一个新的对象用途:复制一个对象或数组,合并参数
注意:此方法会改变位于第一个参数的对象
// Object.assign({}/[], sourceObj1[,sourceObj2 ...]) 合并对象或数组,将多个对象或数组合并到一个指定的对象中
// 参数1:合并后的对象或数组
// 参数2:要合并的对象1
// 参数3:要合并的对象2
// ...
// 注意:后面指定的对象中如果有相同的属性或方法,后面的覆盖前面的
let obj1 = { name: 'Bob' }
let obj2 = { age: 20 }
let obj3 = { name: 'Suxi' }
let json = Object.assign({}, obj1, obj2, obj3)
// 有相同的属性,后面的覆盖前面的
console.log(json) // {name: "Suxi", age: 20}
// 复制数组
let arr = ['apple', 'banana', 'orange']
let arr2 = Object.assign([], arr)
console.log(arr2) // ["apple", "banana", "orange"]
Object.keys()
取出对象的键,返回键的数组Object.values()
取出对象的值,返回值的数组Object.entries()
取出对象的键和值,返回键和值的数组
// Object.keys() 取出对象的键,返回键
// Object.values() 取出对象的值,返回值
// Object.entries() 取出对象的键和值,返回键和值
let colors = {
apple: 'red',
banana: 'yellow',
orange: 'orange'
}
// 遍历对象的键
for (let key of Object.keys(colors)) {
console.log(key)
}
// apple
// banana
// orange
// 遍历对象的值
for (let value of Object.values(colors)) {
console.log(value)
}
// red
// yellow
// orange
// 遍历对象的键和值
for (let item of Object.entries(colors)) {
console.log(item)
}
// ["apple", "red"]
// ["banana", "yellow"]
// ["orange", "orange"]
// 结构对象的 keys()、values() 和 entries() 方法
let { keys, values, entries } = Object
// 使用结构的方法,得到跟上面一样的结果
for (let key of keys(colors)) {
console.log(key)
}
for (let value of values(colors)) {
console.log(value)
}
for (let item of entries(colors)) {
console.log(item)
}
Promise
Promise
对于解决异步问题,基本使用:
// Promise 对象,解决异步
// 被 promise 测试的变量
let a = 10
// 创建一个 Promise 实例
let promise = new Promise(function (resolve, reject) {
// resolve 成功的时候返回的结果
// reject 失败的时候返回的结果
// 这里定义成功或失败后返回的结果,
// 在 Promise 实例的 .then 方法的 回调的 参数中 自动传入
if (a == 10) {
// 这里定义 .then(res) 中 res 的内容
resolve('成功')
} else {
// 这里定义 .then(err) 中 err 的内容
reject('失败')
}
})
// Promise 实例的 .then 方法,接收两个回调,一个是成功的回调,一个是失败的回调
// 这里回调接收到的参数就是创建 Promise 实例时所设置的 resolve 和 reject 的值
promise.then(
(res) => {
// 成功的回调
console.log(res)
},
(err) => {
// 失败的回调
console.log(err)
}
)
// 失败的回调也可以放在 .catch 中:
promise.catch((err) => {
// 等价于 .then 的第二个参数
console.log(err)
})
// 因为 .then 返回的还是 promise 对象,所有可以在 .then 后面直接 .catch
promise
.then((res) => {
console.log(res)
})
.catch((err) => {
console.log(err)
})
// 这就是一个 promise 的基本流程
Promise.resolve()
Promise.resolve()
创建一个 Promise
实例 并 定义这个实例的成功状态
// Promise.resolve() 创建一个 Promise 实例 并 定义这个实例的成功状态
let pTest1 = Promise.resolve('成功')
pTest1.then((res) => console.log(res)) // 成功
等价于:
new Promise((resolve) => {
resolve('成功')
})
Promise.reject()
Promise.reject()
创建一个 Promise
实例并 定义这个实例的的失败状态
// Promise.reject() 创建一个 Promise 实例 并 定义这个实例的的失败状态
let pTest2 = Promise.reject('失败')
pTest2.catch((err) => console.log(err)) // 失败
Promise.all()
Promise.all()
将多个 Promise
实例包装成一个新的 Promise
实例。同时,成功和失败的返回值是不同的,成功的时候返回的是一个结果数组,而失败的时候则返回最先被 reject
失败状态的值。
注意:必须确保放入到数组
Promise
实例都是resolve
状态,也就是成功状态,否则没什么意义
// Promise.all() 将多个Promise实例包装成一个新的Promise实例。
// 成功和失败的返回值是不同的,成功的时候返回的是一个结果数组
// 而失败的时候则返回最先被 reject 失败状态的值。
let p1 = Promise.resolve('成功1')
let p2 = Promise.resolve('成功2')
let p3 = Promise.reject('失败 3')
Promise.all([p1, p2]).then((res) => console.log(res)) // ["成功1", "成功2"]
Promise.all([p1, p2, p3])
.then((res) => console.log(res))
.catch((err) => console.log(err)) // 失败 3
Promise.all
在处理多个异步处理时非常有用,比如说一个页面上需要等两个或多个 ajax
的数据回来以后才正常显示,在此之前只显示 loading
图标。
需要特别注意的是,Promise.all
获得的成功结果的数组里面的数据顺序和 Promise.all
接收到的数组顺序是一致的,即 p1 的结果在前,即便 p1 的结果获取的比 p2 要晚。这带来了一个绝大的好处:在前端开发请求数据的过程中,偶尔会遇到发送多个请求并根据请求顺序获取和使用数据的场景,使用 Promise.all
毫无疑问可以解决这个问题。
Promise.race()
Promise.race()
顾名思义,Promse.race
就是赛跑的意思,意思就是说,Promise.race([p1, p2, p3])
里面哪个结果获得的快,就返回那个结果,不管结果本身是成功状态还是失败状态。
// Promise.race() 把一组 Promise 实例的成功或失败的结果放入到数组中,
// 返回数组中最先有结果(不论是成功或失败)的 Promise 实例 的结果
let p1 = new Promise((resolve, reject) => {
setTimeout(() => {
resolve('成功 1')
}, 1000)
})
let p2 = new Promise((resolve, reject) => {
setTimeout(() => {
reject('失败 2')
}, 500)
})
Promise.race([p1, p2])
.then((res) => {
console.log(res)
})
.catch((err) => {
console.log(err)
}) // 失败 2,因为失败状态先被捕获到
类(class)和继承
ES5 构造函数
先来回顾一下 ES5
的构造函数
// ES5 构造函数
function Person(name, age) {
this.name = name
this.age = age
}
let person = new Person('Bob', 20)
// 打印出对象的构造函数,就是上面定义的 Person
console.log(person.constructor) // ƒ Person(name, age) {...}
// 给构造函数原型添加方法
Person.prototype.sayName = function () {
console.log(this.name)
}
Person.prototype.sayAge = function () {
console.log(this.age)
}
// 实例对象调用方法
person.sayName() // Bob
person.sayAge() // 20
// 添加方法也可以使用下面这种形式
Object.assign(Person.prototype, {
sayMessage() {
console.log(`你的名字是:${this.name},你的年龄是:${this.age}`)
}
})
// 调用方法
person.sayMessage() // 你的名字是:Bob,你的年龄是:20
ES6 class
基本使用
// ES6 class 类
// 创建类:
class Person1 {
constructor(name, age) {
// 使用 new 创建实例时,该 constructor 方法自动运行
// new 传过来的参数都在该方法的形参中
console.log(
`使用 new 创建实例,我被自动执行了,接收到的参数是:${name} ${age}`
)
}
}
// 创建实例
// 不传参数,两个 undefined
let person1 = new Person1() // 使用 new 创建实例,我被自动执行了,接收到的参数是:undefined undefined
// 传递参数
let person2 = new Person1('Jack', 40) // 使用 new 创建实例,我被自动执行了,接收到的参数是:Jack 40
//类中的属性和方法:
// constructor 方法用来接收属性
// 其他方法可以写在 constructor 方法后面,方法之间不需要加逗号
class Person2 {
constructor(name, age) {
this.name = name
this.age = age
} // 方法之间不需要加逗号!
sayName() {
console.log(this.name)
}
sayAge() {
console.log(this.age)
}
}
// 创建实例
let peiQi = new Person2('佩奇', 3)
// 调用方法
peiQi.sayName() // 佩奇
peiQi.sayAge() // 3
// 也可以使用表达式的方式创建类
// 这种方式与上面的方式结果一样,推荐使用上面的方式
const Persons = class {}
注意:使用
class
声明的类没有预解析提升的功能,必须先声明类后再使用
get
set
get
取值,set
存值
// get 取值,set 存值
class Car {
_color = ''
constructor(color) {
this._color = color
}
set color(value) {
console.log('设置Car的颜色为:' + value)
this._color = value
}
get color() {
console.log('获取Car的颜色为:' + this._color)
return this._color
}
}
const car = new Car('red')
// 实例取值
car.color // 获取Car的颜色为:red
// 实例存值
car.color = 'green' // 设置Car的颜色为:green
// 实例取值
car.color // 获取Car的颜色为:green
static
static
声明类本身的静态方法,只能通过类本身调用
通过
static
声明的方法,可以被子类继承(会在后面介绍)
// static 声明类身上的静态方法,只能通过类本身调用
class Person {
constructor() {}
message() {
console.log('这是 message 方法')
}
static method() {
console.log('这是类身上的静态方法 method')
}
}
let person = new Person()
// 通过实例调用普通方法
person.message() // 这是 message 方法
// 通过类本身调用静态方法
Person.method() // 这是类身上的静态方法 method
父类、子类、继承
- 子类继承父类,使用
extends
关键字,语法:class
子类名extends
父类名 - 子类继承父类的属性,使用
super()
方法 - 子类继承父类后,子类的
this
和父类的this
指向同一个地方 - 子类不能有跟父类相同的属性名,否则会覆盖掉父类的属性
- 子类跟父类有相同的方法名时,先在自己的方法中运行父类的方法,然后再运行自己的方法
// 创建父类
class Car {
constructor(color) {
// 初始化属性
this.color = color
}
sayColor() {
console.log(this.color)
}
}
// 创建子类
// extends 指定要继承的父类
// 语法:class 子类名 extends 父类名
class XnyCar extends Car {
// constructor 的第一个参数是父类拥有的属性
constructor(color, name) {
// super() 将父类里的东西接收过来放到指定的参数里
super(color)
this.name = name
}
sayName() {
console.log(this.name)
}
}
// 创建父类实例
let car = new Car('red')
// 正常使用自己的方法
car.sayColor() // red
// 创建子类实例
let xnyCar = new XnyCar('green', 'BMW')
// 子类使用父类的方法
xnyCar.sayColor() // green
// 子类使用自己的方法
xnyCar.sayName() // BMW
如果子类中的方法和父类中的方法同名,则子类会覆盖父类的方法;
可以在冲突的方法中使用 super.父类的方法
,先执行父类的方法,再执行子类的逻辑:
// 子类父类方法名冲突,默认子类覆盖父类
// 解决方法:在子类的方法中先运行父类的方法,再运行自己的方法
class Father {
constructor(fname) {
this.fname = fname
}
sayName() {
console.log(`爸爸的名字是:${this.fname}`)
}
}
class Children extends Father {
constructor(fatherName, name) {
// 继承父类的属性
super(fatherName)
this.name = name
}
sayName() {
// super.sayName() 先运行父类的方法,再定义子类的方法
super.sayName()
console.log(`儿子的名字是:${this.name}`)
}
}
let children = new Children('老王', '小王')
children.sayName()
// 爸爸的名字是:老王
// 儿子的名字是:小王
模块化
注意:模块化需要运行在服务器环境中
定义模块
在需要导出的 js
文件中的变量名前加上 export
:
export let a = 1
export let b = 2
export let c = 3
// 也可以写成下面这种形式
let d = 4
let e = 5
let f = 6
export {
d, // 相当于 d:d
e as eee, // 指定别名 , 这里的别名就是引入文件时的指向该变量的名称
f as fff // 指定别名
}
注意:使用 export 导出的变量在引入时必须加上花括号,使用 export default 导出的变量引入时则不需要加花括号
// 使用 export 导出的数据在引入时必须加上花括号,使用 export default 导出的数据引入时则不需要加花括号
export const a = 12
export default 'test'
引入模块中某个功能
import { xxx } from './xxx'
引入模块中某个功能
// import {xxx} from './xxx' 引入模块中某个功能
// 引入的变量必须是在模块文件中导出(export)了的
// 同时引入一个模块中的多个变量,必须使用 {} 包裹要引入的变量,即使只有一个
import { b, c, d } from './modules/test.js'
console.log(b, c, d)
// 引入模块时指定别名
import { b as q, c as w, d as e } from './modules/test.js'
console.log(q, w, e)
// import * as xx from './xxx' 引入一个模块下的所有导出的变量,
// 返回一个 Module 对象,对象中的属性对应了所有导出的变量
// let a = 1,
// b = 2,
// c = 3;
// export {
// a,
// b,
// c
// }
import * as obj from './modules/2.js'
console.log(obj) // {a: 1, b: 2, c: 3}
console.log(obj.a, obj.b, obj.c) // 1 2 3
// 只有在模块中指定了 export default 时,引入变量才不需要加花括号 {}
// export default "Bob"
import name from './modules/name.js'
console.log(name) // Bob
// 同时引入 export 和 export default 导出的数据,花括号必须在后面
// 数据如下:
// export const a = 12;
// export default "test";
// 引入
import test, { a } from './modules/test.js'
console.log(test, a) // test 12
引入模块文件
在 script
标签中 type="module"
, 然后使用 import
关键字引入模块文件
<script type='module'>
// 引入整个文件 import './modules/test.js' // 也可以引入 url import
'https://code.jquery.com/jquery-3.5.0.min.js'
</script>
注意: 这种方式只是引入了整个文件,并不是引入模块中的某个功能。
import
模块只会导入一次,后面再导入就不会生效
import
特点
import
可以是相对路径,也可以是绝对路径(url)import
导入模块只会导入一次,无论导入了多少次import './modules/1.js
如果这么用,相当于引入了整个模块文件- 有预编译变量提升效果,
import
会自动提升到顶部,首先执行 - 导入的文件是响应式的,源文件发生改变,导入的地方也会改变
import
引入文件,是静态的,只能先引入文件,然后在下面使用,不能写在分支语句(if else
)中
import()
import()
函数,可以动态引入文件,该方法引入文件会返回一个 Promise
对象,可以在 .then
中使用模块中的数据(是异步的)
// import() 动态引入文件,返回 Promise 对象 , 成功状态的数据就是一个 Module 对象,其中包含了模块中的所有数据
// 在分支语句中使用 import() 函数动态引入模块并使用模块中的数据
let temp = 2
if (temp === 1) {
// 模块文件内容如下:
// export default "Bob";
import('./modules/name.js').then((res) => console.log(res)) // {default: "Bob"}
} else {
// 模块文件内容如下:
// export const a = 12;
// export default "test";
// let
// b = 2,
// c = 3,
// d = 4;
// export {
// b,
// c,
// d
// }
import('./modules/test.js').then((res) => console.log(res)) // { a: 12, b: 2, c: 3, d: 4, default: "test" }
}
import()
特点:
- 按需加载
- 可以写在
if
中 - 路径也可以动态
async
、await
async
和 await
主要是为了解决异步队列的问题
在 async
和 await
之前解决异步是使用,生成器 function * fn() { yield xxx }
的方法
语法:
// async 表示这个函数中有异步队列
async function fn() {
let result = await xxx // await 表示执行完当前的异步任务才能执行后面的
let result2 = await yyy // 异步任务2
let result3 = await zzz // 异步任务...
}
async
和 await
特点:
await
只能放到async
函数中- 相比
genrator
语义化更强 await
后面可以是Promise
对象,也可以是数字、字符串、布尔等async
函数返回的是一个Promise
对象- 只要
async
函数中有一个await
的Promise
状态变为reject
(失败),那么整个async
函数就会中断执行注意:因为
async
中的await
只要有任何一个出现错误都会影响整个函数的执行,所以要把每一个可能会出现错误的await
都使用try{}catch(e){}
捕获
Set
Set
是一种数据结构,类似于数组,但里面不能有重复值(类似于 Python 中的集合)
Set
构造函数接收的是一个数组new Set([])
WeakSet
构造函数接收的是一个对象new WeakSet({})
(这种写法不靠谱,而且WeakSet
没有size
,故下面就不展开介绍了)Set
构造函数原型上的属性和方法:WeakSet
构造函数原型上的属性和方法:
Set
类型的属性和方法
// 创建一个 Set
let setArr = new Set(['a', 'b', 'c'])
console.log(setArr) // {"a", "b", "c"}
// Set 不能有重复值
let setArr2 = new Set(['a', 'b', 'c', 'a', 'b'])
console.log(setArr2) // {"a", "b", "c"}
// add() 往 Set 末尾添加值,一次只能添加一个
setArr.add('d')
console.log(setArr) // {"a", "b", "c", "d"}
// delete() 删除 Set 中指定的值
setArr.delete('b')
console.log(setArr) // {"a", "c", "d"}
// has() 判断 Set 中有没有指定的值,返回布尔值
console.log(setArr.has('b')) // false
console.log(setArr.has('a')) // true
// size 查看 Set 里面值的个数,返回数值
console.log(setArr.size) // 3
// clear() 清空 Set
setArr.clear()
console.log(setArr) // {}
// 循环
for (item of setArr2) {
console.log(item)
}
// a
// b
// c
// keys() Set中每一项的键
for (item of setArr2.keys()) {
console.log(item)
}
// a
// b
// c
// values() Set 中每一项的值
for (item of setArr2.values()) {
console.log(item)
}
// a
// b
// c
// 通过上面三个例子我们发现,Set 循环的默认值、keys()、values() 结果都是一样的
// Set 循环的值默认是 values()
// entries() Set 中每一项的键和值
for (item of setArr2.entries()) {
console.log(item)
}
// ["a", "a"]
// ["b", "b"]
// ["c", "c"]
// forEach((item, index) => {}) 遍历 Set,回调函数中的 item 和 index 是每一项的值和键
setArr2.forEach((item, index) => {
console.log(item, index)
})
// a a
// b b
// c c
Set
的特性给数组去重
使用 let arr = [1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 7, 5]
// [...Set] 把 Set 转为数组
let newArr = [...new Set(arr)]
console.log(newArr) // [1, 2, 3, 4, 5, 6, 7]
Map
Map
是一种类似对象的数据类型,Map
的key
可以是任何类型;WeakMap
的Key
只能是对象,用处不大。。Map
构造函数原型上的属性和方法:WeakMap
上的属性和方法:
Map
的基本方法和属性
// map 是一种类似对象的类型
// 使用:
// 创建一个 map
let map = new Map()
console.log(map) // {}
// set(key, value) 设置一个键值对,此方法返回自身,可以链式调用
map.set('name', 'Bob')
console.log(map) // {"name" => "Bob"}
// get(key) 获取一个值
console.log(map.get('name')) // Bob
// delete(key) 删除一项
map.delete('name')
console.log(map) // {}
// 删完了,再增加两项
map.set('name', 'Bob').set('age', 20)
console.log(map) // {"name" => "Bob", "age" => 30}
// has(key) 判断有没有指定的项,返回布尔值
console.log(map.has('name')) // true
console.log(map.has('sex')) // false
// size 返回 Map 中的项数
console.log(map.size) // 2
// clear() 清空 map
map.clear()
console.log(map) // {}
Map
的遍历方法:
// Map 的循环
let newMap = new Map()
newMap.set('name', 'Bob').set('age', 30).set('sex', 'male')
console.log(newMap) // {"name" => "Bob", "age" => 30, "sex" => "male"}
// 基本循环,每一项都是一个数组
for (let item of newMap) {
console.log(item)
}
// ["name", "Bob"]
// ["age", 30]
// ["sex", "male"]
// keys() 拿到 map 的每一个键
for (let key of newMap.keys()) {
console.log(key)
}
// name
// age
// sex
// values() 拿到 map 的每一个值
for (let value of newMap.values()) {
console.log(value)
}
// Bob
// 30
// male
// entries() 拿到 Map 的每一个键值对,默认Map循环的就是 entries
for (let item of newMap.entries()) {
console.log(item)
}
// ["name", "Bob"]
// ["age", 30]
// ["sex", "male"]
// forEach((value, key) => {}) 循环 Map,回调函数中第一个参数是 值,第二个参数是 键
newMap.forEach((value, key) => {
console.log(key, value)
})
// name Bob
// age 30
// sex male
Map
和 Set
各自的特性
Set
里面是数组,不重复,没有key
没有get
和set
方法;Map
是对json
功能的增强,key
可以是任意类型的值
异步循环
ES9
引入了异步迭代器(asynchronous iterators),使 await
可以和 for...of
循环一起使用
async function foo(array) {
for await (let i of array) {
doSomething(i)
}
}
??
空值合并运算符
空值合并操作符 ??
是一个逻辑操作符,当左侧的操作数为 null
或者 undefined
时,返回其右侧操作数,否则返回左侧操作数。
语法:
leftExpr ?? rightExpr
与逻辑或操作符 ||
不同,逻辑或操作符会在左侧操作数为假值时返回右侧操作数。也就是说,如果使用 ||
来为某些变量设置默认值,可能会遇到意料之外的行为。比如为假值(例如,'' 或 0)时。见下面的例子。
const foo = null ?? 'default string';
console.log(foo);
// "default string"
const baz = 0 ?? 42;
console.log(baz);
// 0
const bar = 0 || 1;
console.log(bar);
// 1
?.
可选链操作符
可选链操作符 ?.
允许读取位于连接对象链深处的属性的值,而不必明确验证链中的每个引用是否有效。?.
操作符的功能类似于 .
链式操作符,不同之处在于,在引用为空 (nullish ) (null 或者 undefined) 的情况下不会引起错误,该表达式短路返回值是 undefined
。与函数调用一起使用时,如果给定的函数不存在,则返回 undefined
。
语法:
obj?.prop
obj?.[expr]
arr?.[index]
func?.(args)
当尝试访问可能不存在的对象属性时,可选链操作符将会使表达式更短、更简明。在探索一个对象的内容时,如果不能确定哪些属性必定存在,可选链操作符也是很有帮助的。
const person = {
name: 'qiuxc',
cat: {
name: 'Baiwan'
}
}
console.log(person.dog?.name)
console.log(person.someMethod?.())
// undefined
// undefined
通过连接的对象的引用或函数可能是 undefined
或 null
时,可选链操作符提供了一种方法来简化被连接对象的值访问。
比如,思考一个存在嵌套结构的对象 obj。不使用可选链的话,查找一个深度嵌套的子属性时,需要验证之间的引用,例如:
let nestedProp = obj.first && obj.first.second;
为了避免报错,在访问 obj.first.second 之前,要保证 obj.first 的值既不是 null
,也不是 undefined
。如果只是直接访问 obj.first.second,而不对 obj.first 进行校验,则有可能抛出错误。
有了可选链操作符 ?.
,在访问 obj.first.second 之前,不再需要明确地校验 obj.first 的状态,再并用短路计算获取最终结果:
let nestedProp = obj.first?.second;
通过使用 ?.
操作符取代 . 操作符,JavaScript 会在尝试访问 obj.first.second 之前,先隐式地检查并确定 obj.first 既不是 null
也不是 undefined
。如果 obj.first 是 null
或者 undefined
,表达式将会短路计算直接返回 undefined
。