C语言拾遗——strtok

C语言拾遗——strtok

今天刷PAT的时候用到了这个strtok函数,顺手就记录一下

strtok函数包含于头文件string.h

语法:char *strtok( char *str1, const char *str2 ); 

功能:函数返回字符串str1中紧接“标记”的部分的指针, 字符串str2是作为标记的分隔符。如果分隔标记没有找到,函数返回NULL。为了将字符串转换成标记,第一次调用str1 指向作为标记的分隔符。之后所以的调用str1 都应为NULL

下面是测试代码

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

int main(void)
{
	char p[] = "My name is zzy.";
	char tmp[20];
	char *sep = " ";
	char *token = NULL;

	strcpy(tmp, p);
	
	for (token = strtok(tmp, sep); token != NULL; token = strtok(NULL, sep))
	{
		printf("%s\n", token);
	}
	
	return 0;
}

运行结果:

讨论一下为什么第二个strtok的第一个参数一定为NULL

strtok的原型如下:

_CRTIMP char* __cdecl __MINGW_NOTHROW	strtok (char*, const char*);

这个函数涉及两个指针,除了形参之外

第一个指针 pointer_a 用来指向函数返回的字符串,这个字符串是被原字符串OriginalString被seps中的字符截断后的第一个字符串。

第二个指针 pointer_b 用来指向OriginalString中,匹配截断字串seps的位置

如果在第二次函数调用的时候,第一个参数不是设置为NULL而是设为原来的字符串”My name is zzy.”,那么讲永远返回”My”并且也将陷入死循环。
为何会出现这种结果呢?

其实认真想想就很清楚啦。

第一次传入字符串”My name is zzy.” 中匹配token(即参数seps代表的各种字符)中所指定的字符的位置在My和name之间, 函数的返回值pointer_a为一个指向”My”的指。

指针此时,pointer_b指向My和name之间的位置,代表pointer_b之前的位置已经进行过查找匹配了。

这样,在循环体之内进行第二次函数调用的时候,只要把strtok()的第一个参数设置为NULL,就可以直接从pointer_b位置开始进行查找匹配了。 pointer_b 会保持之前的位置一直指下去

所以,NULL的作用只是为了使得每次调用时,都不是从”My name is zzy.”的头开始,而是从上次调用时查找所停止的位置开始的啦。如此循环下去,直到无法再找到匹配token的位置的时候,这样就实现了把”My name is zzy”按照token进行分隔的效果啦O(∩_∩)O

posted @ 2020-02-10 20:09  秦_殇  阅读(296)  评论(0编辑  收藏  举报