从stdin只读整数

时间:2015-12-05 09:44:46

标签: c

我正在处理一个程序的一部分,该程序必须从stdin读取行,如下所示:

{ 1, 7, 22, 4, 7, 5, 11, 9, 1 }

我必须只保存程序的整数,以便以后使用它们。

我正在考虑的最原始的方法是将while循环放入程序中,如果

getchar() == ',' or getchar() == '}'

然后继续或结束阅读,如果没有','或者'}',然后阅读int。

你能想到更好的东西吗?

2 个答案:

答案 0 :(得分:1)

我一直有这样的心态:编写一般输入例程来处理数字输入而不管格式是否更有意义,编写5个不同的例程来处理5种不同的格式等等。strtol是量身定制的无论格式如何,转换字符串中包含的数字输入。声明是:

long int strtol(const char *nptr, char **endptr, int base);

它将指针作为输入并将endptr更新为nptr中转换后的数字后面的下一个字符这一事实允许您从一行的开头开始工作到它的结尾,转换找到的每个数字 - 无论输入的格式如何。这允许创建灵活的输入解析功能,并具有强大的错误检查。

下面的简短示例将说明这种方法。该示例将默认从stdin读取,或从作为第一个参数提供的任何文件名读取。 strtol调用包含在一个简短的xstrtol函数中,该函数只为每次转换提供正常的错误检查。将其放在函数中可以防止使用重复错误检查来混淆代码的主要逻辑。例如:

#include <stdio.h>
#include <stdlib.h> /* for strtol   */
#include <limits.h> /* for INT_MIN/INT_MAX */
#include <errno.h>  /* for errno    */

#define MAXL 256

long xstrtol (char *p, char **ep, int base);

int main (int argc, char **argv)
{
    int values[MAXL] = {0};
    char line[MAXL] = {0};
    size_t i, idx = 0;
    FILE *fp = argc > 1 ? fopen (argv[1], "r") : stdin;

    if (!fp) { /* validate file open */
        fprintf (stderr, "error: file open failen '%s'.\n", argv[1]);
        return 1;
    }

    /* read each line in file (up to MAXL chars per-line) */
    while (fgets (line, MAXL, fp)) {

        char *p = line;
        char *ep = p;
        errno = 0;

        /* convert each string of digits into number */
        while (errno == 0) {

            /* skip any non-digit characters */
            while (*p && ((*p != '-' && (*p < '0' || *p > '9')) ||
                (*p == '-' && (*(p+1) < '0' || *(p+1) > '9')))) p++;
            if (!*p) break;

            /* convert string to number */
            values[idx++] = (int)xstrtol (p, &ep, 10);

            if (errno) break;   /* check for error */

            if (idx == MAXL) {  /* check if array full */
                fprintf (stderr, "warning: MAXL values read.\n");
                break;
            }

            /* skip delimiters/move pointer to next digit */
            while (*ep && *ep != '-' && (*ep < '0' || *ep > '9')) ep++;
            if (*ep)
                p = ep;
            else  /* break if end of string */
                break;
        }
    }
    if (fp != stdin) fclose (fp);

    printf ("\nvalues read from '%s'\n\n", argc > 1 ? argv[1] : "stdin");
    for (i = 0; i < idx; i++)
        printf ("  values[%3zu] : %d\n", i, values[i]);

    return 0;
}

/** a simple strtol implementation with error checking.
 *  any failed conversion will cause program exit. Adjust
 *  response to failed conversion as required.
 */
long xstrtol (char *p, char **ep, int base)
{
    errno = 0;

    long tmp = strtol (p, ep, base);

    /* Check for various possible errors */
    if ((errno == ERANGE && (tmp == LONG_MIN || tmp == LONG_MAX)) ||
        (errno != 0 && tmp == 0)) {
        perror ("strtol");
        exit (EXIT_FAILURE);
    }

    if (*ep == p) {
        fprintf (stderr, "No digits were found\n");
        exit (EXIT_FAILURE);
    }

    return tmp;
}

输入文件 - 5种不同的格式

$ cat dat/10int_braces.txt
{ 8572, -2213, 6434, 16330, 3034, 12346, 4855, 16985, 11250, 1495 }

$ cat dat/10int.csv
8572, -2213, 6434, 16330, 3034
12346, 4855, 16985, 11250, 1495

$ cat dat/10int_space.txt
8572 -2213 6434 16330 3034 12346 4855 16985 11250 1495

$ cat dat/10int_nl.txt
8572
-2213
6434
16330
3034
12346
4855
16985
11250
1495

$ cat dat/10int_5x2.txt
[[  8572  -2213  ]
 [  6434  16330  ]
 [  3034  12346  ]
 [  4855  16985  ]
 [ 11250   1495  ]]

$ cat dat/10intmess.txt
8572,;a -2213,;--a 6434,;
a- 16330,;a

- The Quick
Brown%3034 Fox
12346Jumps Over
A
4855,;*;Lazy 16985/,;a
Dog.
11250
1495

使用/输出

所有格式都提供相同的输出,例如:

$ ./bin/fgets_xstrtol_simple <dat/10int_braces.txt

values read from 'stdin'

  values[  0] : 8572
  values[  1] : -2213
  values[  2] : 6434
  values[  3] : 16330
  values[  4] : 3034
  values[  5] : 12346
  values[  6] : 4855
  values[  7] : 16985
  values[  8] : 11250
  values[  9] : 1495

您可以根据需要轻松添加动态分配/重新分配。如果您有任何问题,请告诉我。

答案 1 :(得分:0)

不要打扰getchar,您可以尝试scanf

int d1,d2,d3;
scanf( "{%d,%d,%d}", &d1, &d2, &d3);
printf( "d1 = %d, d2 = %d, d3 = %d\n", d1, d2, d3 );

鉴于stdin的输入采用此格式{ 1, 7, 22 },您可以在d1d2d3中获得所需内容。< / p>

如果输入数量未修复,您可以尝试fgetsstrtok的组合:

char input_line[1000];
fgets( input_line, sizeof( input_line ), stdin );
char *pch = strtok (input_line,"{, \t}");
printf( "%d \n", atoi( pch ) );
while (pch != NULL)
{
    pch = strtok (NULL, "{, \t}");
    if( pch != NULL && pch[0] != '\n' )
    {
        printf( "%d \n", atoi( pch ) );
    }
}

如果输入为{ 1, 7, 22, 4, 7, 5, 11, 9, 1 },您将获得:

{ 1, 7, 22, 4, 7, 5, 11, 9, 1}      
1 
7 
22 
4 
7 
5 
11 
9 
1