繁体   English   中英

在C语言中读取可变长度字符串用户输入

[英]Reading in a variable length string user input in C

我试图读取可变长度的用户输入并执行一些操作(例如在字符串中搜索子字符串)。

问题是我不知道我的字符串有多大(文本很可能是3000-4000个字符)。

我附上我尝试过的示例代码和输出:

char t[],p[];
int main(int argc, char** argv) {
    fflush(stdin);
    printf(" enter a string\n");
    scanf("%s",t);

    printf(" enter a pattern\n");
    scanf("%s",p);

    int m=strlen(t);
    int n =strlen(p);
    printf(" text is %s %d  pattrn is %s %d \n",t,m,p,n);
    return (EXIT_SUCCESS);
}

输出为:

enter a string
bhavya
enter a pattern
av
text is bav 3  pattrn is av 2

永远不要使用不安全之类的东西scanf("%s")或我个人不喜欢, gets() -有没有办法来防止缓冲区溢出了这样的事情。

您可以使用更安全的输入法,例如:

#include <stdio.h>
#include <string.h>

#define OK       0
#define NO_INPUT 1
#define TOO_LONG 2
static int getLine (char *prmpt, char *buff, size_t sz) {
    int ch, extra;

    // Get line with buffer overrun protection.
    if (prmpt != NULL) {
        printf ("%s", prmpt);
        fflush (stdout);
    }
    if (fgets (buff, sz, stdin) == NULL)
        return NO_INPUT;

    // If it was too long, there'll be no newline. In that case, we flush
    // to end of line so that excess doesn't affect the next call.
    if (buff[strlen(buff)-1] != '\n') {
        extra = 0;
        while (((ch = getchar()) != '\n') && (ch != EOF))
            extra = 1;
        return (extra == 1) ? TOO_LONG : OK;
    }

    // Otherwise remove newline and give string back to caller.
    buff[strlen(buff)-1] = '\0';
    return OK;
}

然后,您可以设置最大大小,它将检测到行上是否输入了太多数据,并刷新了行的其余部分,因此不会影响您的下一个输入操作。

您可以使用以下方法进行测试:

// Test program for getLine().

int main (void) {
    int rc;
    char buff[10];

    rc = getLine ("Enter string> ", buff, sizeof(buff));
    if (rc == NO_INPUT) {
        // Extra NL since my system doesn't output that on EOF.
        printf ("\nNo input\n");
        return 1;
    }

    if (rc == TOO_LONG) {
        printf ("Input too long [%s]\n", buff);
        return 1;
    }

    printf ("OK [%s]\n", buff);

    return 0;
}

在实践中,您不应该太费力地精确。 给自己一些时间来在堆栈上保留一些内存并对此进行操作。 一旦想进一步传递数据,就可以使用strdup(buffer)并将其放在堆上。 知道你的极限。 :-)

int main(int argc, char** argv) {
    char text[4096]; 
    char pattern[4096]; 
    fflush(stdin);
    printf(" enter a string\n");
    fgets(text, sizeof(text), stdin);

    printf(" enter a pattern\n");
    fgets(pattern, sizeof(pattern), stdin);

    int m=strlen(text);
    int n =strlen(pattern);
    printf(" text is %s %d  pattrn is %s %d \n",text,m,pattern,n);
    return (EXIT_SUCCESS);
}

您遇到的主要问题是具有未知大小的char数组。 只需在声明时指定数组大小即可。

int main(int argc, char** argv) {
    int s1[4096], s2[4096];
    fflush(stdin);
    printf(" enter a string\n");
    scanf("%s", s1);

    printf(" enter a pattern\n");
    scanf("%s", s2);

    int m = strlen(s1);
    int n = strlen(s2);
    printf(" text is %s of length %d, pattern is %s of length %d \n", s1, m, s2, n);
    return (EXIT_SUCCESS);
}

不要使用scanfgets来解决这个问题,因为正如您所说,没有真正的方法来知道输入将要持续多长时间。 而是将stdin作为最后一个参数使用fgets fgets允许您指定应读取的最大字符数。 您随时可以返回并阅读更多内容。

scanf(%s) gets读取,直到找到终止字符,并且该字符可能会超过缓冲区的长度,从而导致一些难以解决的问题。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM