获取每个文件行的整数个数
Get number of integers per file line
我有一个包含 # 符号和整数的文件。我想避免阅读 # 符号并计算每行的整数数。我是 C 的新手,非常感谢任何提示和帮助!
这是我目前拥有的:
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#define MAX_NUM 1000
#define MAX_LINE_LEN 2048
int main(int argc, char *argv[]) {
FILE *fp;
char line[MAX_LINE_LEN], *p;
int i, array[MAX_NUM] = { 0 }, count = 0;
char filename[100];
printf("Enter the file name: \n");
scanf("%s", filename);
fp = fopen(filename,"r");
while (((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) && (count < MAX_NUM)) {
if (strlen(line) > 1) {
//counter for line number.
count++;
//this line does not contains numbers.
if (line[0] == '#')continue;
do {
//notice. this way we count 111 as one number instead of 3.
if (isdigit((int)*p) && (*p == ' ' || *p == '[=10=]')) {
array[count] ++;
}
} while (*p++ != '[=10=]');
}
//printf("array[]: %d", array[i]);
int i;
for(i = 0; i < count; ++ i){
printf("%d",array[i]);
}
}
return(0);
}
问题出在这一行:
if (isdigit((int)*p) && (*p == ' ' || *p == '[=10=]')) {
array[count] ++;
}
不可能 &&
的两个条件都为真。如果字符是 space 或空字节,则它不能同时是数字。试试这个:
while (((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) && (count <= MAX_NUM)) {
if (line[0] != '[=11=]' && line[0] != '\n') { // Skip empty lines
//counter for line number.
count++;
//this line does not contains numbers.
if (line[0] == '#')continue;
int in_number = 0;
do {
if (isdigit((int) *p)) {
in_number = 1;
} else if (in_number && (*p == ' ' || *p == '\n' || *p == '[=11=]')) {
// only count the space or end-of-string immediately after a number
array[count] ++;
in_number = 0;
}
} while (*p++ != '[=11=]');
}
}
这是我认为你正在尝试做的事情的版本 - 计算整数而不是单个数字。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#define MAX_NUM 1000
#define MAX_LINE_LEN 2048
void fatal(char *msg) {
printf("%s\n", msg);
exit (1);
}
int main(int argc, char *argv[]) {
FILE *fp;
char filename[100];
char line[MAX_LINE_LEN];
char *p;
int i;
int array[MAX_NUM] = { 0 };
int count = 0;
printf("Enter the file name: \n");
if (scanf("%s", filename) != 1)
fatal("Bad filename entry");
if ((fp = fopen(filename,"r")) == NULL)
fatal("Unable to open the file");
while ((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) {
if (count >= MAX_NUM)
fatal("Broke the array");
if (line[0] != '#') {
p = strtok(line, " ");
while (p != NULL) {
if(isdigit(*p))
array[count]++;
p = strtok(NULL, " ");
}
}
count++;
}
for(i = 0; i < count; i++) {
if(array[i]) {
printf("Integers: %d\n", array[i]);
}
}
return 0;
}
程序输出:
Enter the file name:
test.txt
Integers: 9
Integers: 4
Integers: 5
Integers: 5
Integers: 5
Integers: 1
Integers: 20
我想我明白你在做什么。您正在阅读文件,跳过每个 blank 或 comment 行,然后计算您在保存整数计数的每一行中找到的整数数(per-line) 在 array[]
。你在正确的轨道上,你只是有一些小问题。
虽然您可以使用各种 string.h
和 ctype.h
工具来解决问题,但有时最好只使用指针并逐行逐行逐字逐句处理。您已经创建了指针 p
,您可以使用它来查找每行中整数的出现次数。
处理它的简单方法是检查 line
中的第一个字符(因此 *p
)。如果它等于 newline
或 '#'
,则该行为空或 comment -- 跳过该行。接下来,对于行中的每个字符,您要前进 p
直到找到一个数字。您可以使用 isdigit
,或者简单地使用:
while (*p && (*p < '0' || '9' < *p)) p++;
找到数字后,增加整数计数:
if ('0' <= *p && *p <= '9') array[count]++;
(注意: 它是 array[count]++;
不是 array[count++];
)
然后简单地遍历该整数中的所有数字,当你到达下一个 non-digit 字符时,重复循环:
while (*p && ('0' <= *p && *p <= '9')) p++;
当您到达行尾时,增加 count
(这是您的 line-counter,而不是您的整数计数)并转到下一行。
把所有的部分放在一起,你可以做类似下面的事情。注意:程序期望 filename 作为第一个参数给出,或者如果没有给出参数,它将默认从 stdin
读取:
#include <stdio.h>
enum { MAX_NUM = 1000, MAX_LINE_LEN = 2048 };
int main(int argc, char **argv) {
char line[MAX_LINE_LEN], *p;
int i, array[MAX_NUM] = { 0 }, count = 0;
FILE *fp = argc > 1 ? fopen (argv[1], "r") : stdin;
if (!fp) { /* validate the file is open */
fprintf (stderr, "error: file open failed '%s'.\n", argv[1]);
return 0;
}
while (count < MAX_NUM && (p = fgets(line, MAX_LINE_LEN, fp))) {
if (*p == '\n' || *p == '#') { count++; continue; }
while (*p) { /* for each char in line */
while (*p && (*p < '0' || '9' < *p)) p++;
if ('0' <= *p && *p <= '9') array[count]++;
while (*p && ('0' <= *p && *p <= '9')) p++;
}
count++; /* increment line count */
}
if (fp != stdin) fclose(fp);
printf ("\nThe lines from '%s' contained the following integers:\n\n",
argc > 1 ? argv[1] : "stdin");
for (i = 0; i < count; i++)
printf (" line[%3d] : %d\n", i, array[i]);
putchar ('\n');
return(0);
}
示例输入
$ cat ../dat/commentint.txt
# each burst is one long
1 2 3 4 5 6 5 4 5
14 62 48 14
# these have 5 integers
1 3 5 7 9
123 456 789 1234 5678
34 34 34 34 34
# special case, I guess
1
# now a burstful sequence
1 2 2 2 2 2 3 3 4 4 4 4 5 5 6 7 7 7 1 1
例子Use/Output
$ ./bin/skipcomment ../dat/commentint.txt
The lines from '../dat/commentint.txt' contained the following integers:
line[ 0] : 0
line[ 1] : 9
line[ 2] : 4
line[ 3] : 0
line[ 4] : 0
line[ 5] : 5
line[ 6] : 5
line[ 7] : 5
line[ 8] : 0
line[ 9] : 0
line[ 10] : 1
line[ 11] : 0
line[ 12] : 0
line[ 13] : 20
如果您有任何问题,请告诉我。
只打印找到整数的行
要仅显示带有整数的行,只需将 if
语句添加到您的打印循环。例如:
for (i = 0; i < count; i++)
if (array[i])
printf (" line[%3d] : %d\n", i, array[i]);
输出
$ ./bin/skipcomment ../dat/commentint.txt
The lines from '../dat/commentint.txt' contained the following integers:
line[ 1] : 9
line[ 2] : 4
line[ 5] : 5
line[ 6] : 5
line[ 7] : 5
line[ 10] : 1
line[ 13] : 20
我有一个包含 # 符号和整数的文件。我想避免阅读 # 符号并计算每行的整数数。我是 C 的新手,非常感谢任何提示和帮助!
这是我目前拥有的:
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#define MAX_NUM 1000
#define MAX_LINE_LEN 2048
int main(int argc, char *argv[]) {
FILE *fp;
char line[MAX_LINE_LEN], *p;
int i, array[MAX_NUM] = { 0 }, count = 0;
char filename[100];
printf("Enter the file name: \n");
scanf("%s", filename);
fp = fopen(filename,"r");
while (((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) && (count < MAX_NUM)) {
if (strlen(line) > 1) {
//counter for line number.
count++;
//this line does not contains numbers.
if (line[0] == '#')continue;
do {
//notice. this way we count 111 as one number instead of 3.
if (isdigit((int)*p) && (*p == ' ' || *p == '[=10=]')) {
array[count] ++;
}
} while (*p++ != '[=10=]');
}
//printf("array[]: %d", array[i]);
int i;
for(i = 0; i < count; ++ i){
printf("%d",array[i]);
}
}
return(0);
}
问题出在这一行:
if (isdigit((int)*p) && (*p == ' ' || *p == '[=10=]')) {
array[count] ++;
}
不可能 &&
的两个条件都为真。如果字符是 space 或空字节,则它不能同时是数字。试试这个:
while (((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) && (count <= MAX_NUM)) {
if (line[0] != '[=11=]' && line[0] != '\n') { // Skip empty lines
//counter for line number.
count++;
//this line does not contains numbers.
if (line[0] == '#')continue;
int in_number = 0;
do {
if (isdigit((int) *p)) {
in_number = 1;
} else if (in_number && (*p == ' ' || *p == '\n' || *p == '[=11=]')) {
// only count the space or end-of-string immediately after a number
array[count] ++;
in_number = 0;
}
} while (*p++ != '[=11=]');
}
}
这是我认为你正在尝试做的事情的版本 - 计算整数而不是单个数字。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#define MAX_NUM 1000
#define MAX_LINE_LEN 2048
void fatal(char *msg) {
printf("%s\n", msg);
exit (1);
}
int main(int argc, char *argv[]) {
FILE *fp;
char filename[100];
char line[MAX_LINE_LEN];
char *p;
int i;
int array[MAX_NUM] = { 0 };
int count = 0;
printf("Enter the file name: \n");
if (scanf("%s", filename) != 1)
fatal("Bad filename entry");
if ((fp = fopen(filename,"r")) == NULL)
fatal("Unable to open the file");
while ((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) {
if (count >= MAX_NUM)
fatal("Broke the array");
if (line[0] != '#') {
p = strtok(line, " ");
while (p != NULL) {
if(isdigit(*p))
array[count]++;
p = strtok(NULL, " ");
}
}
count++;
}
for(i = 0; i < count; i++) {
if(array[i]) {
printf("Integers: %d\n", array[i]);
}
}
return 0;
}
程序输出:
Enter the file name:
test.txt
Integers: 9
Integers: 4
Integers: 5
Integers: 5
Integers: 5
Integers: 1
Integers: 20
我想我明白你在做什么。您正在阅读文件,跳过每个 blank 或 comment 行,然后计算您在保存整数计数的每一行中找到的整数数(per-line) 在 array[]
。你在正确的轨道上,你只是有一些小问题。
虽然您可以使用各种 string.h
和 ctype.h
工具来解决问题,但有时最好只使用指针并逐行逐行逐字逐句处理。您已经创建了指针 p
,您可以使用它来查找每行中整数的出现次数。
处理它的简单方法是检查 line
中的第一个字符(因此 *p
)。如果它等于 newline
或 '#'
,则该行为空或 comment -- 跳过该行。接下来,对于行中的每个字符,您要前进 p
直到找到一个数字。您可以使用 isdigit
,或者简单地使用:
while (*p && (*p < '0' || '9' < *p)) p++;
找到数字后,增加整数计数:
if ('0' <= *p && *p <= '9') array[count]++;
(注意: 它是 array[count]++;
不是 array[count++];
)
然后简单地遍历该整数中的所有数字,当你到达下一个 non-digit 字符时,重复循环:
while (*p && ('0' <= *p && *p <= '9')) p++;
当您到达行尾时,增加 count
(这是您的 line-counter,而不是您的整数计数)并转到下一行。
把所有的部分放在一起,你可以做类似下面的事情。注意:程序期望 filename 作为第一个参数给出,或者如果没有给出参数,它将默认从 stdin
读取:
#include <stdio.h>
enum { MAX_NUM = 1000, MAX_LINE_LEN = 2048 };
int main(int argc, char **argv) {
char line[MAX_LINE_LEN], *p;
int i, array[MAX_NUM] = { 0 }, count = 0;
FILE *fp = argc > 1 ? fopen (argv[1], "r") : stdin;
if (!fp) { /* validate the file is open */
fprintf (stderr, "error: file open failed '%s'.\n", argv[1]);
return 0;
}
while (count < MAX_NUM && (p = fgets(line, MAX_LINE_LEN, fp))) {
if (*p == '\n' || *p == '#') { count++; continue; }
while (*p) { /* for each char in line */
while (*p && (*p < '0' || '9' < *p)) p++;
if ('0' <= *p && *p <= '9') array[count]++;
while (*p && ('0' <= *p && *p <= '9')) p++;
}
count++; /* increment line count */
}
if (fp != stdin) fclose(fp);
printf ("\nThe lines from '%s' contained the following integers:\n\n",
argc > 1 ? argv[1] : "stdin");
for (i = 0; i < count; i++)
printf (" line[%3d] : %d\n", i, array[i]);
putchar ('\n');
return(0);
}
示例输入
$ cat ../dat/commentint.txt
# each burst is one long
1 2 3 4 5 6 5 4 5
14 62 48 14
# these have 5 integers
1 3 5 7 9
123 456 789 1234 5678
34 34 34 34 34
# special case, I guess
1
# now a burstful sequence
1 2 2 2 2 2 3 3 4 4 4 4 5 5 6 7 7 7 1 1
例子Use/Output
$ ./bin/skipcomment ../dat/commentint.txt
The lines from '../dat/commentint.txt' contained the following integers:
line[ 0] : 0
line[ 1] : 9
line[ 2] : 4
line[ 3] : 0
line[ 4] : 0
line[ 5] : 5
line[ 6] : 5
line[ 7] : 5
line[ 8] : 0
line[ 9] : 0
line[ 10] : 1
line[ 11] : 0
line[ 12] : 0
line[ 13] : 20
如果您有任何问题,请告诉我。
只打印找到整数的行
要仅显示带有整数的行,只需将 if
语句添加到您的打印循环。例如:
for (i = 0; i < count; i++)
if (array[i])
printf (" line[%3d] : %d\n", i, array[i]);
输出
$ ./bin/skipcomment ../dat/commentint.txt
The lines from '../dat/commentint.txt' contained the following integers:
line[ 1] : 9
line[ 2] : 4
line[ 5] : 5
line[ 6] : 5
line[ 7] : 5
line[ 10] : 1
line[ 13] : 20