Github项目地址:https://github.com/bravedreamer/wordCount/tree/master/wc

一、题目描述:

Word Count
1. 实现一个简单而完整的软件工具(源程序特征统计程序)。
2. 进行单元测试、回归测试、效能测试,在实现上述程序的过程中使用相关的工具。
3. 进行个人软件过程(PSP)的实践,逐步记录自己在每个软件工程环节花费的时间。

二、项目要求:

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 文件的代码行数、空行数、注释行数。

三、解题思路:

 首先考虑到用C语言读取文件的方法去实现。

  返回文件 file.c 的字符数 :使用fgetc() 函数从所指向的输入文件中一个一个地读取字符,然后累计。

  返回文件 file.c 的词的数目:使用fgetc() 函数从所指向的输入文件有条件地读取a(A)-->z(Z)组成的单词,再累计单词数量。

  返回文件 file.c 的行数:使用fgetc() 函数在所指向的输入文件中遇到换行符“\n”后累计加1 。

  返回代码行 、 注释行:当fgetc() 函数获取的字符遇到“//”和“/*”就开始累计注释行数量,如果是“//”遇到换行符“\n”或文件结束符则停止累计,如果是“/*”则遇到“*”停          止累计。代码行则是运用之前累计行数的函数返回的结果减去注释行即可得到。

四、设计实现过程:

五、遇到的困难及解决方法:

  • 困难描述:对C语言读写文件有关操作不是很熟悉。
  • 做过哪些尝试:上网查阅c语言相关资料
  • 是否解决:已解决
  • 有何收获:对c语言的了解更加深入,遇到知识盲区要及时查阅资料解决。 

六、关键代码:

main函数主要代码:

int main(int argc, char *argv[]) {
char command[];
char tag[];//保存文件路径
  int charNum,wordNum,lineNum;//接收函数返回的字符数、单词数、行数
char mainCmd[][]={"-c","-w","-l","-a" };//保存指令 while(){
printf("功能:\n");
printf(" 1:输入 -c 文件路径 返回文件的字符数\n");
printf(" 2;输入 -w 文件路径 返回文件的词的数目\n");
printf(" 3:输入 -l 文件路径 返回文件的行数\n");
printf(" 4:输入 -a 文件路径 返回文件的代码行数和注释行数\n") int characterCount(char *file);
int WordCount(char *file);
int rowCount(char *file);
int complexCount(char *file); printf("请输入指令:wc.exe");
scanf("%s",&command); printf("请输入文件路径:");
scanf("%s",&tag); if(strcmpi(command,mainCmd[])==){
charNum=characterCount(tag);
printf("字符数:%d",charNum);
}else if(strcmpi(command,mainCmd[])==){
wordNum=WordCount(tag);
printf("词汇数:%d",wordNum);
}else if(strcmpi(command,mainCmd[])==){
lineNum=rowCount(tag);
printf("行数:%d",lineNum);
}else if(strcmpi(command,mainCmd[])==){
complexCount(tag);
}else{
printf("指令错误!");
} printf("\n\n");
}
return ;
}
字符数统计函数主要代码:
int characterCount(char *file){//字符数统计函数
FILE *myfile=NULL;
int charCount=;//用于累计读到的字符数
myfile=fopen(file,"r");
if(myfile==NULL){
printf("文件不存在!\n");
exit(-);
}
char mychar;
mychar = fgetc(myfile);
while(mychar!=EOF){
mychar = fgetc(myfile);
charCount++;
}
fclose(myfile);
return charCount;
}
单词数统计函数代码:
int WordCount(char *file){//单词数统计函数
FILE *myfile=NULL;
int wordcount=;//用于累计读到的单词数量
myfile=fopen(file,"r");
if(myfile==NULL){
printf("文件不存在!\n");
exit(-);
}
char mychar;
mychar = fgetc(myfile);
while(mychar!=EOF){
if((mychar>='a'&&mychar<='z')||(mychar>='A'&&mychar<='Z')){//判断是否为单词
while((mychar>='a'&&mychar<='z')||(mychar>='A'&&mychar<='Z')){
mychar=fgetc(myfile);
}
wordcount++;
}
mychar=fgetc(myfile);
}
fclose(myfile);
return wordcount;
}
行数统计函数:
int rowCount(char *file){//行数统计函数
FILE *myfile=NULL;
int linecount=;//用于累计读到的行数量·
myfile=fopen(file,"r");
if(myfile==NULL){
printf("文件不存在!\n");
exit(-);
}
char mychar;
mychar = fgetc(myfile);
while(mychar!=EOF){
if(mychar=='\n'){
linecount++;
mychar = fgetc(myfile);
}
else{
mychar = fgetc(myfile);
}
}
fclose(myfile);
return linecount;
}
统计注释行,代码行数量:
int complexCount(char *file){//统计注释行,代码行数量
FILE *myfile=NULL;
int noteLine=;//注释行
int codeLine=;//代码行 myfile=fopen(file,"r");
if(myfile==NULL){
printf("文件不存在!\n");
exit(-);
} char mychar;
mychar= fgetc(myfile); while(mychar!=EOF){
if(mychar=='\''||mychar=='\"'){//防止读取到双引号里面的内容
mychar=fgetc(myfile);
while(mychar=='\"'||mychar=='\''){
mychar=fgetc(myfile);
}
} if(mychar=='/'){//注释行计算
mychar=fgetc(myfile);
if(mychar=='/'||mychar=='*'){
if(mychar=='/'){
mychar=fgetc(myfile);
while(mychar=="\n"||mychar!=EOF){
mychar=fgetc(myfile);
}
noteLine++;
mychar=fgetc(myfile);
}
if(mychar=='*'){
mychar=fgetc(myfile);
while(mychar=='*'){
if(mychar=="\n"||mychar=='*')
noteLine++;
mychar=fgetc(myfile);
}
}
}
}else{
mychar=fgetc(myfile);
} } int rowCount(char *file);
codeLine=rowCount(file);//代码行计算
codeLine-=noteLine; fclose(myfile);
printf("注释行:%d, 代码行:%d",noteLine,codeLine);
return ; }

