我曾经考虑过第一次调用时strtok_s()
应该将包含令牌的字符串作为第一个参数传递,如下代码:
char testString[100] = "1|2|3";
char *context = testString;
const char *token = strtok_s( testString, "|", &context );
while ( token )
token = strtok_s( NULL, "|", &context );
但是,我看到有人总是将第一个参数保留为NULL
,例如以下代码:
char testString[100] = "1|2|3";
char *context = testString;
const char *token = strtok_s( NULL, "|", &context );
while ( token )
token = strtok_s( NULL, "|", &context );
我知道它的工作原理以及它是如何工作的。因为context
指向与testString
. 但我觉得有点奇怪,我的疑问是:
strtok_s()
吗?它可能面临哪些潜在的错误?strtok_s()
还需要保留第一个参数?它可以NULL
像往常一样,不是吗?根据函数文档,该函数的正确用法是您提到的第一个。
进一步引用 C11 标准(强调我的),第 K.3.7.3.1 节(第 616 页):
对 strtok_s 函数的调用序列将 s1 指向的字符串分解为一系列标记,每个标记都由 s2 指向的字符串中的一个字符分隔。第四个参数指向调用者提供的 char 指针,strtok_s 函数在其中存储它继续扫描同一个 string 所需的信息。
序列中的第一个调用有一个非空的第一个参数,s1max 指向一个对象,该对象的值是第一个参数指向的字符数组中的元素数。第一次调用在 ptr 指向的对象中存储一个初始值,并更新 s1max 指向的值以反映与 ptr 相关的剩余元素数。序列中的后续调用的第一个参数为 null,并且 s1max 和 ptr 指向的对象需要具有序列中前一个调用存储的值,然后更新这些值。s2 指向的分隔符字符串可能与调用不同。
因此,标准所说的正确用法是strtok_s
使用非 NULL 第一个参数调用,然后使用 NULL 第一个参数调用它。在第一次调用时,该函数初始化一些状态,并使用提供的指针(最后一个参数)来存储它。
该标准没有提到应该如何使用最后一个参数,而不是保持状态,以便在使用未修改的指针调用时函数可以继续搜索相同的字符串。基本上,它消除了对 的内部状态的需要strtok
,例如,您可以同时标记多个字符串。
因此,状态空间的使用方式是实现定义的。很可能是这样,在某些实现中,它可以简单地将初始字符串放在那里,并始终使用第一个参数 NULL 调用它,如您所示。但是不能保证这会发生在所有实现上,或者这种行为在库的未来版本中将保持不变。
直接回答您的问题,是的,它可能有效,但不,这样做不是一个好主意。
本文收集自互联网,转载请注明来源。
如有侵权,请联系 [email protected] 删除。
我来说两句