读取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  #include  #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); } 

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

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

您的主要问题是具有未知大小的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); }