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; }

 

 

 
           

 

 
          

 

 

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM