仓库地址
题目要求
wc.exe 是一个常见的工具,它能统计文本文件的字符数、单词数和行数。这个项目要求写一个命令行程序,模仿已有wc.exe 的功能,并加以扩充,给出某程序设计语言源文件的字符数、单词数和行数。
实现一个统计程序,它能正确统计程序文件中的字符数、单词数、行数,以及还具备其他扩展功能,并能够快速地处理多个文件。
具体功能要求:
- 程序处理用户需求的模式为
wc.exe [parameter] [file_name]
- 基本功能列表:
wc.exe -c file.c //返回文件 file.c 的字符数
wc.exe -w file.c //返回文件 file.c 的词的数目
wc.exe -l file.c //返回文件 file.c 的行数
- 扩展功能:
-s 递归处理目录下符合条件的文件。
-a 返回更复杂的数据(代码行 / 空行 / 注释行)。
空行:本行全部是空格或格式控制字符,如果包括代码,则只有不超过一个可显示的字符,例如“{”。
- 代码行:本行包括多于一个字符的代码。
注释行:本行不是代码行,并且本行包括注释。一个有趣的例子是有些程序员会在单字符后面加注释:
} //注释
在这种情况下,这一行属于注释行。
- 高级功能:
-x 这个参数单独使用。如果命令行有这个参数,则程序会显示图形界面,用户可以通过界面选取单个文件,程序就会显示文件的字符数、行数等全部统计信息。
打码心路过程
- 选啥语言:java太烂,node.js也不是很会,那就回到最基础虽然也不是很会的c语言吧!
- 怎么开始:忘光了忘光了抱着教材先看一遍,参考了一下网上的代码,感觉不太懂,不管了,先打,洋洋洒洒打了一堆,运行发现有错误不会改咋办?后来热心的同学告诉我可以利用main函数的命令行参数来写,于是百度了一下,感觉还行,于是把之前写的都删了,蹭蹭地把基础功能写好啦!
扩展功能:以为写完基础功能应该可以慢慢写扩展,觉得第一个扩展有点难,锁定第二个!然后百度百度,交流交流,后面有了思路,然后一直写,一直不行,换个思路,一直写,还是不行,感觉写着写着乱了,就撤销工作区修改几次,然后一直到目前,测试结果不尽如人意吖。
关键代码及思路
- main函数:利用getopt函数来分析命令行参数,当循环解析直到全部选项被解析出来则停止,并且规定选项字符串后必须带有参数。用switch语句,匹配到对应的选项则输出相应的答案。
int main(int argc, char *argv[]) {
char choice;
count(argv[2]);
otherCount(argv[2]);
while((choice = getopt(argc, argv, "c:w:l:a:")) != -1) {//选项字符串后必须带有参数
switch(choice){
case 'c':
printf("文件的字符数为:%d\n",charNum);
break;
case 'w':
printf("文件的单词数为:%d\n",wordNum);
break;
case 'l':
printf("文件的行数为:%d\n",lineNum);
break;
case 'a':
printf("文件的空行数为:%d\n文件的代码行数为:%d\n",spaceLineNum,codeLineNum);
break;
}
}
return 0;
}
- 基础功能count函数:思路是读取一个字符,然后通过条件计算出字符数,单词数,行数
-
- 计算字符数
ch=fgetc(fp); //从文件读取一个字符,暂存在变量ch中
if((ch >= 'A'&&ch <= 'z')||ispunct(ch)){//如果是字母或者标点符号,字符数加一
charNum++;
}
-
- 计算单词数
if(!(ch >= 'A'&&ch <= 'z')){ //如果是非字母字符,让flag为0
flag = 0;
}else if(flag == 0){ //如果是字母字符且flag原值为0,flag置1,单词数加一
flag = 1;
wordNum++;
}
-
- 计算行数
if(ch=='\n'&&charNum!=0){//计算行数
lineNum++;
}else if(charNum==0){//当空文件行数为0
lineNum=0;
}
- 扩展功能:虽然测试结果不好但还是贴一下吧
-
- 空行:
- 思路1:计算两个换行符之间的字符数量,如果为0则空行加一
//获取空行
if(ch == '\n'&&lineFlag == 0){
lineFlag = 1;
}else if(ch != '\n'&&lineFlag == 1){
num++;
}else if(ch == '\n'&&lineFlag == 1 &&num == 0){
lineFlag = 0;
spaceLineNum++;
}
- 思路2:获取空行之间的字符,把他们放进字符数组,通过判断数组长度来判断空行
//获取空行
if(ch == '\n'&&lineFlag == 0){
lineFlag = 1;
}else if(ch != '\n'&&lineFlag == 1){
charInLineArry[i] = ch;
i++;
}else if(ch == '\n'&&lineFlag == 1&&strlen(charInLineArry)<2){
spaceLineNum++;
lineFlag = 0 ;
}
- 思路3(最终源代码为这个):通过fgets获取文件的一行,计算一行中的非空字符数,通过判断字符数量来判断空行
while(!feof(fp))
{
fgets(st,200,fp);
int len=strlen(st);
int i=0;
for (i=0;i<len;i++)
{
if (st[i]!=' ' && st[i]!='\n' && st[i]!='\t')
num++;//一行中的其他字符数
}
if(num<1){
spaceLineNum++;
} else{
codeLineNum++;
}
}
-
- 注释行:
- 思路(测试不过就删了):遇到//则标记,接下来遇到换行符,注释行加一,遇到/标记,遇到换行符,注释行加一,最后遇到/注释行加一
//获取注释行
if(ch == '//'&&annotateLineFlag==0){
annotateLineFlag = 1;
}else if(ch == '\n'&&annotateLineFlag==1){
annotateLineNum++;
annotateLineFlag = 0;
}
if(ch == '/*'&&annotateLineFlag == 0){
annotateLineFlag = 1;
}else if(ch == '\n'&&annotateLineFlag == 1){
annotateLineNum++;
}else if(ch == '*/'&&annotateLineNum == 1){
annotateLineNum++;
annotateLineFlag = 0;
}
-
- 代码行:通过计算字符的,除了空行和注释行就是代码行。下面是通过计算一行,如果一行字符数大于一,则空行加一
while(!feof(fp))
{
fgets(st,200,fp);
int len=strlen(st);
int i=0;
for (i=0;i<len;i++)
{
if (st[i]!=' ' && st[i]!='\n' && st[i]!='\t')
num++;//一行中的其他字符数
}
if(num<1){
spaceLineNum++;//空行加一
} else{
codeLineNum++;//代码行加一
}
}
测试结果:
- 一个字符:
- 一个单词
一行
实例文件
psp
PSP2.1 | Personal Software Process Stages | 预估耗时(分钟) | 实际耗时(分钟) |
---|---|---|---|
Planning | 计划 | 24*60 | 10*60 |
· Estimate | · 估计这个任务需要多少时间 | 24*60 | 10*60 |
Development | 开发 | 12*60 | 8*60 |
· Analysis | · 需求分析 (包括学习新技术) | 60*2 | 2*60 |
· Design Spec | · 生成设计文档 | 60 | 30 |
· Design Review | · 设计复审 (和同事审核设计文档) | 30 | 30 |
· Coding Standard | · 代码规范 (为目前的开发制定合适的规范) | 30 | 30 |
· Design | · 具体设计 | 30 | 30 |
· Coding | · 具体编码 | 10*60 | 7*60 |
· Code Review | · 代码复审 | 30 | 30 |
· Test | · 测试(自我测试,修改代码,提交修改) | 2*60 | 2*60 |
Reporting | 报告 | 60 | 60 |
· Test Report | · 测试报告 | 60 | 60 |
· Size Measurement | · 计算工作量 | 30 | 30 |
· Postmortem & Process Improvement Plan | · 事后总结, 并提出过程改进计划 | 30 | 30 |
合计 |
思考与收获
- 我有点笨(捂脸)所以要好好学习
- 不会调试,所以很多代码一错就换思路而不是进一步确定哪个地方错,要学会调试
- 要先整理思路,再用代码实现
- c语言还要进一步学习
- 通过psp可以整理自己整个工程的时间耗费,是一个不错的方法