目录
前言分析 asyncAdd直观的基本要求隐藏的考察点 — setTimeout & cb隐藏的考察点 — async & await实现 asyncAdd具体实现进行优化抽离内层函数缓存计算结果前言
在掘金上发现一道既简单但个人觉得还挺有意思的一道题,题目如下:
【资料图】
// 异步加法
function asyncAdd(a,b,cb){
setTimeout(() => {
cb(null, a + b)
}, Math.random() * 1000)
}
async function total(){
const res1 = await sum(1,2,3,4,5,6,4)
const res2 = await sum(1,2,3,4,5,6,4)
return [res1, res2]
}
total()
// 实现下 sum 函数。注意不能使用加法,在 sum 中借助 asyncAdd 完成加法。尽可能的优化这个方法的时间。
function sum(){
}
你可以直接尝试实现下,考察下自己的思维和 JavaScript基础知识的联系如何,大佬请绕行!
估计大多数人第一眼看下都不知道这题目到底要干啥(我不说就没人知道我也是),但是在看第二遍的时候估计就差不多明白具体是要考察什么内容了,下面就一起来分析分析吧!!!
分析 asyncAdd
这里先放置最终结论:
只能修改sum部分的内容,sum可接收任意长度的参数sum中只能通过 asyncAdd实现加法计算sum中需要处理异步逻辑,需要使用 Promise需要优化 sum方法的计算时间
下面是分别通过对代码的不同部分进行分析,获取到的相关的信息。
直观的基本要求
// 实现下 sum 函数。注意不能使用加法,在 sum 中借助 asyncAdd 完成加法。尽可能的优化这个方法的时间。
function sum(){ }
最直观的方式就是通过上述的文字描述部分,可以很容易知道题目具体要求:
实现sum函数,即只能修改 sum部分的内容不能直接使用加法(+),通过 asyncAdd实现加法优化 sum方法的计算时间
隐藏的考察点 — setTimeout & cb
// 异步加法
function asyncAdd(a, b, cb){
setTimeout(() => {
cb(null, a + b)
}, Math.random() * 1000)
}
从上述内容来看,最明显的就是 setTimeout和 cb了,其实这不难理解因为在 asyncAdd中使用了 setTimeout只能通过回调函数 cb将本次计算结果返回出去,那其中的第一个参数 null代表什么呢?
其实可以认为它是一个错误信息对象,如果你比较了解 node的话,就会知道在 node中的异步处理的回调函数通常第一个参数就是错误对象,用于传递给外部在发生错误时自定义后续执行逻辑等。
一句话: cb函数会接收 错误对象 和 计算结果 作为参数传递给外部。
隐藏的考察点 — async & await
async function total(){
const res1 = await sum(1,2,3,4,5,6,4)
const res2 = await sum(1,2,3,4,5,6,4)
return [res1, res2]
}
从上述的这部分来看,sum方法的 返回值 肯定是一个 promise类型的,因为最前面明显的使用了 await sum(...)的形式。
另外 total函数返回值也必然是一个 promise类型,因为整个 total函数被定义为了一个 async异步函数,可点击此处查看详细内容。
一句话:sum需要返回 promise类型的值,即 sum一定会使用到 promise,并且从 sum(1,2,3,4,5,6,4)可知 sum可接收任意长度的参数。
实现 asyncAdd
具体实现
实现思路如下:
考虑到外部参数长度不固定,使用剩余运算符接收所有传入的参数考虑到asyncAdd中的异步操作,将其封装为 Promise的实现,即 caculate函数考虑到 asyncAdd实际只能一次接收两个数字进行计算,使用循环的形式将多个参数分别传入考虑到通过循环处理异步操作的顺序问题,使用 async/await来保证正确的执行顺序,且 async函数的返回值正好符合 sum是 Promise类型的要求
具体代码如下:
// 通过 ES6 的剩余运算符(...) 接收外部传入长度不固定的参数
async function sum(...nums: number[]) {
// 封装 Promise
function caculate(num1: number, num2: number) {
return new Promise((resolve, reject) => {
// 调用 asyncAdd 实现加法
asyncAdd(num1, num2, (err: any, rs: number) => {
// 处理错误逻辑
if (err) {
reject(err);
return;
}
// 向外部传递对应的计算结果
resolve(rs);
});
})
}
let res: any = 0;
// 通过遍历将参数一个个进行计算
for (const n of nums) {
// 为了避免异步执行顺序问题,使用 await 等待执行结果
res = await caculate(res, n);
}
return res;
}
进行优化
抽离内层函数
caculate函数可抽离到 sum函数外层asyncAdd函数的回调函数没必要抽离,因为它依赖的参数和外部方法太多
function caculate(num1: number, num2: number) {
return new Promise((resolve, reject) => {
asyncAdd(num1, num2, (err: any, rs: number) => {
if (err) {
reject(err);
return;
}
resolve(rs);
});
})
}
async function sum(...nums: number[]) {
let res: any = 0;
for (const n of nums) {
res = await caculate(res, n);
}
return res;
}
缓存计算结果
其实你仔细观察 total方法,其中 sum调用了两次,而且参数还是一模一样的,目的就是提示你在第二次计算相同内容时结果直接 从缓存中获取,而不是在通过异步计算。
async function total(){
const res1 = await sum(1,2,3,4,5,6,4)
const res2 = await sum(1,2,3,4,5,6,4)
return [res1, res2]
}
以下只是一个简单的缓存方案的实现,不必过于纠结,具体实现如下:
const cash: any = {};
function isUndefined(target: any) {
return target === void 0;
}
async function sum(...nums: number[]) {
let res: any = 0;
const key = nums.join("+");
if (!isUndefined(cash[key])) return cash[key];
for (const n of nums) {
res = await caculate(res, n);
}
cash[key] = res;
return res;
}
function caculate(num1: number, num2: number) {
return new Promise((resolve, reject) => {
asyncAdd(num1, num2, (err: any, rs: number) => {
if (err) {
reject(err);
return;
}
resolve(rs);
});
})
}以上就是JavaScript手写异步加法asyncAdd方法详解的详细内容,更多关于JavaScript异步加法asyncAdd的资料请关注脚本之家其它相关文章!
X 关闭
X 关闭
- 1联想拯救者Y70发布最新预告:售价2970元起 迄今最便宜的骁龙8+旗舰
- 2亚马逊开始大规模推广掌纹支付技术 顾客可使用“挥手付”结账
- 3现代和起亚上半年出口20万辆新能源汽车同比增长30.6%
- 4如何让居民5分钟使用到各种设施?沙特“线性城市”来了
- 5AMD实现连续8个季度的增长 季度营收首次突破60亿美元利润更是翻倍
- 6转转集团发布2022年二季度手机行情报告:二手市场“飘香”
- 7充电宝100Wh等于多少毫安?铁路旅客禁止、限制携带和托运物品目录
- 8好消息!京东与腾讯续签三年战略合作协议 加强技术创新与供应链服务
- 9名创优品拟通过香港IPO全球发售4100万股 全球发售所得款项有什么用处?
- 10亚马逊云科技成立量子网络中心致力解决量子计算领域的挑战

