导航
因为 var 存在两个特性或者说问题:
而 let、const 解决了这些问题:
在 ES5 中,顶层对象的属性和全局变量是等价的,var 命令和 function 命令声明的全局变量,自然也是顶层对象。
var a = 12;
function f() {};
console.log(window.a); // 12
console.log(window.f); // f() {}
但 ES6 规定,var 命令和 function 命令声明的全局变量,依旧是顶层对象的属性,但 let 命令、const 命令、class 命令声明的全局变量,不属于顶层对象的属性。
let aa = 1;
const bb = 2;
console.log(window.aa); // undefined
console.log(window.bb); // undefined
在哪里?怎么获取?通过在设置断点,看看浏览器是怎么处理的:
通过上图也可以看到,在全局作用域中,用 let 和const 声明的全局变量并没有在全局对象中,只是一个块级作用域(Script)中
怎么获取?在定义变量的块级作用域中就能获取啊,既然不属于顶层对象,那就不加 window(global)呗。
let aa = 1;
const bb = 2;
console.log(aa); // 1
console.log(bb); // 2
const a = () => 1;
console.log(a.prototype); // undefined
...args
是个真数组arguments
是类数组等到当前脚本的同步任务和“任务队列”中已有的事件,全部处理完以后,才会执行 setTimeout 指定的任务
参考:JavaScript 运行机制详解:再谈Event Loop
来源:https://www.mulingyuer.com/archives/758/
//模拟异步请求
function axiosFn() {
return new Promise((resolve, reject) => {
const flge = Math.random(); //随机值
setTimeout(() => {
//大于0.7就是成功
if (flge > 0.7) {
return resolve(flge);
} else {
return reject(flge);
}
}, 1000)
})
}
/**
* @description: promise 失败重试方法
* @param {*} fn 异步函数
* @param {*} times 次数
* @Date: 2022-02-20 22:47:54
* @Author: mulingyuer
*/
Promise.retry = function(fn, times) {
return new Promise(async (resolve, reject) => {
while (times--) {
try {
const res = await fn();
//请求成功,结束while循环
return resolve(res);
} catch (error) {
//请求失败
if (times > 0) {
console.log(`请求失败正在重试,还剩${times}次,错误信息为:${error}`);
} else {
console.log(`请求失败,重试${times}次后,还是失败,错误信息为:${error}`);
return reject(error);
}
}
}
})
}
//测试
Promise.retry(axiosFn, 3).then(res => {
console.log(`获得的数据为:${res}`);
}).catch(error => {
console.log(`失败了,错误信息为:${error}`);
});
是 generator + Promise 的语法糖,主要的作用是用同步方式执行异步操作,await 只能在 async 函数中使用,async 函数执行会返回一个 Promise,值由函数的 return 值所决定
try catch
捕获错误,不能 return比如 sleep(1000) 意味着等待1000毫秒,可从 Promise、Generator、Async/Await 等角度实现。
// Promise
const sleep = delay => new Promise(resolve => setTimeout(resolve, delay))
sleep(1000).then(() => console.log('Done'));
// Generator
function* sleepGenerator(delay) {
yield new Promise(resolve => setTimeout(resolve, delay))
}
sleepGenerator(1000).next().value.then(() => console.log('Done'));
// async / await
const sleep = delay => new Promise(resolve => setTimeout(resolve, delay))
async function output() {
await sleep(1000)
console.log('Done')
}
output()
// ES5
function sleep(cb, delay) {
if (typeof cb === 'function')
setTimeout(cb, delay)
}
function output() {
console.log('Done')
}
sleep(output, 1000)
let fs = require('fs')
let arr = []
function fn(data) {
arr.push(data)
if (arr.length === 2) {
console.log(arr)
}
}
fs.readFile('./name.text', 'utf8', (err, data) => {
fn(data)
})
fs.readFile('./age.text', 'utf8', (err, data) => {
fn(data)
})
let fs = require('fs')
let arr = []
let i = 0
function fn(data, index) {
arr[index] = data
if (++i === 2) {
console.log(arr)
}
}
fs.readFile('./name.text', 'utf8', (err, data) => {
fn(data, 1)
})
fs.readFile('./age.text', 'utf8', (err, data) => {
fn(data, 0)
})
let fs = require('fs')
function after(times, cb) {
let arr = []
return function(data) {
arr.push(data)
if (--times === 0) {
cb(arr)
}
}
}
let newFn = after(3, function(arr) {
console.log(arr)
})
fs.readFile('./name.text', 'utf8', (err, data) => {
newFn(data)
})
fs.readFile('./age.text', 'utf8', (err, data) => {
newFn(data)
})
fs.readFile('./weight.text', 'utf8', (err, data) => {
newFn(data)
})
fn([
() => console.log('start'),
() => sleep(1000),
() => console.log('1'),
() => sleep(2000),
() => console.log('2'),
() => sleep(3000),
() => console.log('end'),
]);
实现:
function sleep(delay) {
return new Promise(resolve => setTimeout(resolve, delay));
}
async function fn(arr) {
for (let i = 0; i < arr.length; i++) {
await arr[i]();
}
// 或者
// for (const item of arr) {
// await item();
// }
}
fn([
() => console.log('start'),
() => sleep(1000),
() => console.log('1'),
() => sleep(2000),
() => console.log('2'),
() => sleep(3000),
() => console.log('end'),
]);
function sleep(delay) {
return new Promise(resolve => setTimeout(resolve, delay));
}
async function fn(arr) {
arr.forEach(item => {
await item();
});
}
fn([
() => console.log('start'),
() => sleep(1000),
() => console.log('1'),
() => sleep(2000),
() => console.log('2'),
() => sleep(3000),
() => console.log('end'),
]);
实现:
Array.prototype.forEach = async function(callback, thisArg) {
let _arr = this,
isArray = Array.isArray(_arr),
_thisArg = thisArg ? Object(thisArg) : window;
if (!isArray) {
throw new TypeError('forEach调用者得是个数组');
}
for (let i = 0; i < _arr.length; i++) {
await callback.call(_thisArg, _arr[i], i, _arr);
}
}
function sleep(delay) {
return new Promise(resolve => setTimeout(resolve, delay));
}
async function fn(arr) {
arr.forEach(async item => {
await item();
});
}
fn([
() => console.log('start'),
() => sleep(1000),
() => console.log('1'),
() => sleep(2000),
() => console.log('2'),
() => sleep(3000),
() => console.log('end'),
]);
为了不影响后续.then的执行,需要在每一个then中指定失败的回调
let asyncFunc = () => new Promise(resolve => {
resolve("123"); // 123, 二楼
// throw new Error("出错了"); // Error: 出错了, 二楼
});
asyncFunc().then(res => {
console.log(res);
return Promise.resolve("二楼");
}, err => { <====== 指定失败的回调
console.log(err);
return Promise.resolve("二楼");
}).then(res => {
console.log(res);
});
/**
@params url: 请求接口地址;
@params body: 设置的请求体;
@params succ: 请求成功后的回调
@params error: 请求失败后的回调
@params maxCount: 设置请求的数量
*/
function request(url, body, succ, error, maxCount = 5) {
return fetch(url, body)
.then(res => succ(res))
.catch(err => {
if (maxCount <= 0) return error('请求超时');
return request(url, body, succ, error, --maxCount);
});
}
// 调用请求函数
request('<https://js.some.com/pc/reqCount>', { method: 'GET', headers: {} },
(res) => {
console.log(res.data);
},
(err) => {
console.log(err);
})
| --- | --- | --- |
| --- | --- | --- |
var p1 = Promise.resolve( 1 );
var p2 = Promise.resolve( p1 );
var p3 = new Promise(function(resolve, reject){
resolve(1);
});
var p4 = new Promise(function(resolve, reject){
resolve(p1);
});
// 宏任务:
// 微任务:
console.log(p1 === p2);
console.log(p1 === p3);
console.log(p1 === p4);
console.log(p3 === p4);
p4.then(function(value){
console.log('p4=' + value);
});
p2.then(function(value){
console.log('p2=' + value);
})
p1.then(function(value){
console.log('p1=' + value);
})
// 答案在下
// true
// false
// false
// false
// p2=1
// p1=1
// p4=1
Promise.resolve(...)可以接收一个值或者是一个Promise对象作为参数。
当参数是普通值时,它返回一个resolved状态的Promise对象,对象的值就是这个参数;
当参数是一个Promise对象时,它直接返回这个Promise参数。因此,p1 === p2。
但通过new的方式创建的Promise对象都是一个新的对象,因此后面的三个比较结果都是false。
另外,为什么p4的then最先调用,但在控制台上是最后输出结果的呢?
因为p4的resolve中接收的参数是一个Promise对象p1,resolve会对p1”拆箱“,获取p1的状态和值,
但这个过程是异步的