网站首页 > 基础教程 正文
我们需要用到2个模块,操作文件的模块fs和路径处理模块path。那么,让我们先引入它们。
const fs = require('fs')
const path = require('path')
然后,我们编写统计函数,我们命名为stat。我们要统计的是源代码,所以需要排除一些文件或目录,具体需要排除哪些文件或目录呢?node_modules肯定是要排除的,我们还需要提供一种方式来指定需要排除的目录或文件,比如源码目录中存在一个jquery.min.js文件,这是第三方库,肯定不能当作源码统计。说到根据文件类型统计,那么我们需要一个扩展名数组。
stat函数的入参定义如下:excludes,要排除的目录或文件;extnames,要统计的扩展名数组。
/** @param {{ excludes: string[], extnames: string[] }} param0 */
function stat ({ excludes, extnames }) {
}
以下代码全部在stat函数内部编写。我们先定义一个默认排除的目录数组defaultExcludes,然后对入参excludes进行重新赋值,合并默认排除的目录数组并调用数组的map方法,转化为新数组。如果路径以 / 开头,相对当前目录,转化为绝对路径,否则将 / 替换为当前操作系统目录分隔符。然后,我们定义了一个函数isExcluded,用于判断该目录或文件是否需要排除。
const defaultExcludes = ['/.git', '/.vscode', '/dist', '/node_modules', '/public', __filename]
excludes = [...(excludes || []), ...defaultExcludes].map(_ => {
return _.startsWith('/')
? path.resolve(__dirname, _.slice(1))
: _.replace(/\//gm, path.sep)
})
/** @param {string} dir */
const isExcluded = dir => excludes.some(_ => dir.includes(_))
接下来,我们定义更新计数函数。该函数接收2个参数,参数类型参见文档注释。
/**
* @param {Object<string, number>} map
* @param {string} file
*/
const updateCount = (map, file) => {
const extname = file.split('.').pop()
const isStat = extnames.includes(extname) && !isExcluded(file)
if (isStat) {
// 同步读取文件
// 转化为utf8编码字符串
// 去掉两端空白符
// 以换行符分割字符串转化为数组
// 获取数组长度就是文件行数
const lineCount = fs.readFileSync(file).toString('utf8').trim().split('\n').length
return {
...map,
all: (map.all || 0) + lineCount,
[extname]: (map[extname] || 0) + lineCount
}
}
return map
}
然后,我们定义合并统计map函数,该函数接收2个如上文档注释定义的类型作为参数,将2个map的统计值进行合并。并返回合并后的map。
const mergeCount = (o1, o2) => {
return Object.keys(o2).reduce((t, k) => {
return {
...t,
[k]: (t[k] || 0) + o2[k]
}
}, o1)
}
最后,我们定义读取文件函数reader,该函数接收父目录作为参数,调用fs.readdirSync方法,返回一个子目录字符串构成的数组,调用数组的reduce方法进行聚合统计。最后,我们以该脚本所在目录开始调用读取函数进行递归读取。
const reader = parentDir => {
return fs.readdirSync(parentDir).reduce((t, d) => {
const dir = path.resolve(parentDir, d)
const isDir = fs.statSync(dir).isDirectory()
return isDir
? isExcluded(dir)
? t
: mergeCount(t, reader(dir))
: updateCount(t, dir)
}, {})
}
return reader(__dirname)
记住,以上代码,除了模块导入部分,全部写在stat函数体中。现在,我们看一下stat函数的使用例子。
const statMap = stat({
extnames: ['js', 'css', 'scss', 'vue']
})
console.log(statMap) // 输出 { all: 2700, js: 831, vue: 1832, scss: 37 }
const statMap = stat({
extnames: ['js', 'css', 'scss', 'vue'],
excludes: ['auth']
})
console.log(statMap) // 输出 { all: 1968, js: 801, vue: 1130, scss: 37 }
const statMap = stat({
extnames: ['js', 'vue'],
excludes: ['auth', 'app/main.js']
})
console.log(statMap) // 输出 { all: 1904, js: 774, vue: 1130 }
是不是很简单?大家理解了吗?有什么不明白之处,欢迎评论区见!感谢阅读!
猜你喜欢
- 2024-11-07 ASP.NET Core 知识速递 - Day 6:每天进步一点
- 2024-11-07 谈谈springboot 获取前端json数据几种方法
- 2024-11-07 GitHub爆火!银四巨作:拼多多/蚂蚁/百度面经分享
- 2024-11-07 RxJs 介绍 rxjs教程
- 2024-11-07 盘点Django展示可视化图表多种方式(建议收藏)
- 2024-11-07 界面美观且友好的前端Bootstrap 3管理模板
- 2024-11-07 可视化:前端数据可视化插件大盘点 图表/图谱/地图/关系图
- 2024-11-07 JQuery 获取多个select标签option的text内容
- 2024-11-07 高清地图产品输出改进过程记录 高精度地图数据格式
- 2024-11-07 HTML页面通过高德地图JS API实现高德地图显示 经纬度功能
- 最近发表
- 标签列表
-
- gitpush (61)
- pythonif (68)
- location.href (57)
- tail-f (57)
- pythonifelse (59)
- deletesql (62)
- c++模板 (62)
- css3动画 (57)
- c#event (59)
- linuxgzip (68)
- 字符串连接 (73)
- nginx配置文件详解 (61)
- html标签 (69)
- c++初始化列表 (64)
- exec命令 (59)
- canvasfilltext (58)
- mysqlinnodbmyisam区别 (63)
- arraylistadd (66)
- node教程 (59)
- console.table (62)
- c++time_t (58)
- phpcookie (58)
- mysqldatesub函数 (63)
- window10java环境变量设置 (66)
- c++虚函数和纯虚函数的区别 (66)