七、测试:

空文件测试:

一个字符的文件测试:

  

一个词的文件测试:

源文件测试:

错误处理:

附:

PSP表格:

PSP2.1

Personal Software Process Stages

预估耗时(分钟)

实际耗时(分钟)

Planning

计划

30

60

· Estimate

· 估计这个任务需要多少时间

30

60

Development

开发

2*24*60

3*24*60

· Analysis

· 需求分析 (包括学习新技术)

4*60

5*60

· Design Spec

· 生成设计文档

20

30

· Design Review

· 设计复审 (和同事审核设计文档)

20

30

· Coding Standard

· 代码规范 (为目前的开发制定合适的规范)

50

60

· Design

· 具体设计

2*60

3*60

· Coding

· 具体编码

24*60

2*24*60

· Code Review

· 代码复审

4*60

6*60

· Test

· 测试(自我测试,修改代码,提交修改)

60

2*60

Reporting

报告

2*60

3*60

· Test Report

· 测试报告

60

2*60

· Size Measurement

· 计算工作量

30

40

· Postmortem & Process Improvement Plan

· 事后总结, 并提出过程改进计划

30

20

合计

3*24*60

4*24*60

总结:

    预估时间与实际时间有较大误差,以后得及时把握开发时间,避免出现计划赶不上变化的情况。

    遇到问题需要及时查阅资料,避免把时间浪费在不必要的尝试上。

    部分功能暂未实现,有些许遗憾,编程能力有待提高。

