C語言string庫strcpy、strcmp、strcat函數(shù)的使用
C語言string庫strcpy、strcmp、strcat函數(shù)
strcpy
即string copy
語法格式為strcpy(str1, str2), 作用是將str2賦值給str1
使用方法類似于
char str1[10], str2[] = "abc"; strcpy(str1, "bcd"); strcpy(str1, str2); printf("%s", str1); // abc
str2可以是字符串, 也可以是字符串首地址(指針)。
strcpy會(huì)從傳入的地址開始寫入, 如代碼為strcpy(str1+1, str2)
,程序會(huì)從str1第二個(gè)元素開始寫入str2的值
值得注意的是, strcpy將指定內(nèi)容添加到字符串中后, 會(huì)在末尾添加一個(gè)空字節(jié)‘\0’, 以表示字符串結(jié)束。
如:
char str[5] = "abcde";
strcpy(str, "xyz");
因此,使用strcpy將某字符串的內(nèi)容賦予長度為n的字符串時(shí), 被復(fù)制的字符串長度最大為n-1(留一個(gè)位置給’\0’),否則會(huì)有溢出的
報(bào)錯(cuò)類似:builtin_memcpy’ writing 11 bytes into a region of size 10 overflows the destination 。
strcat
strcat(str1, str2)
將str2拼接到str1的末尾
char str[10] = "abc"; strcat(str ,"cde"); printf("%s", str); // abccde
使用strcat(str1, str2)時(shí), 程序會(huì)從str1第一個(gè)空字節(jié)開始將str2的內(nèi)容寫入, 并在末尾重寫空字節(jié)。
所以, str1中必須有足夠的空間來放入str2,即str1原先內(nèi)容后面至少要有strlen(str2)+1個(gè)字節(jié)。
strcmp
即string compare
用于比較兩個(gè)字符串。
規(guī)則是從兩個(gè)字符串第一個(gè)字符開始比較(ascii), 若相同則比較下一個(gè)字符,直到不同為止;若str1對應(yīng)位置的字符的ascii值小于str2的, 返回一個(gè)負(fù)整數(shù)(一般為-1, 取決于系統(tǒng), 有的會(huì)返回ascii碼的差值),反之返回一個(gè)正整數(shù)(1);若兩個(gè)字符串的長度和每個(gè)字符都相同, 則返回0。
參考:
C語言實(shí)現(xiàn)各類string函數(shù)
1.實(shí)現(xiàn)strcpy(字符串復(fù)制)
函數(shù)原型:char strcpy(char dest, const char src);
strcpy把含有’\0’結(jié)束符的字符串復(fù)制到另一個(gè)地址空間,返回值的類型為char。
代碼:
#include<stdio.h> #include<windows.h> #include<assert.h> char* my_strcpy(char *des, char const *stc) { ?? ?assert(des != NULL); ?? ?assert(stc != NULL); ?? ?char* res = des; ?? ?while (*stc){ ?? ??? ?*des = *stc; ?? ??? ?stc++; ?? ??? ?des++; ?? ?} ?? ?return res; } int main() { ?? ?char str1[100] = { 0 }; ?? ?char *str2 = "i am wangwenqian."; ?? ?my_strcpy(str1, str2); ?? ?printf("%s\n", str1); ?? ?system("pause"); ?? ?return 0; }
2.實(shí)現(xiàn)strcat
函數(shù)原型:extern char strcat(char dest, const char src);
功能:把src所指向的字符串(包括“\0”)復(fù)制到dest所指向的字符串后面(刪除dest原來末尾的“\0”)。要保證dest足夠長,以容納被復(fù)制進(jìn)來的src。*src中原有的字符不變。返回指向dest的指針。
說明:src和dest所指內(nèi)存區(qū)域不可以重疊且dest必須有足夠的空間來容納src的字符串。
代碼:
#include<stdio.h> #include<windows.h> #include<assert.h> char* my_strcat(char *des, char const *stc) { ?? ?char* ret = des; ?? ?assert(des != NULL); ?? ?assert(stc != NULL); ?? ?while (*des){ ? //des指向\0 ?? ??? ?des++; ?? ?} ?? ?while (*stc){ ?//拼接stc ?? ??? ?*des = *stc; ?? ??? ?des++; ?? ??? ?stc++; ?? ?} ?? ?return ret; } int main() { ?? ?char str1[100] = "abc"; ?? ?char *str2 = "cdefg"; ?? ?my_strcat(str1, str2);//str1實(shí)際傳址 ?? ?printf("%s\n", str1); ?? ?system("pause"); ?? ?return 0; }
3.實(shí)現(xiàn)strstr
函數(shù)原型:extern char *strstr(char *str1, const char *str2);
strstr(str1,str2) 函數(shù)用于判斷字符串str2是否是str1的子串。如果是,則該函數(shù)返回str2在str1中首次出現(xiàn)的地址;否則,返回NULL。
代碼:
#include<stdio.h> #include<windows.h> #include<assert.h> char* my_strstr(const char *str1, const char *str2) { ?? ?assert(str1); ?? ?assert(str2); ?? ?char *ret = (char *)str1; ?? ?char *res = (char *)str2; ?? ?while (*ret){ ?? ??? ?char cp = (char *)str1; ?? ??? ?if (*ret == *res && *res != '\0'){ ?//當(dāng)*ret與*res相等且*res不為0時(shí),倆個(gè)指針同時(shí)向后移動(dòng) ?? ??? ??? ?ret++; ?? ??? ??? ?res++; ?? ??? ?} ?? ??? ??? ?if (*res == "\0"){ ?//*res為0,說明滿足了上一個(gè)if條件,且找到字串 ?? ??? ??? ?return cp; ?? ??? ?} ?? ??? ?if (*ret == '\0'){ ?//*ret為\0,表示其一直向后移動(dòng),說明并沒有找到字串 ?? ??? ??? ?return NULL; ?? ??? ?} ?? ??? ?ret++; ?//讓ret一直向后移動(dòng) ?? ?} } int main() { ?? ?char *str1 = "question"; ?? ?char *str2 = "tion"; ?? ?char *ret = my_strstr(str1, str2); ?? ?printf("%p\n",ret ); ?? ?system("pause"); ?? ?return 0; }
4.實(shí)現(xiàn)strchr
函數(shù)原型:extern char *strchr(const char *s,char c)
可以查找字符串s中首次出現(xiàn)字符c的位置。
代碼:
#include<stdio.h> #include<stdlib.h> char * my_strchr(const char * str, char c) { ?? ?while (*str != '\0' && ?*str != c){ ?? ??? ?str++; ?? ?} ?? ?return str; } int main() { ?? ?char arr[] = "student"; ?? ?char c = 'u'; ?? ?char *ret = my_strchr(arr, c); ?? ?printf("%s\n", arr); ?? ?system("pause"); ?? ?return 0; }
5.實(shí)現(xiàn)strcmp
函數(shù)原型:extern int strcmp(const char *s1,const char *s2);
比較倆個(gè)字符串。若str1=str2,則返回零;若str1<str2,則返回負(fù)數(shù);若str1>str2,則返回正數(shù)。
#include<stdio.h> #include<windows.h> #include<assert.h> int strcmp(const char *str1, const char *str2) { ?? ?assert(str1); ?? ?assert(str2); ?? ?while (*str1 && (*str1==*str2)){ ?? ??? ?str1++; ?? ??? ?str2++; ?? ?} ?? ?if ((*(unsigned char *)str1) > (*(unsigned char *)str2)) ?? ??? ?return 1; ?? ?else if ((*(unsigned char *)str1) < (*(unsigned char *)str2)) ?? ??? ?return -1; ?? ?else ?? ??? ?return 0; } int main() { ?? ?char *str1 = "abcd"; ?? ?char *str2 = "abcdef"; ?? ?int ret = strcmp(str1, str2); ?? ?printf("%d\n", ret); ?? ?system("pause"); ?? ?return 0; }
6.實(shí)現(xiàn)memcpy
函數(shù)原型:void *memcpy(void *dest, const void *src, size_t n);
從源src所指的內(nèi)存地址的起始位置開始拷貝n個(gè)字節(jié)到目標(biāo)dest所指的內(nèi)存地址的起始位置中。
代碼:
#include<stdio.h> #include<stdlib.h> #include<assert.h> #define N 20 char * my_memcpy(char *des, char *stc, int len) { ?? ?assert(des); ?? ?assert(stc); ?? ?char *res = des; ?? ?while (len--){ ?? ??? ?*des = *stc; ?? ??? ?des++; ?? ??? ?stc++; ?? ?} ?? ?return res; } int main() { ?? ?char arr1[N] = ""; ?? ?char arr2[N] = "i am a student"; ?? ?my_memcpy(arr1, arr2, 6); ?? ?printf("%s\n",arr1); ?? ?system("pause"); ?? ?return 0; }
7.實(shí)現(xiàn)memmove
函數(shù)原型:void memmove( void dest, const void* src, size_t count );
memmove用于拷貝字節(jié),如果目標(biāo)區(qū)域和源區(qū)域有重疊的話,memmove能夠保證源串在被覆蓋之前將重疊區(qū)域的字節(jié)拷貝到目標(biāo)區(qū)域中,但復(fù)制后源內(nèi)容會(huì)被更改。但是當(dāng)目標(biāo)區(qū)域與源區(qū)域沒有重疊則和memcpy函數(shù)功能相同。
代碼:
#include<stdio.h> #include<stdlib.h> #define N 20 char *my_memmove(char *des, char *str, int len) { ?? ?void *ret = des; ?? ?//無內(nèi)存重疊 ?? ?if (des <= str || des >= str + len){ ? ?? ??? ?while (len--){ ?? ??? ??? ?*des = *str; ?? ??? ??? ?des++; ?? ??? ??? ?str++; ?? ??? ?} ?? ?} ?? ?else{ ?? ??? ?des += (len - 1); ?? ??? ?str += (len - 1); ?? ??? ?while (len--){ ?? ??? ??? ?*des = *str; ?? ??? ??? ?des--; ?? ??? ??? ?str--; ?? ??? ?} ?? ?} ?? ?return ret; } int main() { ?? ?char arr1[N] = "123456789"; ?? ?my_memmove(arr1+3, arr1, 5); ?? ?printf("%s\n", arr1); ?? ?system("pause"); ?? ?return 0; }?
memmove代碼比較難懂,建議大家在紙上畫一下。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- C語言中strcpy和strcat的使用和模擬實(shí)現(xiàn)
- 利用C語言模擬實(shí)現(xiàn)qsort,strcpy,strcat,strcmp函數(shù)
- C語言strlen,strcpy,strcmp,strcat,strstr字符串操作函數(shù)實(shí)現(xiàn)
- C語言詳細(xì)講解strcpy strcat strcmp函數(shù)的模擬實(shí)現(xiàn)
- 徹底掌握C語言strcat函數(shù)的用法
- C語言 模擬實(shí)現(xiàn)strcpy與strcat函數(shù)詳解
- C語言字符串函數(shù)操作(strlen,strcpy,strcat,strcmp)詳解
- strcat 函數(shù)的使用指南
- strcat函數(shù)實(shí)現(xiàn)簡單示例
- c++實(shí)現(xiàn)strcat字符串連接庫函數(shù)的方法詳解
- c++ 連接兩個(gè)字符串實(shí)現(xiàn)代碼 實(shí)現(xiàn)類似strcat功能
- C語言strcat函數(shù)詳解:字符串追加的利器
相關(guān)文章
C++中關(guān)于Crt的內(nèi)存泄漏檢測的分析介紹
本篇文章介紹了,在C++中關(guān)于Crt的內(nèi)存泄漏檢測的分析說明。需要的朋友參考下2013-04-04詳解C++中十六進(jìn)制字符串轉(zhuǎn)數(shù)字(數(shù)值)
這篇文章主要介紹了詳解C++中十六進(jìn)制字符串轉(zhuǎn)數(shù)字(數(shù)值)的相關(guān)資料,這里提供兩種實(shí)現(xiàn)方法,需要的朋友可以參考下2017-08-08詳解如何將Spire.XLS for C++集成到C++程序中
Spire.XLS for C++ 是一個(gè) Excel 庫,供開發(fā)人員在任何類型的 C++ 應(yīng)用程序中操作 Excel 文檔(XLS、XLSX、XLSB 和 XLSM)。 本文演示了如何以兩種不同的方式將 Spire.XLS for C++ 集成到您的 C++ 應(yīng)用程序中2023-03-03VisualStudio2019構(gòu)建C/C++靜態(tài)庫和動(dòng)態(tài)庫dll的問題 附源碼
這篇文章主要介紹了VisualStudio2019構(gòu)建C/C++靜態(tài)庫和動(dòng)態(tài)庫(dll)(文末附源碼),本文通過實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03C++函數(shù)指針+對象指針+this指針+指向類靜態(tài)和非靜態(tài)成員的指針
這篇文章主要介紹了C++函數(shù)指針+對象指針+this指針+指向類靜態(tài)和非靜態(tài)成員的指針,函數(shù)指針定義和賦值的語法指其中數(shù)據(jù)類型代表指向函數(shù)的返回類型,形參表為指向函數(shù)的形參表,更多相關(guān)資料需要的朋友可以參考一下下面文章內(nèi)容2022-03-03C/C++ 中sizeof(''a'')對比詳細(xì)介紹
這篇文章主要介紹了C/C++ 中sizeof('a')的值對比詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下2017-02-02kernel利用pt?regs劫持seq?operations的遷移過程詳解
這篇文章主要為大家介紹了kernel利用pt_regs劫持seq_operations進(jìn)行遷移的過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05