Github项目地址:https://github.com/xiaobaot/wordcount-wc/tree/master
WC 项目要求
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 返回更复杂的数据(代码行 / 空行 / 注释行)。(实现)空行:本行全部是空格或格式控制字符,如果包括代码,则只有不超过一个可显示的字符,例如“{”。代码行:本行包括多于一个字符的代码。注释行:本行不是代码行,并且本行包括注释。一个有趣的例子是有些程序员会在单字符后面加注释: } //注释在这种情况下,这一行属于注释行。
[file_name]: 文件或目录名,可以处理一般通配符。
高级功能:
-x 参数。这个参数单独使用。如果命令行有这个参数,则程序会显示图形界面,用户可以通过界面选取单个文件,程序就会显示文件的字符数、行数等全部统计信息。需求举例: wc.exe -s -a *.c返回当前目录及子目录中所有*.c 文件的代码行数、空行数、注释行数。PSP开发耗时
PSP2.1 | Personal Software Process Stages | 预估耗时(分钟) | 实际耗时(分钟) |
Planning | 计划 | 60 | 100 |
· Estimate | · 估计这个任务需要多少时间 | 100 | 120 |
Development | 开发 | 600 | 660 |
· Analysis | · 需求分析 (包括学习新技术) | 120 | 150 |
· Design Spec | · 生成设计文档 | 40 | 50 |
· Design Review | · 设计复审 (和同事审核设计文档) | 40 | 50 |
· Coding Standard | · 代码规范 (为目前的开发制定合适的规范) | 30 | 30 |
· Design | · 具体设计 | 60 | 60 |
· Coding | · 具体编码 | 120 | 120 |
· Code Review | · 代码复审 | 60 | 90 |
· Test | · 测试(自我测试,修改代码,提交修改) | 60 | 60 |
Reporting | 报告 | 90 | 90 |
· Test Report | · 测试报告 | 60 | 60 |
· Size Measurement | · 计算工作量 | 30 | 30 |
· Postmortem & Process Improvement Plan | · 事后总结, 并提出过程改进计划 | 30 | 30 |
合计 |
| 1500 | 1700 |
解题思路及代码说明
1.主函数
int main(int argc, char* argv[]) { FILE *fp; while(1) { if((fp=fopen(argv[2],"r"))==NULL) { printf("FileNull\n\n\n"); scanf("%s%s%s",argv[0],argv[1],argv[2]); continue; } else if(!strcmp(argv[1],"-w")) countw(argv[2]); else if(!strcmp(argv[1],"-c")) countc(argv[2]); else if(!strcmp(argv[1],"-l")) countl(argv[2]); else if(!strcmp(argv[1],"-a")) { count_blankline(argv[2]); count_noteline(argv[2]); count_codeline(argv[2]); } else if(!strcmp(argv[1],"-s")) { searchfile(); } else printf("NullPoint\n"); printf("\n\n"); scanf("%s%s%s",argv[0],argv[1],argv[2]); } return 0; }
思路:输入指令分成三组,根据第二组的指令对第三组的指令文件名执行相应的操作。
基本功能
-c 返回文件的字符数
int countc(char *file) //返回文件的字符数 { FILE *f; f = fopen(file, "r"); char a; if(NULL==(f=fopen(file,"r"))) { printf("file is NULL"); } else while (!feof(f)) { a=fgetc(f); if (a != ' '&&a != '\n'&&a != '\t') cchar++; } fclose(f);printf("charnum:%d ",cchar); }
思路:统计非空非换行符非制表符的字符
-w 返回文件词的数目
int countw(char *file)//返回文件词的数目 { FILE *f; f=fopen(file,"r"); char ch; if(NULL==(f=fopen(file,"r"))) { printf("file is NULL"); } else while(!feof(f)) { ch=fgetc(f); if((ch >= 'a'&&ch <= 'z')||(ch>='A'&&ch<='Z')||ch=='_') aword=1; else if (aword) { cword++; aword=0; } } fclose(f); printf("wordnum:%d ",cword); }
-l 返回文件的行数
int countl(char *file) //返回文件的行数{ FILE *f; f = fopen(file, "r"); int cline = 0; char a; if(NULL==(f=fopen(file,"r"))) {printf("file is NULL");} else while(!feof(f)) { a=fgetc(f); if(a=='\n'||a=='\t') cline++; } fclose(f);printf("Linenum:%d ",cline); return 1; }
3.扩展功能
-a 统计空行,代码行,注释行
int count_blankline(char *file) //返回文件的空行数
{ FILE *f;int b_num = 0;int ch_num = 0;char ch;f = fopen(file, "r");if(NULL==(f=fopen(file,"r"))){ printf("file is NULL");}elsewhile (!feof(f)){ ch= fgetc(f);if (ch=='\n'){ if (ch_num<= 1)b_num++;ch_num = 0;}else if (ch!=' '&&ch!='\t'&&ch!='}')ch_num++;else if(ch=='}')b_num++;}fclose(f);printf("blankline:%d ",b_num);}int count_noteline(char *file) //返回文件的注释行数 { FILE *f;int ch_num = 0;int note_num=0;char ch;f=fopen(file, "r");if(NULL==(f=fopen(file,"r"))){ printf("file is NULL");}elsewhile (!feof(f)){ ch= fgetc(f);if(ch=='\n'){if(ch_num==2) note_num++; ch_num=0;}else if(ch=='/') ch_num++;else if(ch_num==1){if(ch=='/') ch_num++;} } fclose(f);printf("noteline:%d ",note_num);} int count_codeline(char *file)//返回文件的代码行数 { int ch_num = 0;int code_num=0;FILE *f;int tag=0; int flag=0;char a; f = fopen(file, "r");if(NULL==(f=fopen(file,"r"))){printf("file is NULL");}elsewhile (!feof(f)){ a=fgetc(f);if(flag==2) { flag=0;tag++;}else{ if(a=='\n'&&ch_num>1){code_num++; ch_num=0; }else if(a != ' '&&a != '\n'&&a != '\t'&&a!='/') { ch_num++;}else if(a=='/'){ flag++;}}}fclose(f); printf("codeline:%d ",code_num-tag); }思路:统计注释行数统计//的出现次数,统计空行时统计一行的字符不超过一个字。统计代码行时排除空行,注释行。
-s 寻找文件夹中的txt文件
int searchfile(void) //寻找文件夹中的txt文件{ struct _finddata_t filefind; long handle; int t=0; if( (handle=_findfirst( "H:\\wordcount\\test\\*txt", &filefind)) == -1L ) { printf( "没找到txt文件\n"); } else do{ t++; printf("找到文件:%s\n", filefind.name); }while (_findnext(handle,&filefind)==0); _findclose(handle); printf("txt文件数量:%d\n",t); return 0; }
思路:查找文件夹下的txt文件。
测试运行
测试文件:
运行结果:
项目小结
①重新复习了C语言,同时也学习到了不少关于C++的知识;
②算法的性能还有待提高和优化,对问题的考虑可以更全面的;
③凡事预则立,不预则废,以后要注意时间的安排度问题;
④这个项目仍然有不少的bug,以后可以继续维护下去;
⑤知道了项目开发的流程,也学习了与他人协作的能力。
设计流程