为什么我不能将文件推送到数组中?

问题描述 投票:1回答:1

我具有递归查找文件的功能。如果指定类型D,我只希望将文件夹添加到阵列。如果键入F,则仅文件。它在文件搜索中正常工作。但是如果输入D,则尽管可以输出到控制台,但是什么也不能添加。为什么我不能添加到阵列中以及如何修复它

const type = T or D

const walk = (dir, done) => {
    let results = [];
    return new Promise((resolve, reject) => {
        fs.readdir(dir, (err, list) => {
            if (err) return done(err);
            let pending = list.length;
            if (!pending) return done(null, results);
            list.forEach((file) => {
                file = path.join(dir, file);
                fs.stat(file, function (err, stat) {
                    if (stat && stat.isDirectory()) {
                        if (type && type === 'D') {
                            console.log(file)
                            results.push(file);
                        }
                        walk(file, (err, res) => {
                            results.push(...res);
                            if (!--pending) done(null, results);
                        });
                    } else {
                        if (type === 'F') {
                            results.push(file);
                            if (!--pending) done(null, results);
                        }
                    }
                });
            });
        });
    })
};

walk(baseDir, (err, results) => {
    if (err) throw err;
    console.log(results);
});
javascript node.js recursion filesystems
1个回答
0
投票

[typeD时,您目前仅在pending块内递减if (stat && stat.isDirectory()),但是由于以下原因,pending的数量还取决于目录中files的数量到let pending = list.length;

解决此问题的一种方法是在pending内减小else,无论如何:

} else {
    if (type === 'F') {
        results.push(file);
    }
    if (!--pending) done(null, results);
}

或者,为了更简洁并避免某些回调地狱,请使用async函数和await,然后可以使用Promise.all而不是手动检查索引(这很繁琐并且容易出错) 。这也使该函数正确返回Promise(未传递done时):

const walk = async (dir, done) => {
    try {
        const list = await readdir(dir);
        const resultsArr = await Promise.all(list.map(async (fileName) => {
            const filePath = path.join(dir, fileName);
            const stats = await stat(filePath);
            if (stats.isDirectory()) {
                if (type === 'D') {
                    return [filePath, ...await walk(filePath)];
                }
                return walk(filePath);
            } else if (!stats.isDirectory() && type === 'F') {
                return filePath;
            }
        }));
        const flatResults = resultsArr.flat().filter(Boolean);
        if (done) {
            done(null, flatResults);
        }
        return flatResults;
    } catch (err) {
        if (done) {
            done(err);
        } else {
            throw err;
        }
    }
};
© www.soinside.com 2019 - 2024. All rights reserved.