strtok()函数的原型如下:
char *__cdecl strtok(char *_String, const char *_Delimiter)
strtok()函数接收两个传入参数,将_String中保存的字符串(待处理字符串),按照_Delimiter中的字符作为分隔符进行分割。如果_String为空,则函数内部保存的SAVE_PTR指针在下一次调用中将作为起始位置。
其实这就说明在函数strtok()内部使用了一个静态变量SAVE_PTR指针,用以记录分割一次之后_String中下一个字符串的位置。这种方法导致了一个问题,就是strtok()函数是线程不安全的(因为其函数内部使用到了静态存储区)。
除此之外,从函数的定义,第一个传入参数_String定义为char*而不是const char*,就说明strtok()函数不保证不修改传入数据的内容。实际上,第一个参数_String传进来的字符串,是会被strtok()函数所修改的,因此调用strtok()函数的时候应当注意。
strtok()函数的一个示例:
#include
#include
#include
char string[] = "A string\tof ,,tokens\nand some more tokens";
char seps[] = " ,\t\n";
char *token=NULL;
int main(void){
printf( "Tokens:\n" );
token = strtok( string, seps);
while(token !=NULL){
printf("%s\n",token);
token=strtok(NULL,seps);//strtok第一个参数传入NULL,使用之前保存的SAVE_PTR定位下一个待处理的字符的位置
}
return 0;
}
上面代码的输出如下:
Tokens:
A
tokens
and
some
more
tokens
由于strtok()函数是线程不安全的,因此C语言对其进行了改进,改进后的函数命名为strtok_s()。
strtok_s()函数的原型如下:
char *__cdecl strtok_s(char *_String, const char *_Delimiter, char **_Context)
strtok_s()函数增加了一个参数_Context,这个参数就是相当于strtok()函数中内部定义的静态SAVE_PTR指针,用来传递对字符串_String的处理进行到了哪里。
一个使用strtok_s()函数的例子如下:
#include
#include
#include
int main()
{
char string[] = "A string\tof ,,tokens\nand some more tokens";
char seps[] = " ,\t\n";
char *token = NULL;
printf("Tokens:\n");
char* ptr = NULL;
token = strtok_s(string, seps, &ptr);//相较于strtok()函数,strtok_s函数需要用户传入一个指针,用于函数内部判断从哪里开始处理字符串
while (token != NULL) {
printf("%s\n", token);
token = strtok_s(NULL, seps, &ptr);//其他的使用与strtok()函数相同
}
return 0;
}
上述代码的输出如下:
Tokens:
A
tokens
and
some
more
tokens
在编程中经常会实现将数据存入txt文件,或者从txt文件中读取数据的功能。对于一些大量的,依靠特定分隔符分割的数据,我们可以使用strtok_s()函数对数据进行分割,然后保存下来。
一个简单的示例如下:
#include
#include
#include
#include
#include
#include
using namespace std;
struct Point {//构造一个Point类,用来存储点的信息(三坐标)
double x, y, z;
Point() :x(0), y(0), z(0) {}
Point(int a, int b, int c) :x(a), y(b), z(c) {}
void printPoints() { cout << "x:" << x << " y:" << y << " z:" << z << endl; }
};
int main()
{
string tmp;
ifstream inFile("test.txt");//txt文件名
vector points;
if (inFile) {
char string[100];
while (getline(inFile, tmp)) {
strcpy_s(string, tmp.c_str());
double a, b, c;
char seps[] = "|";
char* token = NULL;
char* ptr = NULL;
//存入x
token = strtok_s(string, seps, &ptr);
a = atof(token);
//存入y
token = strtok_s(NULL, seps, &ptr);
b = atof(token);
//存入z
token = strtok_s(NULL, seps, &ptr);
c = atof(token);
//将x,y,z构造point类,存入point数组中
Point pt(a, b, c);
points.push_back(pt);
}
}else {
cout << "No such file!" << endl;
}
for (auto pt : points) {//输出points所有点的坐标
pt.printPoints();
}
return 0;
}