while(scanf(“%s”,word)!=EOF)为什么可以输出单词个数
时间: 2024-12-21 14:31:24 浏览: 2
`while (scanf("%s", word) != EOF)` 这段代码是用来从标准输入读取字符串,直到遇到文件结束符(通常表示为 `EOF`,实际上是输入流的特殊值)。当 `scanf` 成功读取一个字符串到 `word` 变量时,它返回成功读取的项数。如果读取的是空行或到达了文件结束,`scanf` 就不会读取更多的数据并返回0。
当你使用 `!= EOF` 作为条件判断,实际上是在说只要不是遇到了文件结束,就继续循环。因此,这个循环会一直执行,直到所有非空行(即单词)都被读取完毕。你可以用一个计数器变量(如 `int count = 0`)来追踪已经读取的单词数量,在每次 `scanf` 返回后增加计数:
```c
#include <stdio.h>
int main() {
int count = 0;
char word[100]; // 假设每个单词长度不超过100字符
while (scanf("%s", word) != EOF) {
if (word[0] != '\n') { // 忽略空行
count++;
}
}
printf("Number of words read: %d\n", count);
return 0;
}
```
在这个例子中,我们额外检查了是否读到了空行(`\n`),因为`scanf("%s")`会把空行也当作一个"单词"。如果不想计算空行,可以直接忽略这个检查。
相关问题
#include <stdio.h> #include <stdlib.h> #include <string.h> #define MAX_LENGTH 100 typedef struct { char word[MAX_LENGTH]; int count; } WordCount; int main() { char searchWord[MAX_LENGTH]; int i, j, found; WordCount *words = NULL; int wordsCount = 0; // 打开文件 FILE *file = fopen("ss.txt", "r"); // 读取文件内容,逐个单词进行统计 char word[MAX_LENGTH]; while (fscanf(file, "%s", word) != EOF) { found = 0; // 检查该单词是否已经存在于统计数组中 for (i = 0; i < wordsCount; i++) { if (strcmp(word, words[i].word) == 0) { found = 1; words[i].count++; break; } } // 如果单词不存在于统计数组中,则将其添加到数组中 if (!found) { words = (WordCount*)realloc(words, (wordsCount + 1) * sizeof(WordCount)); strcpy(words[wordsCount].word, word); words[wordsCount].count = 1; wordsCount++; } } // 关闭文件 fclose(file); // 输入要查找的单词 printf("请输入要查找的单词:"); scanf("%s", searchWord); // 在统计数组中查找该单词并输出重复次数 found = 0; for (i = 0; i < wordsCount; i++) { if (strcmp(searchWord, words[i].word) == 0) { printf("单词 %s 出现了 %d 次\n", searchWord, words[i].count); found = 1; break; } } // 如果未找到该单词,则输出未找到的提示信息 if (!found) { printf("未找到单词 %s\n", searchWord); } // 释放内存 free(words); return 0; }逐行解释代码
好的,以下是对代码逐行的解释:
```c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_LENGTH 100
typedef struct {
char word[MAX_LENGTH];
int count;
} WordCount;
int main() {
char searchWord[MAX_LENGTH];
int i, j, found;
WordCount *words = NULL;
int wordsCount = 0;
```
这部分是包含头文件、定义常量、定义结构体以及声明变量的部分。
```c
// 打开文件
FILE *file = fopen("ss.txt", "r");
```
这行代码打开名为 "ss.txt" 的文件,以只读方式打开。
```c
// 读取文件内容,逐个单词进行统计
char word[MAX_LENGTH];
while (fscanf(file, "%s", word) != EOF) {
found = 0;
// 检查该单词是否已经存在于统计数组中
for (i = 0; i < wordsCount; i++) {
if (strcmp(word, words[i].word) == 0) {
found = 1;
words[i].count++;
break;
}
}
// 如果单词不存在于统计数组中,则将其添加到数组中
if (!found) {
words = (WordCount*)realloc(words, (wordsCount + 1) * sizeof(WordCount));
strcpy(words[wordsCount].word, word);
words[wordsCount].count = 1;
wordsCount++;
}
}
```
这部分代码逐个单词读取文件内容,并进行统计。使用 `fscanf` 函数从文件中读取单词,如果读取到文件末尾(EOF),则停止循环。在循环中,检查该单词是否已经存在于统计数组中,如果存在则增加计数,如果不存在则将其添加到数组中。
```c
// 关闭文件
fclose(file);
```
这行代码关闭文件。
```c
// 输入要查找的单词
printf("请输入要查找的单词:");
scanf("%s", searchWord);
```
这部分代码用于输入要查找的单词。
```c
// 在统计数组中查找该单词并输出重复次数
found = 0;
for (i = 0; i < wordsCount; i++) {
if (strcmp(searchWord, words[i].word) == 0) {
printf("单词 %s 出现了 %d 次\n", searchWord, words[i].count);
found = 1;
break;
}
}
```
这部分代码在统计数组中查找输入的单词,并输出其重复次数。
```c
// 如果未找到该单词,则输出未找到的提示信息
if (!found) {
printf("未找到单词 %s\n", searchWord);
}
```
这部分代码用于处理未找到单词的情况,输出未找到的提示信息。
```c
// 释放内存
free(words);
return 0;
}
```
最后,这行代码用于释放动态分配的内存,并返回程序执行成功的标志。
希望这样的解释对你有帮助!如果还有其他问题,请随时提问。
#pragma GCC optimize ("O3") #pragma pack (16)//所有的存储都是以16个字节为单位的 #include <stdio.h> #include <stdlib.h> #include <string.h> #define tolower(c) (c>='A'&&c<='Z')?c-'A'+'a':c #define DATA 5200000 #define SIZE 1000005 int trie[4200000][26]; typedef struct node { int cnt; int logo; struct node *child[26]; } Node; Node *root; char str[35000000]; typedef struct word { char wor[85]; int cnt; } Word; Word w[300000]; struct node *creat() { Node *Root = (Node *)malloc(sizeof(Node)); Root->logo = 0; Root->cnt = 0; for (int i = 0; i < 26; i++) { Root->child[i] = NULL; } return Root; } void insert(Node *root, char *word, int flag) { struct node *leaf = root; for (int i = 0; word[i] != '\0'; i++) { int index = word[i] - 'a'; if (!leaf->child[index]) { leaf->child[index] = creat(); } leaf = leaf->child[index]; } if (leaf->logo != -1) leaf->logo = flag; leaf->cnt++; } int count = 0; void dfs(Node *leaf, char *word, int level) { if (leaf->logo == 1) { word[level] = '\0'; strcpy(w[count++].wor, word); w[count - 1].cnt = leaf->cnt; } for (int i = 0; i < 26; i++) { if (leaf->child[i]) { word[level] = i + 'a'; dfs(leaf->child[i], word, level + 1); } } } int cmp(const void *p1, const void *p2) { Word *v1, *v2; v1 = (Word *)p1; v2 = (Word *)p2; if (v1->cnt != v2->cnt) return v2->cnt - v1->cnt; else return strcmp(v1->wor, v2->wor); } int main(int argc, char *argv[]) { char s[1024]; int temp; int n, m;//读入n,m; //n = atoi(argv[1]); //m = atoi(argv[2]); scanf("%d%d", &n, &m); //读入stopwords中的元素,并令末序数组值为0,即该单词不计入 root = creat(); FILE *stopwords = fopen("stopwords.txt", "r"); while (fscanf(stopwords, "%s", s) != EOF) { insert(root, s, -1); } int cnt; FILE *article = fopen("article.txt", "r"); cnt = fread(str, sizeof(char), 35000000, article); char word[85]; int w_cnt = 0; for (int i = 0; i < cnt; i++) { char c = tolower(str[i]); if (c >= 'a' && c <= 'z') { word[w_cnt++] = c; } else { word[w_cnt] = '\0'; p = 0; w_cnt = 0; if (strlen(word) > 0) { insert(root, word, 1); } }//对article中的所有单词进行计数 } dfs(root, word, 0); qsort(w, count, sizeof(w[0]), cmp); printf("%s", w[0].cnt); return 0; }
这段代码是一个基于Trie树的单词统计程序,在给定的文章中统计出现频率最高的单词。程序从文件中读入stopwords(不计入统计的常用单词)和article(待统计的文章),使用Trie树统计文章中出现的单词频率,并按照出现频率从高到低输出结果。
具体实现中,程序使用了一个结构体Node表示Trie树中的节点,其中包含了该节点出现次数cnt和单词编号logo。同时,程序定义了一个结构体Word表示单词及其出现次数。程序中的insert函数用于向Trie树中插入单词,dfs函数用于遍历Trie树并统计单词出现次数,cmp函数用于按照出现频率从高到低排序输出结果。
注意,程序中的一些细节需要注意,比如将所有字母转换为小写字母,以及对于长度为0的单词不进行计数等。
阅读全文