基于C語言字符串函數(shù)的一些使用心得
就字符串的拼接函數(shù)為例strcat.
原型:extern char *strcat(char *dest,char *src);
用法:#include <string.h>
功能:把src所指字符串添加到dest結(jié)尾處(覆蓋dest結(jié)尾處的'\0')并添加'\0'。
說明:src和dest所指內(nèi)存區(qū)域不可以重疊且dest必須有足夠的空間來容納src的字符串。
返回指向dest的指針。
舉例:
// strcat.c
#include <syslib.h>
#include <string.h>
main()
{
char d[20]="Golden Global";
char *s=" View";
clrscr();
strcat(d,s);
printf("%s",d);
getchar();
return 0;
}
上面的這段代碼可以沒有問題的輸出 Golden Global View。
但是這里有如果這樣改呢:
// strcat.c
#include <syslib.h>
#include <string.h>
main()
{
char *p="Golden Global";
char *s=" View";
clrscr();
strcat(p,s);
printf("%s",p);
getchar();
return 0;
}
參數(shù)是符合它的要求2個(gè)指針參數(shù)的,但是這個(gè)程序卻運(yùn)行不了。開始百思不得其解,為什么參數(shù)的類型都正確但是傳不回自己想要的結(jié)果。這樣只能去看函數(shù)原型了。
strcat函數(shù)原型
char *strcat(char *strDest, const char *strScr) //將源字符串加const,表明其為輸入?yún)?shù)
{
char * address = strDest; //該語句若放在assert之后,編譯出錯(cuò)
assert((strDest != NULL) && (strScr != NULL)); //對(duì)源地址和目的地址加非0斷言
while(*strDest) //是while(*strDest!='\0')的簡(jiǎn)化形式
{ //若使用while(*strDest++),則會(huì)出錯(cuò),因?yàn)?+是不受循環(huán)
strDest++; //約束的。所以要在循環(huán)體內(nèi)++;因?yàn)橐?strDest最后指
} //向該字符串的結(jié)束標(biāo)志'\0'。
while(*strDest++ = *strScr++) //是while((*strDest++ = *strScr++)!='\0')的簡(jiǎn)化形式
{
NULL; //該循環(huán)條件內(nèi)可以用++,
} //此處可以加語句*strDest='\0';有無必要?
return address; //為了實(shí)現(xiàn)鏈?zhǔn)讲僮?,將目的地址返?BR>}
從這句話就知道為什么了
while(*strDest++ = *strScr++)
{
NULL;
}
如果strDest是一個(gè)指針,這里的 *strDest 就是取一個(gè)未知地址的值,這個(gè)是編譯器不能容忍的。但是為什么當(dāng)strDest是一個(gè)數(shù)組的時(shí)候可以呢,因?yàn)閿?shù)組等于給它分配連續(xù)地址。申請(qǐng)到的安全地址當(dāng)然可以使用了。當(dāng)然我們也可以寫一個(gè)傳入真正的以指針為參數(shù)的字符串拼接函數(shù),下面是我自己寫的一個(gè)函數(shù)原型:
char *strcatDemo2(char *str1, const char *str2) //將源字符串加const,表明其為輸入?yún)?shù)
{
assert((str1 != NULL) &&(str2 != NULL));
char *address = (char *)malloc((strlen(str1) + strlen(str2) + 1) *sizeof(char));
char *des = address;
assert(address != NULL);
while(*str1)
{
*address = *str1;
str1++;
address++;
}
while(*str2)
{
*address = *str2;
str2++;
address++;
}
*address = '\0';
return des;
}
在這個(gè)里面給指針address 申請(qǐng)了空間來存放2個(gè)字符串的東西,注意,這里要多申請(qǐng)一個(gè),因?yàn)樽址笠粋€(gè)'\0'結(jié)尾。使用就這樣使用了:
int main(int argc, char *argv[])
{
char *p = "hello, ", *s = "world!";
char *t = strcatDemo2(p, s);
puts(t);
system("PAUSE");
return 0;
}
上面寫的那個(gè)就類似于C#里面字符串相加的功能了。
其實(shí)大多數(shù)C語言里面的字符串都是,一個(gè)字符數(shù)組參數(shù),一個(gè)字符指針參數(shù)來使用的。下面就是這些東西的原型,可以好好看下,避免以后犯錯(cuò)。
strcat函數(shù)原型:
char *strcat(char *strDest, const char *strScr) //將源字符串加const,表明其為輸入?yún)?shù)
{
char * address = strDest; //該語句若放在assert之后,編譯出錯(cuò)
assert((strDest != NULL) && (strScr != NULL)); //對(duì)源地址和目的地址加非0斷言
while(*strDest) //是while(*strDest!='\0')的簡(jiǎn)化形式
{ //若使用while(*strDest++),則會(huì)出錯(cuò),因?yàn)?+是不受循環(huán)
strDest++; //約束的。所以要在循環(huán)體內(nèi)++;因?yàn)橐?strDest最后指
} //向該字符串的結(jié)束標(biāo)志'\0'。
while(*strDest++ = *strScr++) //是while((*strDest++ = *strScr++)!='\0')的簡(jiǎn)化形式
{
NULL; //該循環(huán)條件內(nèi)可以用++,
} //此處可以加語句*strDest='\0';有無必要?
return address; //為了實(shí)現(xiàn)鏈?zhǔn)讲僮鳎瑢⒛康牡刂贩祷?BR>}
strcpy函數(shù)原型:
char *strcpy(char *strDest, const char *strScr)
{
char *address=strDest;
assert((strDest != NULL) && (strScr != NULL));
while(*strScr) //是while(*strScr != '\0')的簡(jiǎn)化形式;
{
*strDest++ = *strScr++;
}
*strDest = '\0'; //當(dāng)strScr字符串長(zhǎng)度小于原strDest字符串長(zhǎng)度
return address; //時(shí),如果沒有改語句,就會(huì)出錯(cuò)了。
}
strcmp函數(shù)原型:
int strcmp (const char *str1,const char *str2)
{
int len = 0;
assert((str1 != '\0') && (str2 != '\0'));
while(*str1 && *str2 && (*str1 == *str2))
{
str1++;
str2++;
}
return *str1-*str2;
}
strlen函數(shù)原型:
int strlen(const char *str)
{
int len = 0;
assert(str != NULL);
while(*str++)
{
len++;
}
return len;
}
相關(guān)文章
C語言堆與二叉樹的順序結(jié)構(gòu)與實(shí)現(xiàn)
堆是計(jì)算機(jī)科學(xué)中一類特殊的數(shù)據(jù)結(jié)構(gòu)的統(tǒng)稱,通常是一個(gè)可以被看做一棵完全二叉樹的數(shù)組對(duì)象。而堆排序是利用堆這種數(shù)據(jù)結(jié)構(gòu)所設(shè)計(jì)的一種排序算法。本文將詳細(xì)介紹堆與二叉樹的順序結(jié)構(gòu)與實(shí)現(xiàn),需要的可以參考一下2022-05-05C++ new與malloc和delete及free動(dòng)態(tài)內(nèi)存管理及區(qū)別介紹
這篇文章主要介紹了深入理解C++中的new/delete和malloc/free動(dòng)態(tài)內(nèi)存管理,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12C++封裝成DLL并調(diào)用的實(shí)現(xiàn)
本文主要介紹了C++封裝成DLL并調(diào)用的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03C++深入刨析優(yōu)先級(jí)隊(duì)列priority_queue的使用
最近我學(xué)習(xí)了C++中的STL庫中的優(yōu)先級(jí)隊(duì)列(priority_queue)容器適配器,對(duì)于優(yōu)先級(jí)隊(duì)列,我們不僅要會(huì)使用常用的函數(shù)接口,我們還有明白這些接口在其底層是如何實(shí)現(xiàn)的2022-08-08C語言實(shí)現(xiàn)員工工資管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)員工工資管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02實(shí)例講解C++編程中l(wèi)ambda表達(dá)式的使用
這篇文章主要介紹了C++編程中l(wèi)ambda表達(dá)式的使用實(shí)例,lambda表達(dá)式特性的引入在C++11中可謂千呼萬喚始出來,非常重要,需要的朋友可以參考下2016-01-01