手写代码:实现一个ts版的Promise!如何使用 Typescript 实现 Promise,包括 Promise t
Promise API 应该算是写前端的同学使用最多的东西了,今天就来使用 ts 来实现一个 Promise,在手写实现的过程里,可以更加有助于我们对于 Promise 的理解,对以后在工作中使用到 Promise 也会更有帮助!
在这篇文章中我们将一步步实现一个 ts 版的 Promise,包括的 API 如下:
- 创建 Promise: new Promise
- then 方法
- then 方法级联调用
- 异步的 Promise
- Promise.all 方法
一、环境配置
1.使用 npm init -y 初始化一个项目 2.安装 Typescript(我使用的 ts 版本为: "typescript": "^5.5.4") 3.在 package.json 文件中创建一个 tsc 命令,使用 ts-node 用来编译 ts 文件
{
"name": "ts-promise",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"tsc": "ts-node ./src/test.ts"
},
"keywords": [],
"author": "",
"license": "ISC",
"dependencies": {
"typescript": "^5.5.4"
}
}
4.创建 src 文件夹,并在 src 文件夹中新建 index.ts,test.ts,type.ts 文件 结构如下:
index.ts 为手写的 Promise 源码文件 type.ts 为手写 Promise 需要的类型文件 test.ts 为测试代码文件
二、实现 Promise 代码
-
创建 Promise 代码实现
在创建 Promise 之前,先看一下在我们正常使用 Promise 的时候,是如何进行创建的:
const promise = new Promise((resolve, reject) => {});
从上面的使用代码可以看出,我们只是 new 了一个 Promise 对象,并且传入一个函数,函数接受两个参数,resolve 和 reject,那么我们就可以先定义这几个类型:
// type.ts
type ResolveType = (value: any) => void;
type RejectType = (reason: any) => void;
type Executor = (resolve: ResolveType, reject: RejectType) => void;
export { ResolveType, RejectType, Executor };
接下来根据创建 Promise 的方式开始实现 Promise 类:
// index.ts
import { Executor, RejectType, ResolveType } from "./types";
export default class Promise {
public resolve: ResolveType;
public reject: RejectType;
public status: "pending" | "fulfilled" | "rejected";
public resolveValue: any;
public rejectValue: any;
constructor(executor: Executor) {
this.status = "pending";
this.resolve = (value: any) => {
if (this.status === "pending") {
this.status = "fulfilled";
this.resolveValue = value;
}
};
this.reject = (value: any) => {
if (this.status === "pending") {
this.status = "rejected";
this.rejectValue = value;
}
};
executor(this.resolve, this.reject);
}
}
创建 Promise 其实不难理解,只是在 Promise 的构造函数中创建出两个方法,并且在 Promise 类中记录下 resolve 和 reject 的值,以及这个 Promise 的状态。因为 Promise 的状态是不可逆转的,所以需要增加对 status 的判断。
-
then 方法实现
还是先看一下 then 方法的使用方式:
const promise = new Promise((resolve, reject) => {});
promise.then(
(res1) => {
console.log("res: ", res1);
},
(err1) => {
console.log("err1", err1);
}
);
可以看到 then 方法实际上接受两个函数,当我们调用 resolve 的时候,会执行 then 方法中的第一个函数,调用 reject 时,会执行第二个函数,那么 Promise 的 then 方法就可以实现了:
import { Executor, RejectType, ResolveType } from "./types";
export default class Promise {
public resolve: ResolveType;
public reject: RejectType;
public status: "pending" | "fulfilled" | "rejected";
public resolveValue: any;
public rejectValue: any;
constructor(executor: Executor) {
this.status = "pending";
this.resolve = (value: any) => {
if (this.status === "pending") {
this.status = "fulfilled";
this.resolveValue = value;
}
};
this.reject = (value: any) => {
if (this.status === "pending") {
this.status = "rejected";
this.rejectValue = value;
}
};
executor(this.resolve, this.reject);
}
then(onResolved: ResolveType, onRejected: RejectType) {
if (this.status === "fulfilled") {
onResolved(this.resolveValue);
}
if (this.status === "rejected") {
onRejected(this.rejectValue);
}
}
}
-
then 方法级联调用
在使用 Promise 的 then 函数时,then 函数可以进行级联调用,使用方式如下:
const promise = new Promise((resolve, reject) => {
resolve(1);
// reject(2);
});
promise
.then(
(res1) => {
console.log("res: ", res1);
return "ok1";
},
(err1) => {
console.log("err1", err1);
}
)
.then(
(res2) => {
console.log("res2", res2);
return "ok2";
},
(err2) => {
console.log("err2", err2);
}
);
这种级联调用实际上是 then 方法返回了一个新的 Promise 对象,并且将 上次的 resolve 或者 reject 的执行结果给传递到新的 Promise 对象中:
then(onResolved: ResolveType, onRejected: RejectType) {
let result: any;
return new Promise((resolve, reject) => {
if (this.status === "fulfilled") {
result = onResolved(this.resolveValue);
resolve(result);
}
if (this.status === "rejected") {
result = onRejected(this.rejectValue);
reject(result);
}
});
}
-
异步的 Promise
在实际项目中使用 Promise,有可能会遇到在执行一个异步操作后,再执行 resolve 或者 reject 方法,那么就需要修改上面的 Promise 代码来实现: 比如,测试类中要执行一个耗时的异步任务,执行 resolve 函数:
import Promise from "./index";
const promise = new Promise((resolve, reject) => {
setTimeout(() => {
resolve("异步的数据!");
}, 1000);
});
promise.then(
(res1) => {
console.log("res: ", res1);
return "ok1";
},
(err1) => {
console.log("err1", err1);
}
);
执行测试代码后,发现并没有像同步执行 resolve 时那样,将数据打印出来,原因就在于,在我们写的 Promise 类中,并没有处理异步的情况,先执行了 then 函数,然后才执行 resolve 函数,这样 then 函数执行的时候,我们还没有给 resolveValue 赋值!
所以,想要让我们写的 Promise 支持异步,我们就需要修改一下 Promise 类中的代码!
首先,在 Promise 中增加两个数组,用来保存 resolve 函数和 reject 函数:
public resolveCallback: any[] = [];
public rejectCallback: any[] = [];
然后,在 then 方法中增加对 pending 状态的判断,如果是 pending 状态,则把 resolve 和 reject 函数放到数组中:
then(onResolved: ResolveType, onRejected: RejectType) {
console.log("then");
let result: any;
return new Promise((resolve, reject) => {
if (this.status === "fulfilled") {
result = onResolved(this.resolveValue);
resolve(result);
}
if (this.status === "rejected") {
result = onRejected(this.rejectValue);
reject(result);
}
if (this.status === "pending") {
this.resolveCallback.push(() => {
result = onResolved(this.resolveValue);
if (isPromise(result)) {
result.then(
(res) => {
resolve(res);
},
(err) => {
reject(err);
}
);
} else {
resolve(result);
}
});
this.rejectCallback.push(() => {
result = onRejected(this.rejectValue);
if (isPromise(result)) {
result.then(
(res) => {
resolve(res);
},
(err) => {
reject(err);
}
);
} else {
reject(result);
}
});
}
});
}
在执行 resolve 或者 reject 方法的时候,使用 foreach 循环执行 resolveCallback 或者 rejectCallback: 完整代码如下:
import { Executor, RejectType, ResolveType } from "./types";
function isPromise(value: any): value is Promise {
// 是对象并且对象中有 then 方法,则认为是 Promise
return value && typeof value === "object" && typeof value.then === "function";
}
export default class Promise {
public resolve: ResolveType;
public reject: RejectType;
public status: "pending" | "fulfilled" | "rejected";
public resolveValue: any;
public rejectValue: any;
public resolveCallback: any[] = [];
public rejectCallback: any[] = [];
constructor(executor: Executor) {
this.status = "pending";
this.resolve = (value: any) => {
console.log("resolve: ", value);
if (this.status === "pending") {
this.status = "fulfilled";
this.resolveValue = value;
this.resolveCallback.forEach((fn) => fn());
}
};
this.reject = (value: any) => {
if (this.status === "pending") {
this.status = "rejected";
this.rejectValue = value;
this.rejectCallback.forEach((fn) => fn());
}
};
executor(this.resolve, this.reject);
}
then(onResolved: ResolveType, onRejected: RejectType) {
console.log("then");
let result: any;
return new Promise((resolve, reject) => {
if (this.status === "fulfilled") {
result = onResolved(this.resolveValue);
resolve(result);
}
if (this.status === "rejected") {
result = onRejected(this.rejectValue);
reject(result);
}
if (this.status === "pending") {
this.resolveCallback.push(() => {
result = onResolved(this.resolveValue);
if (isPromise(result)) {
result.then(
(res) => {
resolve(res);
},
(err) => {
reject(err);
}
);
} else {
resolve(result);
}
});
this.rejectCallback.push(() => {
result = onRejected(this.rejectValue);
if (isPromise(result)) {
result.then(
(res) => {
resolve(res);
},
(err) => {
reject(err);
}
);
} else {
reject(result);
}
});
}
});
}
}
-
Promise.all 方法
还是先看 Promise.all 方法的使用:
const promise1 = new Promise((resolve, reject) => {
resolve("ok1");
});
const promise2 = new Promise((resolve, reject) => {
resolve("ok2");
});
const promise3 = new Promise((resolve, reject) => {
resolve("ok3");
});
Promise.all([promise1, promise2, promise3]).then(
(res) => {
console.log("res: ", res);
},
(err) => {
console.log("err: ", err);
}
);
可以看到,Promise.all 实际上是一个静态方法,接收一个 promise 的数组,返回一个 Promise 对象,并且会把传递的 promise 数组里每一个 promise 的执行结果,放到一个数组里返回,所以,可以在 Promise 类中这样实现:
static all(promises: Promise[]) {
return new Promise((resolve, reject) => {
const resolveArray: any[] = [];
promises.forEach((p, index) => {
p.then(
(res) => {
processData(res, index);
},
(err) => {
reject(err);
return;
}
);
});
function processData(resolveData: any, index: number) {
resolveArray[index] = resolveData;
if (index == promises.length - 1) {
resolve(resolveArray);
return;
}
}
});
}
Promise.all 方法的实现关键在于要循环执行 promise,将执行结果按顺序放到结果的数组中,然后等到所有的 promise 都执行完成后,一起返回。
总结
通过实现 Promise 的关键 API,可以明白在 Promise 的底层大概是一个怎样的逻辑,可以加深我们对于 Promise API 的理解。
转载自:https://juejin.cn/post/7397640923347599399