如何在C中正确使用strtok,以便没有内存泄漏?

问题描述 投票:11回答:5

我对在C中的char指针上调用strtok时会发生什么感到有些困惑。我知道它会修改字符串的内容,所以如果我在名为'line'的变量上调用strtok,它的内容将会改变。假设我遵循以下方法:

void function myFunc(char* line) {

    // get a pointer to the original memory block
    char* garbageLine = line;

    // Do some work
    // Call strtok on 'line' multiple times until it returns NULL
    // Do more work

    free(garbageLine);
}

进一步假设'line'在传递给myFunc之前被malloced。我应该在使用strtok之后释放原始字符串还是为我们完成工作?另外,如果'line'没有被malloced并且我尝试使用上面的函数会发生什么?取而代之的是更安全吗? (假设程序员知道该行未被malloced,则不会免费拨打电话)

调用

char* garbageLine = line;
myFunc(line);
free(garbageLine);

功能定义

void function myFunc(char* line) {
    // Do some work
    // Call strtok on 'line' multiple times until it returns NULL
    // Do more work
}
c malloc free strtok
5个回答
9
投票

strtok()不会释放任何东西,因为它不知道字符串的存储位置。它可能在堆栈或堆上,它不知道或关心! :)

取而代之的是更安全吗?

你的第二个例子要好得多,因为它简化了myFunc(),并且在更多情况下使它很有用,因为函数不需要知道字符串的分配位置。通过从myFunc()中删除对free()的调用,您可以使用该函数来解析堆栈或堆中的字符串。调用者分配内存,调用者释放内存!

进一步阅读:strtok()


4
投票

在你的问题的评论中,你说你“多次在'线'上调用strtok直到它返回NULL”。这听起来好像你可能错误地使用了strtok。第一次调用它时,你应该用'line'作为参数调用它;在后续调用中,您应该将其传递为NULL。以下面的例子为例:

void function myFunc(char* line) {
  char *segment; // This will point at each delimited substring in turn.

  segment = strtok(line, " ");

  // Do something with segment.

  segment = strtok(NULL, " ");

  // Do something with the new segment.

  free(line);
}

正如DrTwox所说,你的第二个例子更好 - 'line'应该被malloced它(或不是)的相同上下文释放,所以对free()的调用不属于这个函数。你最好还是循环它 - 比如:

void function myFunc(char* line) {
  char *segment;

  segment = strtok(line, " ");

  while (segment != NULL) {
    // Do something with segment.

    segment = strtok(NULL, " ");
  }
}

调用是这样的:

char *line = malloc(20*sizeof(char));

// Check that malloc succeeded here.
// Put some data into 'line'.

myFunc(line);

free(line);

// No 'garbageLine' required.

strtok的工作方式有点复杂,但你有重要的部分 - 它不会分配或释放任何内存。相反,它通过修改传递给它的字符串来工作。


1
投票

strtok没有比strlen更能释放内存了。你为什么期望它?它会释放什么记忆?也许你认为strtok需要释放内存,因为它存储了一个NUL,但内存的内容是无关紧要的。分配内存时,分配器会跟踪您分配的块的大小,并在释放时释放整个块。


1
投票

值得解释的是,strtok的工作是:

  1. 返回指向INTO原始字符串的指针;和
  2. 用NULL替换它找到的每个分隔符。

因此,一切都是就地的,它不需要分配任何内存。


0
投票

这与strtok()有什么关系?如果你分配内存,你需要释放它。您的应用程序决定分配和释放内存取决于您。但是如果你将内存传递给strtok(),那么只要内存被分配或释放就没有区别。

© www.soinside.com 2019 - 2024. All rights reserved.