1.GitHub项目地址

https://github.com/QiuBin666/WC

项目介绍:

题目描述

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

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

2.PSP预计时间

PSP2.1

Personal Software Process Stages

预估耗时(分钟)

实际耗时(分钟)

Planning

计划

30

· Estimate

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

30

Development

开发

180

· Analysis

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

200

· Design Spec

· 生成设计文档

30

· Design Review

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

0

· Coding Standard

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

20

· Design

· 具体设计

30

· Coding

· 具体编码

360

· Code Review

· 代码复审

20

· Test

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

120

Reporting

报告

120

· Test Report

· 测试报告

60

· Size Measurement

· 计算工作量

30

· Postmortem & Process Improvement Plan

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

60

合计

1290

3.解题思路

拿到题目后首先想到的是:要对文件进行读写操作,但这是以前的编程经历中从未涉及到的,所以首先去学习了c语言文件读写相关的知识。

学习相关知识后,就到了该如何实现功能的问题了,解决方法是划分为几个功能模块,每个模块实现一个功能。

4.设计实现过程

设计字符数统计函数void CharCount():当判定为字符便计数,非字符则不计。

设计单词数统计函数void WordCount():通过从第一个字母开始读取连续的字母。

设计行数统计函数 void LineCount():判断行数可以判断是否出现‘\n’或‘\t’来判断。
设计扩展功能统计函数int Extend():

    1、空行计算,如果一行中没有字符或者最多有一个'{'或者'}'(可能遇到‘ ’或者‘\t’)算为空行。

    2、注释行计算,通过读取第一个字符为/进入判断,读取第二个字符为*或者/实现。

    3、代码行计算,排除空行和注释行之外的即为代码行。

5.代码说明 

主函数:

 int a=;
int b=;
int c=;
int main() { //主函数
void CharCount(char filename[]);//字符数统计函数
void WordCount(char filename[]);//单词数统计函数
void LineCount(char filename[]);//行数统计函数
int Extend(char filename[]);//扩展功能统计函数
char input[], filename[];
while () {
//输入指令
printf("wc.exe - c file.c 返回文件的字符数\nwc.exe - w file.c 返回文件的词的数目\nwc.exe - l file.c 返回文件的行数\nwc.exe - a file.c 返回更复杂的数据(代码行 / 空行 / 注释行)\nwc.exe - o file.c 退出\n请输入指令:- ");
scanf("%s",&input); switch(input[]){
case 'c': //字符数
printf("请输入文件名:");//输入文件名
scanf("%s", &filename);
CharCount(filename);
printf("文件的字符数为:%d",a);
break; case 'w': //词数
printf("请输入文件名:");//输入文件名
scanf("%s", &filename);
WordCount(filename);
printf("文件的词数为:%d", b);
break; case 'l': //行数
printf("请输入文件名:");//输入文件名
scanf("%s", &filename);
LineCount(filename);
printf("文件的行数为:%d", c);
break; case 'a': //返回更复杂的数据(代码行 / 空行 / 注释行)。
printf("请输入文件名:");//输入文件名
scanf("%s", &filename);
Extend(filename);
break; case 'o': //退出
return ; default:
printf("你输入的指令有误!请重新输入");
break;
} }
system("pause");
return ;
}

各子模块功能函数:

计算字符数:

 void CharCount(char filename[]){//字符数统计函数
FILE *fp=NULL;
fp=fopen(filename,"r");
if(fp==NULL){
printf("寻找文件失败\n");
exit(-);
}
char ch;
ch = fgetc(fp);
while(ch!=EOF){
if (ch != ' '&&ch != '\n'&&ch != '\t'&&ch != ','&&ch != '.'&&ch != '!'&&ch != ';'&&ch != '=');
a++;
ch = fgetc(fp);
}
a--;
fclose(fp);
}

计算词数:

 void WordCount(char filename[]){//单词数统计函数
FILE *fp = NULL;
fp=fopen(filename,"r");
if(fp==NULL){
printf("寻找文件失败\n");
exit(-);
}
char fchar = ;
fchar = fgetc(fp);
while (fchar != EOF)
{
if ((fchar >= 'a' && fchar <= 'z') || (fchar >= 'A' && fchar <= 'Z') )//遇到字母
{
while ((fchar >= 'a' && fchar <= 'z') || (fchar >= 'A' && fchar <= 'Z'))
fchar = fgetc(fp);
b++;
fchar = fgetc(fp);
}
else {
fchar = fgetc(fp);
}
}
fclose(fp); }

计算行数:

 void LineCount(char filename[]){//行数统计函数
FILE *fp = NULL;
fp=fopen(filename,"r");
if(fp==NULL){
printf("寻找文件失败\n");
exit(-);
}
char ch;
ch = fgetc(fp);
while (ch != EOF)
{
if (ch == '\n' || ch == '\t')
c++;
ch = fgetc(fp);
}
fclose(fp);
}

