就字符串的拼接函數為例strcat.
原型:extern char *strcat(char *dest,char *src);
用法:#include <string.h>
功能:把src所指字符串添加到dest結尾處(覆蓋dest結尾處的'\0')並添加'\0'。
說明:src和dest所指內存區域不可以重疊且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;
}
參數是符合它的要求2個指針參數的,但是這個程序卻運行不了。開始百思不得其解,為什麼參數的類型都正確但是傳不回自己想要的結果。這樣只能去看函數原型了。
strcat函數原型
代碼如下:
char *strcat(char *strDest, const char *strScr) //將源字符串加const,表明其為輸入參數
{
char * address = strDest; //該語句若放在assert之後,編譯出錯
assert((strDest != NULL) && (strScr != NULL)); //對源地址和目的地址加非0斷言
while(*strDest) //是while(*strDest!='\0')的簡化形式
{ //若使用while(*strDest++),則會出錯,因為++是不受循環
strDest++; //約束的。所以要在循環體內++;因為要是*strDest最後指
} //向該字符串的結束標志'\0'。
while(*strDest++ = *strScr++) //是while((*strDest++ = *strScr++)!='\0')的簡化形式
{
NULL; //該循環條件內可以用++,
} //此處可以加語句*strDest='\0';有無必要?
return address; //為了實現鏈式操作,將目的地址返回
}
從這句話就知道為什麼了
代碼如下:
while(*strDest++ = *strScr++)
{
NULL;
}
如果strDest是一個指針,這裡的 *strDest 就是取一個未知地址的值,這個是編譯器不能容忍的。但是為什麼當strDest是一個數組的時候可以呢,因為數組等於給它分配連續地址。申請到的安全地址當然可以使用了。當然我們也可以寫一個傳入真正的以指針為參數的字符串拼接函數,下面是我自己寫的一個函數原型:
代碼如下:
char *strcatDemo2(char *str1, const char *str2) //將源字符串加const,表明其為輸入參數
{
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;
}
在這個裡面給指針address 申請了空間來存放2個字符串的東西,注意,這裡要多申請一個,因為字符串要求一個'\0'結尾。使用就這樣使用了:
代碼如下:
int main(int argc, char *argv[])
{
char *p = "hello, ", *s = "world!";
char *t = strcatDemo2(p, s);
puts(t);
system("PAUSE");
return 0;
}
上面寫的那個就類似於C#裡面字符串相加的功能了。
其實大多數C語言裡面的字符串都是,一個字符數組參數,一個字符指針參數來使用的。下面就是這些東西的原型,可以好好看下,避免以後犯錯。
strcat函數原型:
代碼如下:
char *strcat(char *strDest, const char *strScr) //將源字符串加const,表明其為輸入參數
{
char * address = strDest; //該語句若放在assert之後,編譯出錯
assert((strDest != NULL) && (strScr != NULL)); //對源地址和目的地址加非0斷言
while(*strDest) //是while(*strDest!='\0')的簡化形式
{ //若使用while(*strDest++),則會出錯,因為++是不受循環
strDest++; //約束的。所以要在循環體內++;因為要是*strDest最後指
} //向該字符串的結束標志'\0'。
while(*strDest++ = *strScr++) //是while((*strDest++ = *strScr++)!='\0')的簡化形式
{
NULL; //該循環條件內可以用++,
} //此處可以加語句*strDest='\0';有無必要?
return address; //為了實現鏈式操作,將目的地址返回
}
strcpy函數原型:
代碼如下:
char *strcpy(char *strDest, const char *strScr)
{
char *address=strDest;
assert((strDest != NULL) && (strScr != NULL));
while(*strScr) //是while(*strScr != '\0')的簡化形式;
{
*strDest++ = *strScr++;
}
*strDest = '\0'; //當strScr字符串長度小於原strDest字符串長度
return address; //時,如果沒有改語句,就會出錯了。
}
strcmp函數原型:
代碼如下:
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函數原型:
代碼如下:
int strlen(const char *str)
{
int len = 0;
assert(str != NULL);
while(*str++)
{
len++;
}
return len;
}