ECMAScript 6 新特性(二)
ECMAScript 6 新特性(二)
ECMAScript 6 新特性(一)
ECMAScript 6 新特性(二)(本文)
ECMAScript 7~10 新特性
1. 生成器
生成器函数是 ES6 提供的一种解决异步编程方案,一种特殊的函数,语法行为与传统函数完全不同。
function* gen() {yield 1; // yield 关键字用来暂停函数的执行yield 2;yield 3;
}let iterator = gen();
// 调用
// console.log(iterator); // 不能调用
// iterator.next(); // 正常调用// for (let v of gen()) {
// console.log(v); // 每次返回一个 yield 的值
// }
// 等同于
console.log(iterator.next());
console.log(iterator.next());
console.log(iterator.next());
console.log(iterator.next());
代码说明:
*的位置没有限制;- 生成器函数返回的结果是迭代器对象,调用迭代器对象的 next 方法可以得到 yield 语句后的值;
- yield 相当于函数的暂停标记,也可以认为是函数的分隔符,每调用一次 next 方法,执行一段代码;
- next 方法可以传递实参,作为 yield 语句的返回值 。
1.1 生成器函数参数
function* gen(arg) {console.log(arg);let one = yield 1;console.log(one);let two = yield 2;console.log(two);let three = yield 3;console.log(three);
}let iterator = gen("aaaaa");
console.log(iterator.next());
// next 方法可以传入实参,传入参数会被赋值给 yield 表达式的值
console.log(iterator.next("BBB"));
console.log(iterator.next("CCC"));
console.log(iterator.next('DDD'));
1.2 实例
-
1s后控制台输出111,2s后输出222,3s后输出333
传统实现
setTimeout(() => {console.log("111");setTimeout(() => {console.log("222");setTimeout(() => {console.log("333");}, 3000);}, 2000); }, 1000);代码较为复杂,不易扩展,使用生成器函数实现
function one() {setTimeout(() => {console.log("111");iterator.next();}, 1000); } function two() {setTimeout(() => {console.log("222");iterator.next();}, 2000); } function three() {setTimeout(() => {console.log("333");iterator.next();}, 3000); }function* gen() {yield one();yield two();yield three(); }// 调用生成器函数 let iterator = gen(); iterator.next(); -
模拟获取用户数据、订单数据、商品数据
function getUsers() {setTimeout(() => {let data = "用户数据";// 调用 next 方法,并将数据传入iterator.next(data);}, 1000); } function getOrders() {setTimeout(() => {let data = "订单数据";iterator.next(data);}, 1000); } function getGoods() {setTimeout(() => {let data = "商品数据";iterator.next(data);}, 1000); }function* genData() {let users = yield getUsers();console.log(users);let orders = yield getOrders();console.log(orders);let goods = yield getGoods();console.log(goods); }// 调用生成器函数 let iterator = genData(); iterator.next();
2. Promise
Promise 是 ES6 引入的异步编程的新解决方案。语法上 Promise 是一个构造函数,用来封装异步操作并可以获取其成功或失败的结果。
- Promise 构造函数:
Promise (excutor) {}; Promise.prototype.then方法;Promise.prototype.catch方法。
2.1 then 方法
实例化 Promise 时,使用回调函数作为参数,回调函数通常有两个参数:
-
resolve 参数
当执行到
resolve( ... )时,会调用 then 方法中的第一个参数(回调); -
reject 参数
当执行到
reject( ... )时,会调用 then 方法中的第二个参数(回调);
then 方法中通常有两个回调函数作为参数,第一个回调在成功时(resolve)调用,第二个回调在出错时(reject)调用,第二个参数可以省略。
基本使用
// 实例化 Promise 对象
const p = new Promise((resolve, reject) => {setTimeout(() => {let data = "Hello, world!";// resolve(data); // 调用 then 方法中第一个回调reject(new Error("出错了")); // 调用 then 方法中第二个回调}, 1000);
});// 调用 Promise 对象的 then 方法
p.then((value) => {console.log(value);}, // 成功回调(reason) => {console.error(reason);} // 失败回调
);
下面列举几个使用 Promise 进行封装的案例:
2.1.1 读取文件
// 引入模块
const fs = require("fs");// 使用Promise 封装
const p = new Promise((resolve, reject) => {fs.readFile("./resources/为学.md", (err, data) => {// 如果失败,则抛出错误if (err) reject(err);// 否则,打印文件内容resolve(data);});
});p.then((value) => {console.log(value.toString());},(reason) => {console.log("出错了:" + reason);}
);
2.1.2 发送 AJAX 请求
// 封装 Promise 对象
const p = new Promise((resolve, reject) => {// 1. 创建对象const xhr = new XMLHttpRequest();// 2. 初始化xhr.open("GET", "https://dog.ceo/api/breeds/image/random");// 3. 发送xhr.send();// 4. 绑定事件,处理响应结果xhr.onreadystatechange = function () {if (xhr.readyState === 4) {if (xhr.status >= 200 && xhr.status < 300) {resolve(xhr.responseText);} else {reject(xhr.statusText);}}};
});// 指定回调
p.then((data) => {console.log(data);},(error) => {console.log(error);}
);
2.1.3 Promise.prototype.then
// 创建 Promise 对象
const p = new Promise((resolve, reject) => {resolve("用户数据");// reject("出错了");
});const result = p.then((data) => {console.log(data);},(error) => {console.warn(error);}
);console.log(result);
2.1.4 说明
调用 then 方法的返回结果是 Promise 对象,对象状态由回调函数的执行结果决定:
-
返回结果是非 Promise 类型的属性
返回状态 resolved(成功),返回值为对象成功的值
const result = p.then((data) => {console.log(data);return 123;},(error) => {console.warn(error);} );console.log(result); // 返回值为 123如果未使用 return 进行返回,则返回值为 undefined。
-
返回 Promise 对象
返回值和返回状态均由返回的 promise 对象的返回值和状态决定
const result = p.then((data) => {console.log(data);return new Promise((resolve, reject) => {resolve("ok");// reject("出错了");});},(error) => {console.warn(error);} ); console.log(result); // 返回状态为 resolved,返回值为 ok // console.log(result); // 返回状态为 rejected,返回值为 出错了 -
抛出错误
返回状态 rejected(失败)
const result = p.then((data) => {console.log(data);// throw new Error("出错了");throw "出错了";},(error) => {console.warn(error);} );console.log(result); // 返回状态为 rejected,返回值为 出错了
由于 promise 可以返回 promise 对象,因此可以进行链式调用
// 链式调用
p.then((data) => {},(error) => {}
).then((data) => {},// 失败回调可以省略
)...;
同时避免的回调地域的问题
2.2 catch 方法
指定 Promise 对象失败的回调
const p = new Promise((resolve, reject) => {setTimeout(() => {reject("出错了");}, 1000);
});p.catch((err) => {console.log(err); // 输出 "出错了"
});
3. Set
ES6 提供了新的数据结构 Set(集合)。它类似于数组,但成员的值都是唯一的,集合实现了 iterator 接口,所以可以使用扩展运算符和 for…of… 进 行遍历,集合的属性和方法:
- size 返回集合的元素个数
- add 增加一个新元素,返回当前集合
- delete 删除元素,返回 boolean 值
- has 检测集合中是否包含某个元素,返回 boolean 值
- clear 清空集合,返回 undefined
// 声明一个 set
let s1 = new Set();
let s2 = new Set([1, 2, 3, 4, 5]);console.log(s1, typeof s1); // Set(0) {} object
console.log(s2, typeof s2);// 元素个数
console.log(s2.size); // 5
// 添加元素
s2.add(6);
console.log(s2); // Set(6) {1, 2, 3, 4, 5, 6}
// 删除元素
s2.delete(1);
console.log(s2); // Set(5) {2, 3, 4, 5, 6}
// 判断元素是否存在
console.log(s2.has(2)); // true
console.log(s2.has(7)); // false
// 清空集合
s2.clear();
console.log(s2); // Set(0) {}
4. Map
ES6 提供了 Map 数据结构。它类似于对象,也是键值对的集合。但是“键” 的范围不限于字符串,各种类型的值(包括对象)都可以当作键。Map 也实现了 iterator 接口,所以可以使用扩展运算符和 for…of… 进行遍历。Map 的属 性和方法:
- size 返回 Map 的元素个数
- set 增加一个新元素,返回当前 Map
- get 返回键名对象的键值
- has 检测 Map 中是否包含某个元素,返回 boolean 值
- clear 清空集合,返回 undefined
// 声明 map
let m = new Map();// 向 map 中添加键值对
m.set("name", "张三");
m.set("change", () => {console.log("change");
});
m.set({ age: 25 }, ["李四"]);
// size
console.log(m.size);
// 删除
m.delete("name");
// 获取
console.log(m.get("change"));
// 清空
m.clear();console.log(m);
5. Class
ES6 提供了更接近传统语言的写法,引入了 Class(类)这个概念,作为对象的模板。通过 class 关键字,可以定义类。基本上,ES6 的 class 可以看作只是一个语法糖,它的绝大部分功能,ES5 都可以做到,新的 class 写法只是让对象原型的写法更加清晰、更像面向对象编程的语法而已。
知识点:
-
class 声明类
class 类名 {属性 = 值;...方法() {...} } -
constructor 定义构造函数初始化
class 类名 {// 构造方法,实例化时自动调用constructor(参数1, 参数2) {this.属性1 = 参数1;this.属性2 = 参数2;}... // 属性、方法 }// 实例化对象 let 对象 = new 类名(参数1, 参数2); -
extends 继承父类
class 父类 {... }class 子类 extends 父类 {... } -
super 调用父级构造方法
class 父类 {constructor(参数1, 参数2) {this.属性1 = 参数1;this.属性2 = 参数2;} }class 子类 extends 父类 {constructor(参数1, 参数2, 参数3, 参数4) {super(参数1, 参数2); // 继承父类的构造方法// 子类的属性this.属性1 = 参数3;} } -
static 定义静态方法和属性
class 类名 {static 属性 = 值;static 方法() {...} }静态属性和方法不能被读取和继承。
-
子类重写父类方法
子类中可以声明一个跟父类同名的方法
class 父类 {方法() {...} }class 子类 extends 父类 {// 重写父类中的同名方法方法() {...} }
5.1 get 和 set
class Phone {get price() {console.log("价格属性被读取");return 3999}set price(value) {console.log("价格属性被修改");}
}let p = new Phone();console.log(p.price);
p.price = "free";
6. 数值扩展
Number.EPSILON 是 JavaScript 的最小精度,即 2.220446049250313e-16。
当两个数的差值小于该值,就可以认为这两个数相等,主要用于浮点数的计算。
console.log(0.1 + 0.2 === 0.3); // false,因为浮点数计算不精确
console.log(Math.abs(0.1 + 0.2 - 0.3) < Number.EPSILON); // true
// 这时就可以认为 0.1 + 0.2 等于 0.3
6.1 二进制、八进制、十六进制
| 进制 | 前缀 |
|---|---|
| 二进制 | 0b |
| 八进制 | 0o |
| 十六进制 | 0x |
let a = 0b1010; // 10
let b = 0o777; // 511
let c = 0x1F; // 31console.log(a, b, c);
6.2 Number 方法
-
Number.isFinite()检查是否为有限数console.log(Number.isFinite(Infinity)); // false console.log(Number.isFinite(-Infinity)); // false console.log(Number.isFinite(NaN)); // false console.log(Number.isFinite(0)); // trueInfinity:无穷大NaN:非数值
-
Number.isNaN()检查是否为NaNconsole.log(Number.isNaN(NaN)); // true console.log(Number.isNaN(0)); // false -
Number.parseInt()和Number.parseFloat()用于将字符串转换为数字let d = "123"; let e = "123.456"; let f = "0b1010";console.log(Number.parseInt(d)); // 123 console.log(Number.parseFloat(e)); // 123.456 console.log(Number.parseInt(f)); // 0这两个方法都会忽略字符串开头的空格,并且只解析到第一个非数字字符为止。
-
Number.isInteger()检查是否为整数console.log(Number.isInteger(123)); // true console.log(Number.isInteger(123.456)); // false -
Math.trunc()用于截断小数部分,返回整数部分let i = 123.456; let j = -123.456;console.log(Math.trunc(i)); // 123 console.log(Math.trunc(j)); // -123 -
Math.sign()用于判断一个数的正负号数 返回值 正数 1 零 0 负数 -1 console.log(Math.sign(5)); // 1 console.log(Math.sign(-5)); // -1 console.log(Math.sign(0)); // 0
7. 对象方法扩展
-
Object.is()判断两个值是否完全相等console.log(Object.is(120, 121)); // false console.log(Object.is(1.1 + 1.2, 2.3)); // true作用类似于 == 或 ===,但区别在于对 NaN 的判断
console.log(Object.is(NaN, NaN)); // true console.log(NaN === NaN); // false -
Object.assign()用于两个对象的合并const config1 = {host: "localhost",port: 3306,pass: "root", }; const config2 = {host: "www.baidu.com",port: "8080", }; console.log(Object.assign(config1, config2));合并两个对象,相同属性的值,后者覆盖前者
-
Object.setPrototypeOf()设置对象的原型对象Object.getPrototypeOf()获取对象的原型对象const school = {name: "学院", }; const cities = {xiaoqv: ["商丘", "开封", "洛阳"], }; Object.setPrototypeOf(school, cities); console.log(school); console.log(Object.getPrototypeOf(school));
8. 模块化
模块化是指将一个大的程序文件,拆分成许多小的文件,然后将小文件组合起来。
8.1 模块化的好处
模块化的优势有以下几点
- 防止命名冲突
- 代码复用
- 高维护性
- 开发人员对文件的修改不会产生过多冲突
- 当某个功能需要修改、更新时,只需要对单个模块修改即可
8.2 模块化规范产品
ES6 之前的模块化规范有:
| 规范 | 产品 |
|---|---|
| CommonJS | NodeJS、Browserify |
| AMD | requireJS |
| CMD | seaJS |
8.3 ES6 模块化语法
模块功能主要由两个命令构成:export 和 import。
export命令用于规定模块的对外接口(暴露模块接口);import命令用于输入其他模块提供的功能(导入其它模块)。
暴露接口的几种方式
-
默认暴露:
export default 接口;每个模块只能有一个默认暴露。
导入:
import 接口 from ...; -
命名暴露:
export 属性名/对象名/...;命名导出允许你导出多个值,并且每个导出都有一个名称。
导入:
import { 属性名/对象名/... } from ...; -
混合暴露:
export { 属性名/对象名/..., 属性名2/对象名2/... as default };在一个模块中同时使用默认暴露和命名暴露。
导入:
import 属性名2/对象名2/..., { 属性名/对象名/... } from ...; -
重新暴露
可以从一个模块中导出另一个模块的导出内容。
模块1:
export 属性名/对象名/...;模块2:
export { 属性名/对象名/... } from ...;…
导入:
import { 属性名/对象名/... } from ...;
相关文章:
ECMAScript 6 新特性(二)
ECMAScript 6 新特性(二) ECMAScript 6 新特性(一) ECMAScript 6 新特性(二)(本文) ECMAScript 7~10 新特性 1. 生成器 生成器函数是 ES6 提供的一种解决异步编程方案,一…...
SpringBoot接口覆盖上一次调用的实现方案
调用springboot接口时,如何实现覆盖上一次调用 Spring Boot 接口覆盖上一次调用的实现方案 以下是多种实现覆盖上一次接口调用的方案,适用于不同场景。 方案一:同步锁控制(单机环境) 适用场景:单实例…...
Spring 的 IoC 和 DI 详解:从零开始理解与实践
Spring 的 IoC和 DI 详解:从零开始理解与实践 一、IoC(控制反转) 1、什么是 IoC? IoC 是一种设计思想,它的核心是将对象的创建和管理权从开发者手中转移到外部容器(如 Spring 容器)。通过这种…...
Python Cookbook-5.12 检查序列的成员
任务 你需要对一个列表执行很频繁的成员资格检査。而in操作符的 O(n)时间复杂度对性能的影响很大,你也不能将序列转化为一个字典或者集合,因为你还需要保留原序列的元素顺序。 解决方案 假设需要给列表添加一个在该列表中不存在的元素。一个可行的方法…...
签名过期怎么办?
1无论是证书到期还是被封停,只需要找到签名服务商,重新签名就可以了,但签名经常性过期会造成app用户流失,所以我们在选择签名时需要注意,在资金充足的情况下,优先选择独立、稳定签名,接下来我们…...
ZYNQ笔记(四):AXI GPIO
版本:Vivado2020.2(Vitis) 任务:使用 AXI GPIO IP 核实现按键 KEY 控制 LED 亮灭(两个都在PL端) 一、介绍 AXI GPIO (Advanced eXtensible Interface General Purpose Input/Output) 是 Xilinx 提供的一个可…...
实操(环境变量)Linux
环境变量概念 我们用语言写的文件编好后变成了程序,./ 运行的时候他就会变成一个进程被操作系统调度并运行,运行完毕进程相关资源被释放,因为它是一个bash的子进程,所以它退出之后进入僵尸状态,bash回收他的退出结果&…...
【补题】P9423 [蓝桥杯 2023 国 B] 数三角
题意:小明在二维坐标系中放置了 n 个点,他想在其中选出一个包含三个点的子集,这三个点能组成三角形。然而这样的方案太多了,他决定只选择那些可以组成等腰三角形的方案。请帮他计算出一共有多少种选法可以组成等腰三角形ÿ…...
Word / WPS 页面顶部标题 段前间距 失效 / 不起作用 / 不显示,标题紧贴页眉 问题及解决
问题描述: 在 Word 或者 WPS 里面,如果不是新的一节,而是位于新的一页首行时,不管怎么设置段前间距,始终是失效的,实际段前间距一直是零。 解决方案: 查询了很多方案均无法解决问题ÿ…...
Mysql自动增长数据的操作(修改增长最大值)
在MySQL中,如果你想要修改一个表的自增长(auto-increment)属性的起始值,可以使用ALTER TABLE语句。这对于初始化新环境或修复损坏的自增长计数器特别有用。下面是如何操作的一些步骤: 查看当前自增长值 首先ÿ…...
Linux自行实现的一个Shell(15)
文章目录 前言一、头文件和全局变量头文件全局变量 二、辅助函数获取用户名获取主机名获取当前工作目录获取最后一级目录名生成命令行提示符打印命令行提示符 三、命令处理获取用户输入解析命令行执行外部命令 四、内建命令添加环境变量检查和执行内建命令 五、初始化初始化环境…...
在 Q3D 中提取汇流条电感
汇流条排简介和设计注意事项 汇流条排是用于配电的金属导体,在许多应用中与传统布线相比具有设计优势。在设计母线排时,必须考虑几个重要的因素: 低电感:高频开关内容会导致无功损耗,从而降低效率电容:管…...
MySQL:事务的理解
一、CURD不加控制,会有什么问题 (1)因为,MySQL里面存的是数据,所以很有可能会被多个客户访问,所以mysqld可能一次会接受到多个关于CURD的请求。(2)且mysql内部是采用多线程来完成数…...
[raspberrypi 0w and respeaker 2mic]实时音频波形
0. 环境 ubuntu22主机, 192.168.8.162, raspberry 0w, 192.168.8.220 路由器 1. 树莓派 # rpi - send.py # 或者命令行:arecord -D plughw:1,0 -t wav -f cd -r 16000 -c 2 | nc 192.168.8.162 12345import socket imp…...
python 基础:句子缩写
n int(input()) for _ in range(n):words input().split()result ""for word in words:result word[0].upper()print(result)知识点讲解 input()函数 用于从标准输入(通常是键盘)读取用户输入的内容。它返回的是字符串类型。例如在代码中…...
Ruoyi-vue plus 5.2.2 flowble 结束节点异常错误
因业务要求, 我在结束节点的结束事件中,制作了一个归档的事件,来执行一个业务。 始终都会报错, 错误信息 ${archivTemplateListener} did not resolve to an implementation of interface org.flowable.engine.delegate.Execution…...
Sublime Text使用教程(用Sublime Text编写C语言程序)
Sublime Text 是一款当下非常流行的文本编辑器,其功能强大(提供有众多的插件)、界面简洁、还支持跨平台使用(包括 Mac OS X、Linux 和 Windows)。 在程序员眼中,Sublime Text 不仅仅是一个文本编辑器&…...
【1】k8s集群管理系列--包应用管理器之helm
一、helm概述 Helm核心是模板,即模板化K8s YAML文件。 通过模板实现Chart高效复用,当部署多个应用时,可以将差异化的字段进行模板化,在部署时使用-f或 者–set动态覆盖默认值,从而适配多个应用 helm工作流程…...
【书籍】DeepSeek谈《持续交付2.0》
目录 一、深入理解1. 核心理念升级:从"自动化"到"双环模型"2. 数字化转型的五大核心能力3. 关键实践与案例4. 组织与文化变革5. 与其它框架的关系6. 实际应用建议 二、对于开发实习生的帮助1. 立刻提升你的代码交付质量(技术验证环实…...
Mysql表的操作(2)
1.去重 select distinct 列名 from 表名 2.查询时排序 select 列名 from 表名 order by 列名 asc/desc; 不影响数据库里面的数据 错误样例 : 但结果却有点出乎意料了~为什么会失败呢? 其实这是因为书写的形式不对,如果带了引号,…...
智能物联网网关策略部署
实训背景 某智慧工厂需部署物联网网关,实现以下工业级安全管控需求: 设备准入控制:仅允许注册MAC地址的传感器接入(白名单:AA:BB:CC:DD:EE:FF)。协议合规性:禁止非Modbus TCP(端口…...
神经网络语言模型与统计语言模型的比较
神经网络语言模型(Neural Language Models, NLMs)与统计语言模型(Statistical Language Models, SLMs)是自然语言处理(NLP)中两类核心的语言建模方法,其核心差异体现在建模方式、表示能力、数据…...
Java学习总结-线程池
线程池是什么? 线程池就是一个可以复用线程的技术。 假若不用线程池的问题:创建新线程开销很大,不能来一个任务就就创建一个新线程。 如何创建线程池对象? 方法一:使用ExecutorService的实现类ThreadPoolExecutor创…...
Android 中绕过hwbinder 实现跨模块对audio 的HAL调用
需求 Audio 模块中专门为 TV 产品添加了一些代码,需要在 hdmi 的 HAL 代码中进行调用以完成某些功能。 解决方法 首先将 hdmi HAL 要调用的 audio 接口函数所在的 .so 链接到最基本的 lib.primay.amlogic.so 中(其它平台上这个 .so 文件的名字也可能是…...
【DB2】事务日志满/归档占用较大问题处理记录
某DB2环境经常报错The active log is full and is held by...,并且归档磁盘占用较大 事务日志满 事务日志满可以理解为Oracle的redo追尾,即业务写入量大于redo刷盘速度,这时候其他SQL会陷入等待,容易造成性能问题 一般由两方面原…...
基于CNN-BiLSTM-GRU的深度Q网络(Deep Q-Network,DQN)求解移动机器人路径规划,MATLAB代码
一、深度Q网络(Deep Q-Network,DQN)介绍 1、背景与动机 深度Q网络(DQN)是深度强化学习领域的里程碑算法,由DeepMind于2013年提出。它首次在 Atari 2600 游戏上实现了超越人类的表现,解决了传统…...
CVE-2025-29927 Next.js 中间件鉴权绕过漏洞
Next.js Next.js 是一个基于 React 的现代 Web 开发框架,用来构建高性能、可扩展的 Web 应用和网站。 CVE-2025-29927 Next.js 中间件鉴权绕过漏洞 CVE-2025-29927是Next.js框架中的一个授权绕过漏洞,允许攻击者通过特制的HTTP请求绕过在中间件中执行…...
数据结构(五)——AVL树(平衡二叉搜索树)
目录 前言 AVL树概念 AVL树的定义 AVL树的插入 右旋转 左旋转 左右双旋 右左双旋 插入代码如下所示 AVL树的查找 AVL树的遍历 AVL树的节点个数以及高度 判断平衡 AVL树代码如下所示 小结 前言 前面我们在数据结构中介绍了二叉搜索树,其中提到了二叉搜…...
C++类型转换详解
目录 一、内置 转 内置 二、内置 转 自定义 三、自定义 转 内置 四、自定义 转 自定义 五、类型转换规范化 1.static_case 2.reinterpret_cast 3.const_cast 4.dynamic_cast 六、RTTI 一、内置 转 内置 C兼容C语言,在内置类型之间转换规则和C语言一样的&am…...
【前端】【React】性能优化三件套useCallback,useMemo,React.memo
一、总览:性能优化三件套 useCallback(fn, deps):缓存函数,避免每次渲染都新建函数。useMemo(fn, deps):缓存值(计算结果),避免重复执行计算。React.memo(Component):缓存组件的渲染…...