扩展功能函数:

 int Extend(char filename[]) {  //扩展功能
FILE *fp = NULL;
fp = fopen(filename, "r");
if (fp == NULL) {
printf("寻找文件失败\n");
exit(-);
} char fchar = ;
int emptyline = , code = , note = ;
int l = , n = , c = ;//l为空行类型,0是无字符空行,1是1字符空行;n为注释类型,0为非注释行,1为单注释行,2为多注释行,c为代码类型 while (fchar != EOF) {
fchar = fgetc(fp);
if (l == || l == ) {//如果开始是空格或制表,开始判断是不是空行或者注释行或者代码行
if (fchar == '\n') {//回车之前没有字符,空行;1字符进入后也是没有其他字符
emptyline++;
l = ;
continue;
}
else if (fchar == '/') {//注释判断
fchar = fgetc(fp);
if (fchar == '/') {//单行注释
n = ;
while (fchar != EOF && fchar != '\n') {
fchar = fgetc(fp);
}
n = ;
note++;
continue;
}
else if (fchar == '*') {//多行注释
n = ;
continue;
}
}
else if( fchar == '}' && l == ){
l = ; //一字符空行
continue;
}
else if(fchar == ' ' || fchar == '\t'){
continue;
}
else {
c = ;
}
}
if (c == ) {//代码行
while (fchar != EOF && fchar != '\n') {
fchar = fgetc(fp);
}
l = ;//状态变回空行
code++;
continue;
}
if (n == ) {//多行注释
while (fchar != EOF && fchar != '*') {
while (fchar != EOF && fchar != '\n') {
fchar = fgetc(fp);
}
note++;
fchar = fgetc(fp);
}
fchar = fgetc(fp);
if (fchar == '/') {//多行结束
n = ;
}
continue;
}
}
printf("\n共有空行数: %d \n代码行数: %d \n注释行数: %d\n", emptyline, code, note);
fclose(fp);
}

6.测试运行

基本功能测试用例

测试-c功能:

测试-w功能:

测试-l功能:

扩展功能测试用例

测试-a功能:

7.实际花费的时间

PSP2.1

Personal Software Process Stages

预估耗时(分钟)

实际耗时(分钟)

Planning

计划

30

35

· Estimate

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

30

20

Development

开发

180

240

· Analysis

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

200

180

· Design Spec

· 生成设计文档

30

20

· Design Review

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

0

0

· Coding Standard

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

20

10

· Design

· 具体设计

30

60

· Coding

· 具体编码

360

540

· Code Review

· 代码复审

20

60

· Test

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

120

180

Reporting

报告

120

100

· Test Report

· 测试报告

60

30

· Size Measurement

· 计算工作量

30

20

· Postmortem & Process Improvement Plan

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

60

40

合计

1290

1535

8.项目小结

收获:作为第一次项目经历,对软件工程的设计流程有了新的认识,一个项目的实现,重要的往往不是代码的质量,而是一个好的框架和一个良好的开发习惯,在这次开发过程中深刻的认识到这两点的重要性。

不足:基础知识的不足,导致前期准备时间过长。解决问题能力不足,导致编码所需时间长、错误多,部分功能完全没能力实现,需多学、多练。开发经验少,整个开发过程并不流畅。

最新文章

  1. Windows2003 IIS6.0支持32位和64位两种模式的设置方法
  2. Masonry控制台打印约束冲突问题解决
  3. win32 COM组件编写
  4. java 中 静态块的作用
  5. 解决 ERROR: JDWP Transport dt_socket failed to initialize, TRANSPORT_INIT(510)异常
  6. 连贯接口(fluent interface)的Java实现及应用。
  7. Linux基本命令之逻辑测试二
  8. android的color整理(一)
  9. Memcached源码分析之从SET命令开始说起
  10. java+++IO流操作
  11. Chrome_断点调试(使用vue-resource进行数据交互)
  12. 实现RunOnUiThread和RunOnUiThreadBlock
  13. Navicat连接Oracle的几个问题及解决方案
  14. mybatis拦截器案例之获取结果集总条数
  15. bzoj千题计划216:bzoj1499: [NOI2005]瑰丽华尔兹
  16. 【机器学习算法-python实现】矩阵去噪以及归一化
  17. VS2012不能创建数据库连接出现或者连接数据库时提示:未能载入文件或程序集“Microsoft.SqlServer.Management.Sdk.Sfc, Version=10.0.0.0, C
  18. EF进阶篇(二)——CRUD
  19. Python学习笔记(二)网络编程的简单示例
  20. 点击a标签 跳到当前页面指定div

热门文章

  1. python-django-fastdfs+Nginx的安装和配置_20191122
  2. 在CentOS安装CMake (CentOS7 64位适用)
  3. 简单php递归无限mysql记录
  4. CSS性能优化探讨
  5. POJ 3522 用不同的排序方式
  6. python-pymysql学习
  7. python 通过socket实现ssh功能
  8. jq ajaxPrefilter 防止重复提交ajax
  9. uploadifive如何动态传参
  10. 1——PHP常见的系统常量