c语言中strtok函数的使用与模拟实现

1.功能

功能
<cstring>
char * strtok ( char * str, const char * delimiters );
将字符串拆分为令牌

对此函数的一系列调用将 str 拆分为标记,这些标记是由分隔符中的任何字符分隔的连续字符序列。

在第一次调用时,该函数需要一个 C 字符串作为 str 的参数,其第一个字符用作扫描令牌的起始位置。在后续调用中,该函数需要一个空指针,并使用最后一个令牌末尾的位置作为扫描的新起始位置。

为了确定标记的开头和结尾,该函数首先从起始位置扫描分隔符中未包含的第一个字符(该字符将成为标记的开头)。然后从标记的开头开始扫描分隔符中包含的第一个字符,该字符将成为标记的末尾。如果找到终止 null 字符,扫描也会停止。

标记的这一端会自动替换为 null 字符,并且标记的开头由函数返回。

在对 strtok 的调用中找到 str 的终止 null 字符后,对此函数的所有后续调用(以 null 指针作为第一个参数)都将返回一个 null 指针。

找到最后一个令牌的点由下次调用时使用的函数保留在内部(不需要特定的库实现来避免数据争用)。

2.功能不用细看,实操理解一切

代码和运行结果:

#include<stdio.h>
#include<string.h>
int main()
{
	char arr[] = "a.b.cd@ef.1";
	const char* sep = "@.";
	char* ret = NULL;
	for (ret = strtok(arr, sep); ret != NULL; ret = strtok(NULL, sep))
	{
		printf("%s\n", ret);
	}
	return 0;
}

运行结果:

理解步骤:

1.也就是在arr中寻找含有sep中的字符(符号:@和符号:.),这里sep中的符号可以有一个或多个。

2.然后将对应位置重新设置为\0,这里\0的设置是一次循环设置一次。

3.最后一次循环(从左到右)返回一个拆分的值,上面的打印结果就是先返回a,然后返回b,然后……

3.模拟实现

代码

#include<stdio.h>
#include<string.h>
char* my_strtok(char* s, const char* dmt)
{
	static char* p = NULL;
	if(s)
	{
		p = s;
	}
	if (p == NULL)
		return NULL;
	char* left = p;
	while (*p)
	{
		for (int i = strlen(dmt) - 1; i >= 0; i--)
		{
			if (*p == *(dmt + i))
			{
				*p = '\0';
				p++;
				return left;
			}
		}
		p++;
	}
	p = NULL;
	return left;
}
int main()
{
	char arr[] = "a.b.cd@ef.1";
	const char* sep = "@.";
	char* ret = NULL;
	for (ret = my_strtok(arr, sep); ret != NULL; ret = my_strtok(NULL, sep))
	{
	    printf("%s\n", ret);
	}

	return 0;
}

运行结果与上述一致(调试与运行结果均相同)。

作者还测试了多打印几次的结果,如下:

都是相同的。

4.思路解析

将想要模拟实现的函数理解透彻(可以从调试和运行两方面推测它的运行原理)。

然后拆分步骤,我这里就是先处理第一个函数调用,也就是传入实参arr的情况下函数内部运作情况。

接着处理第二个函数调用,这里传入的实参为NULL,明显两次传入的参数不同,明显函数内部应该是使用了静态变量(或类似功能的),不然传入的参数变化那么大,功能怎能正常运行。

总之先写个大致框架,在一步一步去完善。比如,先写个循环,注释我这个循环要做到什么。

  • 6
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值