- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试为用户输入重新创建一个简单的字符串解析器。我可以毫无问题地获取输入字符串,并确定其中有多少个单独的字符串,以便正确地为 malloc 提供正确的大小。最初,我没有在标准输入中找到字符串的数量,只是重新分配我的字符串数组,为另一个字符串留出空间,然后 malloc 新的字符串。现在我已经有了大小,我相信这是没有必要的,因为我只需为 malloc 提供适当的大小即可。
(尝试使用 calloc(),在同一位置出现段错误)
令我困惑的是,当我在代码中保留 realloc 时,没有任何问题(除了知道它可能会以不期望的方式调整我的数组之外)。但是,如果我删除 realloc,我的代码会在 fgets() 尝试执行时出现段错误。
这就是我调用解析器和我的#includes 的地方。 “appserver.h”保存所有原型(prototype)和 std(lib/bool/io),string/unistd/limits.h
#include "appserver.h"
int main(int argc, char *argv[])
{
if(argc != 4)
{
exit(0);
}
struct userInput user = { parseInt(*argv[1]), parseInt(*argv[2]), argv[3] };
printf("> ");
char **newArgs = stringParser();
for (int i = 0; newArgs[i] != NULL; i++)
{
free(newArgs[i]);
}
free(newArgs);
}
下面是我的解析器代码。如果我不注释 realloc,我可以毫无问题地单步执行我的代码(在 gdb 中单步执行会显示来自 stdin 的整个字符串的“输入”),但是,如果我将其注释掉,那么当我尝试在 fgets( 中检索用户输入时,我会出现段错误) )。为什么注释掉 realloc() 甚至会对 fgets() 产生影响?
char **stringParser()
{
char *input;
fgets(input, INT_MAX, stdin);
int size = numberOfStrings(input);
char **inputArray = malloc((size)*(sizeof(char*)));
inputArray[0] = malloc(sizeof(char*));
strcpy(inputArray[0], strtok(input, " "));
for(int i = 1; i < size /*inputArray[i-1] != NULL*/; i++)
{
// inputArray = realloc(inputArray, (i+1)*sizeof(char*));
inputArray[i] = malloc(sizeof(char*));
strcpy(inputArray[i], strtok(NULL, " "));
printf("Inside inputArray[%d]: %s\n", i-1, inputArray[i-1]);
}
return inputArray;
}
这是我的 numberOfStrings() 方法的代码,如果这可能值得研究,但是我也使用 gdb 逐步完成了它,它看起来非常具体。
int numberOfStrings(char *input)
{
int count = 0;
char *tempCopy = malloc(sizeof(char*));
strcpy(tempCopy, input);
char* token = strtok(tempCopy, " ");
while(token != NULL)
{
token = strtok(NULL, " ");
count++;
}
free(tempCopy);
return count;
}
<小时/>编辑:我想跟进以确保我已经避免了大多数潜在的未定义行为。我没有更改 main 中的任何内容,因此这是我对
**stringParser()
的更改
char **stringParser()
{
char *input = nextLine(stdin);
int size = numberOfStrings(input, strlen(input));
char **inputArray = calloc(size, sizeof(*inputArray));
char *token = strtok(input, " ");
inputArray[0] = malloc(strlen(token));
strcpy(inputArray[0], token);
for(int i = 1; i < size - 1; i++)
{
token = strtok(NULL, " ");
inputArray[i] = malloc(strlen(token));
strcpy(inputArray[i], token);
}
free(input);
inputArray[size-1] = (char*)NULL;
return inputArray;
}
主要变化领域是:
calloc()
而不是 malloc()
,但如果我的理解正确的话 calloc(size, sizeof(*inputArray))
code> 的行为与 malloc((size)*sizeof(*inputArray)) 相同(忽略“0”或垃圾值初始化)
*nextLine(FILE *input)
,它允许我通过 stdin
逐个字符地扫描,直到到达 EOF
或 '\n'
。这样做的好处是可以准确分配用户输入所需的内存量。sizeof(char*)
相反)count++
。这些变化让我不再有任何内存泄漏。我不敢假设我已经避免了所有未定义行为的行为,但我想我在这一点上做得更好一点。我确保不会用一堆其他问题来堵塞这篇文章,但我想把这个更新留在这里,以防有人发现它有用或相关。
最佳答案
对 realloc()
的调用不会影响 fgets()
- 至少不会直接、可预测或可靠地影响。
在 stringParser()
中第一次调用 fgets(input, INT_MAX, stdin)
具有未定义的行为,因为指针 input
未初始化。
实际上,fgets()
的调用可能会覆盖一些不应该覆盖的内存区域。
通过添加(或注释掉)realloc()
的调用,结果将是对程序使用的内存布局进行一些调整(例如,更改代码或数据是否位于内存位置被 fgets()
覆盖。
但这并不是因为realloc()
直接影响fgets()
。这只是调用 realloc()
改变程序中某些内容这一事实的副作用。
注释掉 realloc()
调用或重新插入它的效果可以是任何东西。例如,如果您的代码是使用不同的编译器设置(例如优化标志)甚至不同的编译器构建的,则可能会得到不同的效果。
要消除 fgets()
的问题,请向其传递一个数组
char input[some_positive_value];
fgets(input, sizeof(input), stdin);
或初始化指针以指向合适的缓冲区。
char *input = malloc(some_positive_value); /* remember to `free()` when done */
fgets(input, some_positive_value, stdin);
请注意,如上所述,循环确实需要 realloc()
调用
// inputArray = realloc(inputArray, (i+1)*sizeof(char*));
inputArray[i] = malloc(sizeof(char*));
如果没有 realloc()
调用,对 inputArray[i]
的赋值也将具有未定义的行为。如果您还没有看到这种症状,那么您只是很幸运 - 再说一次,这是一种您不能依赖的影响。
检查各种函数(fgets()
、realloc()
等)是否实际成功也是一个好主意。您的代码基于以下假设继续进行:所有函数均按预期工作,但如果它们失败,则行为将是未定义的(例如,如果 realloc()
的内存重新分配失败)。
关于c - 重新分配影响 fgets,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46822799/
我有以下代码:(cpp14) static int const max_len = 1000; FILE* m_in_log = NULL; FILE* m_log = NULL; ... ... b
标准库函数 fgets() 有两个缺点: 函数的第二个参数是int类型 它在提供的缓冲区中留下一个尾随的换行符 我做了一个类似于 fgets() 的简单函数,排除了上述缺点,试图提高我的一个程序的效率
我看过各种关于在 fgets 之前使用 scanf 可能会导致问题的帖子。但是,对我来说,为什么在 AFTER 之后添加它会导致第一个语句中断,这是没有意义的。 这就是我所拥有的: unsigne
这可能是一个非常愚蠢的问题(可能),但我正在使用 fgets 处理文本文件。没什么太花哨的: while (fgets(buf, sizeof(buf), inputFile) != NULL) 正如
如果我想使用 fgets 从文本中读取多行,根据我的教科书,我会这样做: char str[53]; ... while(fgets(str, max, f)!=NULL){ ... }
我试图在完成未定义数量的另一个输入后获取用户输入。但问题是 while 循环之后的第二个 fgets 永远不会被调用。我用 EOF 结束循环,也许这就是错误。但我不知道应该如何结束循环。 另一个有趣的
我正在尝试将文件中的邮政编码读入 Object * 数组。文件包括123 无处不在柯克兰加州99223 我的 .h 文件看起来像 typedef struct { char *street;
我已经实现了我自己的 fgets(即 myfgets)。当我的文件中有必须由 myfgets 函数读取的 NULL 字符串时,它会打印所有字符串(好)但有一些垃圾(坏),但如果我使用预定义 fgets
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: Why does printf not flush after the call unless a newl
假设 FILE* 有效,请考虑: char buf[128]; if(fgets(buf,sizeof buf,myFile) != NULL) { strlen(buf) == 0; //ca
void verification() { char pass[50]; printf(" Enter Password : "); fgets(pass, 50, stdin
我想知道如何从汇编代码中调用fgets。我读到的这些问题与这个问题完全相同:How to call fgets in x86 assembly?这个:How to use c library func
我有一个程序,用户可以在其中输入两个输入。由于我无法控制用户输入的内容,因此用户可以超过数组的固定大小。由于 fgets() appends 在空字符之前保留了一个换行符,如果用户超出预期大小时换行符
我想使用 fgets() 简单地读取一行并去掉换行符。这是否适用于所有情况,包括 Windows 和 UN*X? fgets(buf, sizeof buf, stdin); strtok(buf,
我正在运行以下代码: #include #include #include int main(){ FILE *fp; if((fp=fopen("test.txt","r"))==N
如果我有以下缓冲区: char buffy[40]; // includes \0 at the end fgets 函数应该有 STLEN 40 还是 39?为什么? char buffy[40];
使用fgets输入字符串,我对读取的字符串长度有疑问。 例如,考虑以下程序。 char str[50]; int i; int len; printf("Enter name:\n"); fgets(
fgets()函数是否将文件指针自动移动到该位置,直到我提到的大小参数为止? 例如 : p.txt文件的内容是“我是个好男孩”。使用fgets(a,5,fp1)之后 文件指针会向前移动5个位置吗? 在
gcc 4.4.4 c89 我正在读取一个文件,在该文件中,我必须对每一行进行一些操作。 在一个循环中,我调用了 2 个函数,它们传入此文件指针以对文件执行 2 种不同的操作。 但是,当我使用 fge
我试图从文件中获取我的 9x9 数字网格并将其存储在 2D int 数组中。为此,我编写了代码,以便它使用文件中的 fgets(),然后将 char 数组值转换为 int,然后将其存储在数组中的相应位
我是一名优秀的程序员,十分优秀!