获取每个文件行的整数数

时间:2016-01-21 16:56:10

标签: c file-io c-strings

我有一个#符号和整数的文件。我想避免读取#符号并计算每行的整数。我是C的新手,非常感谢任何提示和帮助!

这是我到目前为止所做的:

df

3 个答案:

答案 0 :(得分:2)

我想我明白你在做什么。您正在读取该文件,跳过每个空白注释行,然后计算您在每行中找到的整数数,从而在{{中保存整数计数(每行) 1}}。你走在正确的轨道上,你只是有几个打嗝。

虽然您可以使用各种array[]string.h工具来解决问题,但有时最好简单地使用指针并一次按下每行一个字符。您已经创建了指针ctype.h,您可以使用它来查找每行中整数的出现次数。

接近它的简单方法是检查p中的第一个字符(因此line)。如果它等于*pnewline,则该行为空或注释 - 跳过该行。接下来,对于该行中的每个字符,您希望前进'#'直到找到一个数字。您可以使用p,或只使用:

isdigit

找到数字后,递增整数:

    while (*p && (*p < '0' || '9' < *p)) p++;

注意:它是 if ('0' <= *p && *p <= '9') array[count]++; array[count]++;

然后只需前进该整数中的所有数字,当你到达下一个非数字字符时,重复循环:

array[count++];

当你到达行尾时,递增 while (*p && ('0' <= *p && *p <= '9')) p++; (用作行计数器,而不是整数计数)并转到下一行。

将所有部分组合在一起,您可以执行与以下类似的操作。注意:程序期望 filename 作为第一个参数给出,或者如果没有给出参数,它将默认从count读取:

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

如果您有疑问,请告诉我。

仅发现带有整数的打印行

要仅显示带整数的行,只需在打印循环中添加$ ./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 语句即可。 e.g:

if

<强>输出

    for (i = 0; i < count; i++)
        if (array[i])
            printf ("  line[%3d] : %d\n", i, array[i]);

答案 1 :(得分:1)

这是我认为你要做的我的版本 - 计算整数而不是个别数字。

#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

答案 2 :(得分:0)

问题在于这一行:

    if (isdigit((int)*p) && (*p == ' ' || *p == '\0')) {
        array[count] ++;
    }

&&的两个条件都不可能成立。如果字符是空格或空字节,则它也不能是数字。试试这个:

while (((p = fgets(line, MAX_LINE_LEN, fp)) != NULL) && (count <= MAX_NUM)) {
    if (line[0] != '\0' && 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 == '\0')) {
                // only count the space or end-of-string immediately after a number
                array[count] ++;
                in_number = 0;
            }
        } while (*p++ != '\0');
    }
}