count.exe 个人项目的更多相关文章

  1. wc.exe个人项目

    1.GitHub项目 https://github.com/Littlehui3/wc 2.用时表格 PSP2.1 任务内容 计划完成需要的时间(min) 实际完成需要的时间(min) Plannin ...

  2. django创建项目

    django创建项目 安装django pip install django==1.9 Note: C:\Python34\Scripts\pip.exe 创建项目 django-admin star ...

  3. 无法从命令行或调试器启动服务,必须首先安装Windows服务(使用installutil.exe),然后用ServerExplorer、Windows服务器管理工具或NET START命令启动它

    无法从命令行或调试器启动服务,必须首先安装Windows服务(使用installutil.exe),然后用ServerExplorer.Windows服务器管理工具或NET START命令启动它 1. ...

  4. 如何用VS调试不属于解决方案的EXE和DLL程序

    如果你手里有一个现成的EXE, 以及EXE相关联PDB文件, 还有相关联的CPP文件和H文件. 你如何用VS调试? (当然你可以选择WinDbg.不过这里就讨论VS) 你或许想问我干嘛不从一开始就用V ...

  5. Usage、Usage Minimum和Usage Maximum项目详解

    (1)一个产生多个数据域(Report Count>1)的主项目之前有一个以上的[用途]时,每个[用途]与一个数据域依次对应,如果数据域个数(Report Count)超过[用途]的个数,则剩余 ...

  6. `vue-router`的`History`模式下的项目发布

    背景 脚手架版本号:vue cli 3.x 项目路由:vue-router的History模式 原理:url路由处理逻辑从后端转移到前端. 参考:https://developer.mozilla.o ...

  7. 软工作业1—java实现wc.exe

    github项目地址 https://github.com/liyizhu/wc.exe WC 项目要求 基本功能列表: wc.exe -c file.c     //返回文件 file.c 的字符数 ...

  8. 用c语言基本实现wc.exe功能

    网址:https://github.com/3216005214/wc.exe wc项目要求 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要求写一个命令行程序,模仿 ...

  9. django新建项目

    命令: django-admin startproject 项目名 pycharm创建: 1. File --> New project --> 左侧选Django --> 右侧填项 ...

随机推荐

  1. PHPExcel将Excel数据导入数据库

    <?php //PHPExcel读取导入Excel数据到数据库(2003,2007通用)使用方法: //先用excel2array()方法将excel表中的数据存储到数组,在从遍历二维数组将数据 ...

  2. MyBatis动态插入的实现

    mybatis通过定义前缀后缀和分割字符来拼接sql语句,实现动态插入的功能 <insert id="addNewsTypeByNewsId"> insert into ...

  3. Django学习路12_objects 方法(all,filter,exclude,order by,values)

    Person.objects.all() 获取全部数据 def get_persons(request): persons = Person.objects.all() # 获取全部数据 contex ...

  4. 转载——完整的ASCII码表

    完整的ASCII码表,转载自下面的博主: http://www.cnblogs.com/xmxu/archive/2012/07/10/2584032.html

  5. luogu P3412 仓鼠找sugar II 期望 树形dp

    LINK:仓鼠找sugar II 以前做过类似的期望题目 加上最后的树形dp不算太难 还是可以推出来的. 容易发现 当固定起点和终点的时候 可以先固定根 这样就不用分到底是正着走还是倒着走了. 1为根 ...

  6. 剑指 Offer 57 - II. 和为s的连续正数序列

    本题 题目链接 题目描述 我的题解 方法三双100%, 方法一 适合范围广 方法一:双指针(也叫 滑动窗口) 思路分析 用两个指针i和表示当前枚举到的以i为起点,j为终点的区间,sum表示[i,j]的 ...

  7. ios 富文本 加颜色 删除线

    UILabel *valueL = [JAppViewTools getLabel:CGRectMake(JFWidth(15), CGRectGetMaxY(proName.frame)+JFWid ...

  8. Android SQLite轻量级数据库的删除和查找操作

    今天主要是补充昨天的内容,本打算合成一章的,但是毕竟一天一天的内容写习惯了. 就这样继续昨天的,昨天只讲了创建以及增加和查询, 其实用法都差不多,今天学长也是在原有的基础上写的,还顺便融合了Share ...

  9. 2020-07-16:如何获得一个链表的倒数第n个元素?

    福哥答案2020-07-16: 1.快慢指针.快指针先走n步,然后快慢指针同时走,直到快指针走到尾.2.两次遍历.第一次遍历获取链表长度,然后计算出序号,然后遍历获取序号下的元素.3.数组保存.遍历一 ...

  10. 编译原理——求解First,Follow,Firstvt和Lastvt集合

    转载地址 http://dongtq2010.blog.163.com/blog/static/1750224812011520113332714/ 学编译原理的时候,印象最深的莫过于这四个集合了,而 ...