编辑
NodeJS版本10.12.0
增加了对两者的本机支持,mkdir
并mkdirSync
使用以下命令递归创建目录recursive: true
以下选项:
fs.mkdirSync(targetDir, { recursive: true });
如果愿意fs Promises API
,可以写
fs.promises.mkdir(targetDir, { recursive: true });
原始答案
如果目录不存在,则递归创建目录!(零依赖)
const fs = require('fs');
const path = require('path');
function mkDirByPathSync(targetDir, { isRelativeToScript = false } = {}) {
const sep = path.sep;
const initDir = path.isAbsolute(targetDir) ? sep : '';
const baseDir = isRelativeToScript ? __dirname : '.';
return targetDir.split(sep).reduce((parentDir, childDir) => {
const curDir = path.resolve(baseDir, parentDir, childDir);
try {
fs.mkdirSync(curDir);
} catch (err) {
if (err.code === 'EEXIST') { // curDir already exists!
return curDir;
}
// To avoid `EISDIR` error on Mac and `EACCES`-->`ENOENT` and `EPERM` on Windows.
if (err.code === 'ENOENT') { // Throw the original parentDir error on curDir `ENOENT` failure.
throw new Error(`EACCES: permission denied, mkdir '${parentDir}'`);
}
const caughtErr = ['EACCES', 'EPERM', 'EISDIR'].indexOf(err.code) > -1;
if (!caughtErr || caughtErr && curDir === path.resolve(targetDir)) {
throw err; // Throw if it's just the last created dir.
}
}
return curDir;
}, initDir);
}
用法
// Default, make directories relative to current working directory.
mkDirByPathSync('path/to/dir');
// Make directories relative to the current script.
mkDirByPathSync('path/to/dir', {isRelativeToScript: true});
// Make directories with an absolute path.
mkDirByPathSync('/path/to/dir');
演示版
试试吧!
说明
- [更新]此解决方案可处理特定
EISDIR
于平台的错误,例如Mac EPERM
和EACCES
用于Windows。感谢@PediT。,@ JohnQ,@ deed02392,@ robyoder和@Almenon的所有报告意见。
- 该解决方案同时处理相对和绝对路径路径。感谢@john评论。
- 对于相对路径,将在当前工作目录中创建(解析)目标目录。要相对于当前脚本目录解析它们,请传递
{isRelativeToScript: true}
。
- 使用
path.sep
和path.resolve()
,而不仅仅是/
连接,以避免跨平台问题。
- 使用
fs.mkdirSync
和处理try/catch
如果抛出的错误以处理竞争条件:另一个进程可能会在对fs.existsSync()
和的调用之间添加文件fs.mkdirSync()
并导致异常。
- 实现此目标的另一种方法是检查文件是否存在,然后创建它,即
if (!fs.existsSync(curDir) fs.mkdirSync(curDir);
。但这是一种反模式,使代码容易受到竞争条件的影响。感谢@GershomMaes评论目录是否存在。
- 需要Node v6及更高版本才能支持解构。(如果您在使用旧版Node实施此解决方案时遇到问题,请给我留言)