第二周结对编程作业:词频统计

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第二周结对编程作业:词频统计相关的知识,希望对你有一定的参考价值。

程序实现的功能是统计文件中出现的词和每个词的频数。

这里的词的规定和平时语言上的规定不太一样,但是更标准化,规定了长度、组成字符和分隔符,在实现中也从这几个方面来寻找词。

/*判断字符是否是字母*/
bool isAlphabetic (char c)  {

    return ((c >= a && c <= z)
            || (c >= A && c <= Z));

}

/*判断字符是否是数字*/
bool isNumerical (char c)  {

    return (c >= 0 && c <= 9);

}

/*判断字符是否是分隔符*/
bool isSeparator (char c)  {

    return (! isAlphabetic(c) && ! isNumerical(c));

}


/*判断字符串是否是单词*/
bool isWord (char s[])  {

    for (int i = 0; i < 4; i ++) {        //字符串前四位必须是字母
        if (!isAlphabetic (s[i]))  {
            return false;
        }
    }

    return true;

}

要统计词频首先要能够判断两个词是否相同,这里用简单的字符串匹配来实现

/*判断两个单词是否相同(无视大小写)*/
bool isSame (char a[], char b[]) {

    char s1[20], s2[20];
    strcpy (s1, a);
    strcpy (s2, b);
    strupr (s1);
    strupr (s2);
    return (!strcmp (s1, s2));

}

统计的过程很简单,扫描文件,每找到一个词,先判断它是否是一个新词,如果是,在词典中加入一个新词,词频为1;如果不是,词典中该词频数+1。

用到的数据结构:包含记录词的字符串和记录频数的计数器

/*结构体,包括单词以及它出现的次数*/
struct word_count {

    char word[30];           //单词
    int count;               //出现的次数

};

使用以上的结构和功能来统计,并且给统计结果排序,输出到文件

/*统计单词的出现次数,并排序,输出到文件中*/
void count (FILE * in, FILE * out)  {

    char s[30];                            //当前正在处理的字符串
    char c =  ;                          //当前字符
    word_count w[20000];                   //word_count数组,存放出现的单词及出现次数
    int countUpNow = 0, i, j, k;           //countUpNow记录目前已有的单词数
    bool haveRecord;                       //记录一个单词之前是否有出现过


    //统计单词的出现次数
    while (! feof(in))  {

        memset (s, 0, sizeof(s));
        haveRecord = false;
        k = 0;
        c =  ;

        while (isSeparator (c) && !feof(in))  {           //跳过分隔符
            c = fgetc (in);
        }

        while (! isSeparator (c) && !feof(in))  {         //读入一个字符串
            s[k++] = c;
            c = fgetc (in);
        }

        if (isWord (s))  {                                //如果是一个单词

            for (i = 0; i < countUpNow; i ++)  {          //查找之前该单词是否出现过

                if (isSame (s, w[i].word))  {             //若出现过

                    w[i].count ++;                        //频数+1
                    if (s < w[i].word) {                  //如果该单词的字典序较小
                        strcpy (w[i].word, s);            //替换之前的单词
                    }
                    haveRecord = true;
                    break;
                }

            }

            if (! haveRecord)  {                          //若没有出现过

                w[countUpNow].count = 1;
                strcpy (w[countUpNow].word, s);
                countUpNow ++;

            }

        }

    }


    //按照次数和字典序进行排序
    word_count temp;

    for (i = 0; i < countUpNow; i ++) {

        for (j = 0; j < i; j ++)  {
            if (w[j].count < w[i].count)  {
                temp = w[j];
                w[j] = w[i];
                w[i] = temp;
            }
            else if (w[j].count == w[i].count && w[j].word > w[i].word) {
                temp = w[j];
                w[j] = w[i];
                w[i] = temp;
            }
        }

    }


    //结果输出到目标文件中
    for (i = 0; i < countUpNow; i ++) {

        itoa (w[i].count, s, 20);
        fputs ("<", out);
        fputs (w[i].word, out);
        fputs (">: ", out);
        fputs (s, out);
        fputc (\n, out);

    }

}

下面是运行结果:

 (发表的时候网络好像出了问题,不能上传图片,稍后会把图片补上)

GitHub地址:

https://github.com/EverBlue1997/MyFirstRepository/blob/master/CountWords.cpp

以上是关于第二周结对编程作业:词频统计的主要内容,如果未能解决你的问题,请参考以下文章

第二周作业-词频统计

软件工程第二周作业----词频统计

第二周作业

作业4:结对编程—词频统计

第二周进度条

结对编程-词频统计