C/C++——字符串分割(strtok, strtok_s)

1. 使用strtok分割字符串

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

2. 使用strtok_s分割字符串

由于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

3. 读取并分割txt中的数据

在编程中经常会实现将数据存入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;
}

 

你可能感兴趣的:(C++,C语言,c++,c语言,字符串)