在nodejs
封装了许多的异步的回调函数,例如
在官方示例中,这个函数是这样使用:
fs.readFile("1.txt", (err,data) => {
if(!err) {
console.log(data );
}
})
可是这种使用的方式,往往无法满足大多数的使用场景,所以可能会有新手朋友和我一样,写成这样
let obj = {};
fs.readFile("C:\\Users\\yeyeyeping\\Desktop\\1", (err,data) => {
if(!err) {
obj.data = data;
}
})
这样的话,obj的值是不能被拿到的,原因可以从以下图中找到
可以看到,nodejs
在解释代码的时候,首先会扫描一遍,异步任务会按照顺序进入队列尾部中,非异步的直接放在队首先执行,然后才会按照队列顺序解释异步代码,所以在同步代码运行的时候,我们是没办法看到回调函数中的值更新到全局变量中,这是一件很痛苦的事,比如js
,实际上很难有一行一行读取文本且可以任意位置随意停下的库。
下面是关于我对于这个猜想的验证
//首先执行
let obj = {};
//异步io进入队列尾部
fs.readFile("C:\\Users\\yeyeyeping\\Desktop\\1", (err,data) => {
if(!err) {
obj.data = data;
console.log("inside the callback, data is ", obj.data.toString());
}
})
//第二个执行
console.log( obj.data );
//在开一个异步任务,进入队列尾部,处理到这个任务时,上面一个异步任务已经完成,看看有没有更新到obj
fs.readFile("C:\\Users\\yeyeyeping\\Desktop\\3", (err,data) => {
if(!err) {
console.log( obj.data.toString());
}
})
输出
那有没有解决方案呢,当然有,这些异步的函数很多都可以改成同步例如这样
const fs = require("fs");
async function test(){
var v = await new Promise((resolve, reject) => {
fs.readFile("C:\\Users\\yeyeyeping\\Desktop\\1", (err,data) => {
if (err) reject(err);
resolve(data.toString());
})
})
console.log(v);
}
test()