a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 編程 > C > 正文

C語言切割多層字符串(strtok_r strtok使用方法)

2020-01-26 15:41:12
字體:
來源:轉載
供稿:網友

1. strtok介紹

眾所周知,strtok可以根據用戶所提供的分割符(同時分隔符也可以為復數比如“,。”)
將一段字符串分割直到遇到"/0".

比如,分隔符=“,” 字符串=“Fred,John,Ann”

通過strtok 就可以把3個字符串 “Fred”      “John”       “Ann”提取出來。

上面的C代碼為

復制代碼 代碼如下:

int in=0;
char buffer[]="Fred,John,Ann"
char *p[3];
char *buff = buffer;
while((p[in]=strtok(buf,","))!=NULL) {
i++;
buf=NULL; }

如上代碼,第一次執行strtok需要以目標字符串的地址為第一參數(buf=buffer),之后strtok需要以NULL為第一參數 (buf=NULL)。指針列p[],則儲存了分割后的結果,p[0]="John",p[1]="John",p[2]="Ann",而buf就變 成    Fred/0John/0Ann/0。

2. strtok的弱點

讓我們更改一下我們的計劃:我們有一段字符串 "Fred male 25,John male 62,Anna female 16" 我們希望把這個字符串整理輸入到一個struct,

復制代碼 代碼如下:

struct person {
char [25] name ;
char [6] sex;
char [4] age;
}

要做到這個,其中一個方法就是先提取一段被“,”分割的字符串,然后再將其以“ ”(空格)分割。
比如: 截取 "Fred male 25" 然后分割成 "Fred" "male" "25"
以下我寫了個小程序去表現這個過程:

復制代碼 代碼如下:

#include<stdio.h>
#include<string.h>
#define INFO_MAX_SZ 255
int main()
{
int in=0;
char buffer[INFO_MAX_SZ]="Fred male 25,John male 62,Anna female 16";
char *p[20];
char *buf=buffer;

while((p[in]=strtok(buf,","))!=NULL) {
buf=p[in];
while((p[in]=strtok(buf," "))!=NULL) {
in++;
buf=NULL;
}
p[in++]="***"; //表現分割
buf=NULL; }

printf("Here we have %d strings/n",in);
for (int j=0; j<in; j++)
printf(">%s</n",p[j]);
return 0;
}

這個程序輸出為:
Here we have 4 strings
>Fred<
>male<
>25<
>***<
這只是一小段的數據,并不是我們需要的。但這是為什么呢? 這是因為strtok使用一個static(靜態)指針來操作數據,讓我來分析一下以上代碼的運行過程:

紅色為strtok的內置指針指向的位置,藍色為strtok對字符串的修改

1."Fred male 25,John male 62,Anna female 16" //外循環

2."Fred male 25/0John male 62,Anna female 16" //進入內循環

3."Fred/0male 25/0John male 62,Anna female 16"

4."Fred/0male/025/0John male 62,Anna female 16"

5 "Fred/0male/025/0John male 62,Anna female 16" //內循環遇到"/0"回到外循環

6 "Fred/0male/025/0John male 62,Anna female 16" //外循環遇到"/0"運行結束。

3. 使用strtok_r

在這種情況我們應該使用strtok_r, strtok reentrant.
char *strtok_r(char *s, const char *delim, char **ptrptr);

相對strtok我們需要為strtok提供一個指針來操作,而不是像strtok使用配套的指針。
代碼:

復制代碼 代碼如下:

#include<stdio.h>
#include<string.h>
#define INFO_MAX_SZ 255
int main()
{
int in=0;
char buffer[INFO_MAX_SZ]="Fred male 25,John male 62,Anna female 16";
char *p[20];
char *buf=buffer;

char *outer_ptr=NULL;
char *inner_ptr=NULL;

while((p[in]=strtok_r(buf,",",&outer_ptr))!=NULL) {
buf=p[in];
while((p[in]=strtok_r(buf," ",&inner_ptr))!=NULL) {
in++;
buf=NULL;
}
p[in++]="***";
buf=NULL; }

printf("Here we have %d strings/n",in);
for (int j=0; j<in; j++)
printf(">%s</n",p[j]);
return 0;
}

這一次的輸出為:
Here we have 12 strings
>Fred<
>male<
>25<
>***<
>John<
>male<
>62<
>***<
>Anna<
>female<
>16<
>***<


讓我來分析一下以上代碼的運行過程:

紅色為strtok_r的outer_ptr指向的位置,
紫色為strtok_r的inner_ptr指向的位置,
藍色為strtok對字符串的修改

1. "Fred male 25,John male 62,Anna female 16" //外循環
2. "Fred male 25/0John male 62,Anna female 16"http://進入內循環
3.   "Fred/0male 25/0John male 62,Anna female 16"
4   "Fred/0male/025/0John male 62,Anna female 16"
5 "Fred/0male/025/0John male 62,Anna female 16" //內循環遇到"/0"回到外循環
6   "Fred/0male/025/0John male 62/0Anna female 16"http://進入內循環

原來, 該函數修改了原串.

所以,當使用char *test2 = "feng,ke,wei"作為第一個參數傳入時,在位置①處, 由于test2指向的內容保存在文字常量區,該區的內容是不能修改的,所以會出現內存錯誤. 而char test1[] = "feng,ke,wei" 中的test1指向的內容是保存在棧區的,所以可以修改

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 亚洲男人的天堂网站 | 久热精品视频在线播放 | 亚洲精品视频一区二区三区 | 久久国产精品一区 | 91福利在线播放 | 日韩欧美一区二区三区 | 欧美a一级| 黄色电影在线免费观看 | 精品久久毛片 | 黄av在线| 中文字幕国产视频 | 在线播放一区二区三区 | 午夜视频网址 | 91免费在线看 | 久草免费电影 | 国产成人精品午夜在线播放 | 看全黄大色黄大片老人做 | 免费色网址 | 96久久久久久| 国产色在线 | 中文字幕亚洲自拍 | 国产91精品一区二区麻豆网站 | 亚洲精品www久久久久久 | 日韩二区三区 | 欧美成人手机在线视频 | 久久精av| 国产区视频在线观看 | www.一区| 欧美男人天堂网 | 一本一道久久a久久精品蜜桃 | 日本高清视频在线播放 | 91精品国产综合久久久久久漫画 | 欧美婷婷| 久久亚洲一区二区三区四区五区高 | 99久久精品免费看国产免费软件 | 精品久久久久久久久久久院品网 | 少妇被艹视频 | 日韩永久精品 | 国产免费av大片 | 91精品一区二区三区久久久久久 | 草比网站 |