wc(c语言)

仓库地址

https://github.com/18yy/MYWC

题目要求

wc.exe 是一个常见的工具,它能统计文本文件的字符数、单词数和行数。这个项目要求写一个命令行程序,模仿已有wc.exe 的功能,并加以扩充,给出某程序设计语言源文件的字符数、单词数和行数。

实现一个统计程序,它能正确统计程序文件中的字符数、单词数、行数,以及还具备其他扩展功能,并能够快速地处理多个文件。

具体功能要求:

  1. 程序处理用户需求的模式为
wc.exe [parameter] [file_name]
  1. 基本功能列表:
wc.exe -c file.c     //返回文件 file.c 的字符数
wc.exe -w file.c    //返回文件 file.c 的词的数目  
wc.exe -l file.c      //返回文件 file.c 的行数
  1. 扩展功能:
 -s 递归处理目录下符合条件的文件。
 -a 返回更复杂的数据(代码行 / 空行 / 注释行)。
  • 空行:本行全部是空格或格式控制字符,如果包括代码,则只有不超过一个可显示的字符,例如“{”。

  • 代码行:本行包括多于一个字符的代码。

  • 注释行:本行不是代码行,并且本行包括注释。一个有趣的例子是有些程序员会在单字符后面加注释:

 } //注释

在这种情况下,这一行属于注释行。

  1. 高级功能:
-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. 思路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++;
		}
		
  1. 思路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 ;
	}
  1. 思路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++;
		}
    	
	}	
    • 注释行:
  1. 思路(测试不过就删了):遇到//则标记,接下来遇到换行符,注释行加一,遇到/标记,遇到换行符,注释行加一,最后遇到/注释行加一
//获取注释行
		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可以整理自己整个工程的时间耗费,是一个不错的方法
posted @ 2018-09-14 22:02  杨艺  阅读(413)  评论(0编辑  收藏  举报