strtok(), strtok_s() 字符串分割函数

strtok(), strtok_s() 字符串分割函数

 //strtok()函数原型
 /*_Check_return_ _CRT_INSECURE_DEPRECATE(strtok_s) _CRTIMP 
                      char *  __cdecl strtok(_Inout_opt_z_ char * _Str, _In_z_ const char * _Delim);*/


当strtok()在参数_Str的字符串中发现参数_Delim中包涵的分割字符时,则会将该字符改为\0 字符。

在第一次调用时,strtok()必需给予参数_Str字符串,往后的调用则将参数_Str设置成NULL。每次调用成功则返回指向被分割出片段的指针。

需要注意的是,使用该函数进行字符串分割时,会破坏被分解字符串的完整,调用前和调用后的s已经不一样了。

第一次分割之后,原字符串str是分割完成之后的第一个字符串,剩余的字符串存储在一个静态变量中。

复制代码

//将字符串"ab,cde,fghi"按照","分割。
char str[] = "ab,cde,fghi";
char *p[4];
p[0]= strtok(str, ",");
int i = 0;
while (p[i] != NULL)
{
    i++;
    p[i]= strtok(NULL, ",");
}
p[i] = "\0";
for (int i = 0; i < 4; i++)
{
    cout << p[i] << endl;
}

复制代码

 

 

//strtok_s()函数原型
/*_Check_return_ _CRTIMP_ALTERNATIVE 
    char *  __cdecl strtok_s(_Inout_opt_z_ char * _Str, _In_z_ const char * _Delim, _Inout_ _Deref_prepost_opt_z_ char ** _Context);*/

仍然会破坏原字符串的完整性。但是函数将剩余的字符串存储在_Context变量中,而不是静态变量中,从而保证了安全性。

char ** _Context参数是一个指向char *的指针变量,用来在strtok_s内部保存切分时的上下文,以应对连续调用分解相同源字符串。 

strtok函数在提取字符串时使用了静态缓冲区,因此,线程不安全。如果要顾及到线程的安全性,应该使用strtok_s。

strtok_s实际上就是将strtok内部隐式保存的this指针,以参数的形式与函数外部进行交互。由调用者进行传递、保存甚至是修改。

需要调用者在连续切分相同源字符串时,除了将str参数赋值为NULL,还要传递上次切分时保存下的_Context。

复制代码

//将字符串"hello world,my name is congcong."按照","分割。
char str[] = "hello world,my name is congcong.";
char *p[3];
char *buf;
p[0]=strtok_s(str, ",",&buf);
int i = 0;
while (p[i])
{
    i++;
    p[i] = strtok_s(NULL, ",", &buf);
}
p[i] = "\0";
for (int i = 0; i < 3; i++)
{
    cout << p[i] << endl;
}

复制代码

 连续调用分解相同源字符串时,strtok_s()函数更加直观的体现了其便捷性。

复制代码

//将字符串"hello world,my name is congcong."按照","和" "分割。
//先按","分割,在每个字串里面按照" "分割。双重循环过程中,需记录出剩下的串。
char str[] = "hello world,my name is congcong.";
char *p[8];
char *buf;
char *buftmp;
char *strtmp = new char[strlen(str)+1];
strcpy(strtmp, str);
//p[0] = strtok_s(strtmp, ",", &buf);
int i = 0;
while (p[i] = strtok_s(strtmp, ",", &buf))
{
    strtmp = p[i];
    //p[i] = strtok_s(strtmp, " ", &buftmp);
    while (p[i] = strtok_s(strtmp, " ", &buftmp))
    {
        i++;
        //p[i] = strtok_s(NULL, " ", &buftmp);
        strtmp = NULL;
    }
    //p[i] = strtok_s(NULL, ",", &buf);
    strtmp = NULL;
}
//p[i] = "\0";
for (int j = 0; j < i; j++)
{
    cout << p[j] << endl;
}

你可能感兴趣的:(c++,STL)