概述
- 函数的原型:
char* strtok : strtok (char* _String, char const* _Delimiter);
char* strtok_s: strtok_s( char* _String, char const* _Delimiter, char** _Context);- 函数的参数:
_String : 传入一个字符串
_Delimiter: 传入一个字符字符串,表示分隔符
_Context: 指向用于存储解析器内部状态的内存,以便下次调用strtok_s时解析器可以从中断处继续。-- 最开始传入一个NULL的指针地址- 函数的作用:
将第一个参数传入的字符串,根据第二个参数指定的分隔符拆解开。- 函数的返回值:
返回一个字符指针,指向当前根据字符串中指定的分隔符拆解的字符串,如果没有分隔符那么就返回NULL。- 基本原理
我们每次调用函数,其就会将字符串中与指定分割符匹配的字符切换成空字符。
此方法会拆解传入的第一个参数,所以我们传入的字符串不能为字符串常量或者被const修饰的字符串- 头文件
需要包含头文件#include <string.h>
例子
#include <iostream>
#include <string.h>
#include <stdlib.h>
int main(void) {
char arr[] = "1,2,3,4,5";
char* context = NULL;
char* ch = strtok_s(arr,",",&context);
std::cout << *ch << std::endl;
system("pause");
return 0;
}
- 输出结果:
上面代码的输出结果为 1。- 原因:
strtok_s()中三个参数:
arr -- 表示需要拆解的字符串,
"," -- 表示根据arr中的哪个字符进行拆解,
&context -- 表示一个指针的地址,它是在底层用来记录我们拆解到哪个位置的,我们使用的时候只需要定义一个指向NULL的指针,传入其地址就行。- 注意:
1. 第一个参数不能直接传入字符串常量,即使你将其强制转换为char*,(比如: (char*)"1,2,3,4,5") 也是不行的,因为字符串常量的值我们是不能修改的。
2. 如果我们使用c++的字符串类string,需要在传入参数的时候转化成C语言的字符串, 调用函数c_str()。c_str()返回的是const char*类型的字符串,我们需要强制转换为 char*。
我们使用strtok_s函数将字符串"1,2,3,4,5"按照分隔符","进行拆解,最后的结果应该是分解为5个字符串,但是我们调用一次 strtok_s() 只会返回一个(第一次调用返回的就是拆解后第一个字符串),所以我们需要使用循环,返回获取其它的字符串。
#include <iostream>
#include <string.h>
#include <stdlib.h>
int main(void) {
char arr[] = "1,2,3,4,5";
char a[] = ",";
char* context = NULL;
char* ch = strtok_s(arr,a,&context);
std::cout << *ch << std::endl;
while (1) {
ch = strtok_s(NULL,a,&context);
if (ch == NULL) {
break;
}
std::cout << *ch << std::endl;
}
system("pause");
return 0;
}
结果:
- 解析
当我们调用一次strtok_s()函数,此函数就会根据我们指定的分隔符,拆解出第一个字符串,然后将下次拆解的位置记录在第三个参数中。
我们通过多次调用strtok_s()函数实现了对字符串中以指定分隔符相连接的字符串进行拆解。
对于第一个参数,我们只需要第一次调用的时候传入目标字符串就行,因为第一次调用之后,函数将下次拆解的位置记录在了第三个参数中,所以当我们对字符串进行第二次拆解的时候,就不需要在第一个参数的位置再传入字符串了(直接写NULL),其它的参数不变。(第三个参数必须和第一次调用使用的那个指针)
所以,使用strtok_s()根据分隔符对字符串进行拆解,需要调用多次才行,但是第二次及以后就不需要传入字符串了。(前提是拆分的还是之前的字符串)
然后我们根据函数的返回值,就能获取到每次拆解之后的字符串。
当字符串拆解结束,就是字符串中已经没有分割符了,函数会返回一个NULL。(我们也是通过此返回值来判断什么时候全部拆解结束,然后跳出循环的)。
注意: 如果函数已经返回了NULL,我们必须直接退出,停止拆解,否则会导致程序终止,因为NULL对应的内存我们没有权限。
strtok()和strtok_s()的区别
和scanf和scanf_s的关系是一样的,vs认为strtok()不安全, strtok_s()只是vs中使用的一个安全版本而已。
虽然上面没有给出strtok()的使用实例,其实两者是一样的,只是strtok_s()多一个参数,其它用法都是一样的,只要去掉第三个参数就是strtok()的用法。