WC.exe【C】

时间:2022-04-06 21:00:16

gitee传送门!!!(电脑打不开github,多次尝试未果,决定先用gitee存着先)

项目要求

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,-w,-l。如果处理好这个问题,整个基本功能就差不多完成了。想过用if语句加strcmp(),但是编译器报错,原因可能是strcmp的两个参数是字符串,而命令行参数中的argv[]是指针数组,表示的是一系列字符串。尝试过转换,但都不成功。后来和同学讨论的时候发现了getopt()函数,它成功地解决了这个问题。

④尝试做扩展功能。效果很不好,到现在还实现不了。

关键代码及思路

  • 主函数
    #include<stdio.h>
    #include<ctype.h>
    #include<stdbool.h>
    #include<stdlib.h>
    #include<unistd.h>
    int main(int argc,char *argv[])
    {
    char ch;
    int choice; FILE *fp;
    unsigned long lines=;//行数
    unsigned long count=;//字符数
    int n_words=;//单词数
    bool inword=false;//一个flag,如果c在单词中,inword等于true unsigned long cd_lines=;//代码行数
    unsigned long e_rows=;//空行数
    unsigned long cm_lines=;//注释行数 while((choice=getopt(argc,argv,"c::w::l::a"))!=-){
    switch(choice){
    //计算字符数
    case 'c':{
    printf("File %s has %lu characters\n",argv[],count);
    break; }
    //计算单词数
    case 'w':{
    printf("File %s has %d words\n",argv[],n_words);
    break; }
    //计算行数
    case 'l':{
    printf("File %s has %lu lines\n",argv[],lines);
    break; }
    /*case 'a':{
    printf("File %s has %lu lines of code\n",argv[2],cd_lines);//代码行数
    printf("File %s has %lu empty rows\n",argv[2],e_rows);//空行数
    printf("File %s has %lu comment lines\n",argv[2],cm_lines);//注释行数
    break; }此代码仅作记录,没有实现*/
    }
    return ;
    }
    }
  • 基础功能——计算字符数
    //ch从fp中读取,每读取一个就加一,直至文件结尾
    while((ch=getc(fp))!=EOF)
    {
    count++;
    }
    //关闭文件
  • 基础功能——计算单词数
        while((ch=getc(fp))!=EOF){
    //若ch不为空格,不为标点符号,不在单词中
    if(!isspace(ch)&&!ispunct(ch)&&!inword){
    inword=true;
    n_words++;
    }
    //若ch不在单词中 并且 是空格或者标点符号
    if(inword&&(isspace(ch)||ispunct(ch)))
    inword=false;
    }
  • 基础功能——计算行数
    //当ch等于换行符或‘\0’,行数加一
    while((ch=getc(fp))!=EOF)
    {
    if(ch=='\n'||ch=='\0')lines++;
    }
  • 扩展功能——返回更复杂的数据(代码行 / 空行 / 注释行)(PS:仅仅为了记录思路,没有实现成功)
      //用数组来统计每行的字符数,根据字符数作判断
    char st[];
    while(!feof(fp)){
    int show_char=;
    int i=;
    if(fgets(st,strlen(st),fp)){
    for(i=;i<strlen(st);i++)
    {
    if(st[]=='\n'){
    e_rows++;
    break;
    }
    else if(st[i]=='\0'||(st[i]=='{'&&st[i+]=='\0')||(st[i]=='}'&&st[i+]=='\0')){
    e_rows++;
    break;
    }
    }
    }
    }
    //计算注释行
    ch=getc(fp);
    while (ch!= EOF){ (ch == '/'){
    ch = getc(fp);
    if (ch == '/')
    while (ch != '\n')
    {
    ch = getc(fp);
    }
    cm_lines++;
    ch = getc(fp);
    }
    }

测试结果

WC.exe【C】WC.exe【C】

WC.exe【C】WC.exe【C】

PSP

PSP2.1

Personal Software Process Stages

预估耗时(分钟)

实际耗时(分钟)

·Planning

·计划

 60  60

· Estimate

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

 10*120 12*120 

·Development

·开发

 8*120 10*120 

· Analysis

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

 540 360 

· Design Spec

· 生成设计文档

 20  15

· Design Review

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

 20  25

· Coding Standard

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

 10  10

· Design

· 具体设计

 20  20

· Coding

· 具体编码

 90  100

· Code Review

· 代码复审

 30  15

· Test

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

 60  180

·Reporting

·报告

 60  70

· Test Report

· 测试报告

 30  20

· Size Measurement

· 计算工作量

 10  10

· Postmortem & Process Improvement Plan

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

 120  150

合计

   3230  3675

总结

这个项目让我对命令行参数有所了解。

总体来说基本功能写的还挺顺利的,但是扩展功能花了两三天,也想了很多方法,但就是实现不出来。感觉是自己学的知识还不够,所以继续努力吧。希望能早点把这个坑给填了。