您好,登錄后才能下訂單哦!
本篇文章為大家展示了C語言的一些重要字符串與內(nèi)存函數(shù)是怎樣的,內(nèi)容簡明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。
size_t strlen ( const char * str );
字符串以 ‘\0' 作為結(jié)束標(biāo)志,strlen函數(shù)返回的是在字符串中 ‘\0' 前面出現(xiàn)的字符個(gè)數(shù)(不包含 ‘\0' )。
參數(shù)指向的字符串必須要以 ‘\0' 結(jié)束。
注意函數(shù)的返回值為size_t,是無符號(hào)的。
模擬實(shí)現(xiàn)strlen
size_t my_strlen(const char*str) { size_t count=0; while(*str) { str++; count++; }
int strcmp ( const char * str1, const char * str2 );
從str1和str2指向的位置開始比較,如果遇到兩個(gè)不相等的字符或者\(yùn)0函數(shù)結(jié)束并且返回值。
1 第一個(gè)字符串的字符大于第二個(gè)字符串的字符返回 >0的數(shù)字。
2 第一個(gè)字符串的字符等于第二個(gè)字符串的字符返回 =0的數(shù)字。
3 第一個(gè)字符串的字符小于第二個(gè)字符串的字符返回 <0的數(shù)字。
模擬實(shí)現(xiàn)strcmp
int my_strcmp(const char*str1,const char*str2) { while (*str1 == *str2) { if (*str1 && *str2)//判斷'\0' return *s2 - *s1; *str1++; *str2++; }//循環(huán)結(jié)束標(biāo)志*str1!=*str2 return *str2 - *str1; }
char* strcpy(char * destination, const char * source)
拷貝source到destination中,返回一個(gè)指向dest的char* 的指針。
源字符串必須以 ‘\0' 結(jié)束。
會(huì)將源字符串中的 ‘\0' 拷貝到目標(biāo)空間。
目標(biāo)空間必須足夠大,以確保能存放源字符串。
目標(biāo)空間必須可變。
模擬實(shí)現(xiàn)strcpy
char *my_strcpy(char *dest, const char*src) { char *ret = dest; while((*dest++ = *src++)) { ; } return ret; }
char * strcat ( char * destination, const char * source );
從dest的末尾 ‘\0'開始添加src直到‘\0'
源字符串必須以 ‘\0' 結(jié)束。
目標(biāo)空間必須有足夠的大,能容納下源字符串的內(nèi)容。
目標(biāo)空間必須可修改。
模擬實(shí)現(xiàn)strcat
char* my_strcat(const char* str1, const char* str2) { while (*str1) str1++; while (*(char*)str1++ = *(char*)str2++) { ; } }
char * strstr ( const char *str2, const char * str1);
在str2中查找str1的字符串,如果找到了返回str2中這個(gè)字符串的首地址。如果找不到返回NULL。.0
char* my_strstr(const char* str1, const char* str2) { if (!*str2)//判斷字符串是否為空 return (char*)str1; char* ret1= (char*)str1;//將str類型轉(zhuǎn)換 char *cp = ret1; while (*ret1) { cp = ret1; char* ret2 =(char*) str2; while (*cp== *str2) {//遍歷字符串 if (!*ret2)//判斷str2是否到達(dá)‘0'的位置 return cp; cp++, ret2++; } ret1++;//循環(huán)結(jié)束沒有返回,從下一個(gè)字符開始查找 if (!*ret1) return NULL; } return NULL; }
void * memcpy ( void * destination, const void * source, size_t num );
dest復(fù)制src中num個(gè)字節(jié)的數(shù)據(jù)。
模擬實(shí)現(xiàn)memcpy
void * my_memcpy ( void * dst, const void * src, size_t count) { void * ret = dst; while (count--) { *(char *)dst = *(char *)src; dst = (char *)dst + 1; src = (char *)src + 1; } return(ret); }
·如果dst 和src指向同一個(gè)數(shù)組會(huì)發(fā)生什么?
int main() { int arr1[10] = { 1,2,3,4,5,6,7,8,9,0 }; // 預(yù)期結(jié)果 1 2 1 2 3 4 7 8 9 0 my_memcpy(arr1+2, arr1, 16); for (int i = 0; i < 10; i++) { printf("%d ", arr1[i]); } }
實(shí)際為
1 2 1 2 1 2 7 8 9 0
因?yàn)榈? 6 的時(shí)候3 4被改成了1 2 ,5 6也就被改成1 2。
也就是說被復(fù)制的元素在復(fù)制前被改變了,導(dǎo)致復(fù)制結(jié)果失敗。
如果是這樣指向同一個(gè)數(shù)組呢?
int main() { int arr1[10] = { 1,2,3,4,5,6,7,8,9,0 }; my_memcpy(arr1, arr1+2, 16); for (int i = 0; i < 10; i++) { printf("%d ", arr1[i]); } }
復(fù)制結(jié)果沒有問題。
對(duì)于這種情況,c語言有一個(gè)更強(qiáng)大的函數(shù)memmove.
void * memmove( void * destination, const void * source, size_t num );
與memcpy的功能一樣,但是memmove可以指向同一塊空間。
模擬實(shí)現(xiàn)memmove
**void* my_memmove(void* dest, void* src, size_t num) { char* ret = dest; //如果指向同一塊空間 判斷地址大小,避免數(shù)據(jù)在被復(fù)制前被改變 if ( (char*)dest-(char*)src< 0){ while (num) { *((char*)dest)++ = *((char*)src)++; num--; } } else { while(num--){ *((char*)dest+num) = *((char*)src+num); } } return ret; }
int memcmp ( const void * ptr1, const void * ptr2, size_t num );
從ptr1和ptr2的位置開始比較num個(gè)字節(jié),當(dāng)兩個(gè)字節(jié)數(shù)據(jù)不同時(shí)就會(huì)返回。
ptr1>ptr2 返回值>0;
ptr1=ptr2 返回值=0;
ptr1<ptr2 返回值<0;
與strcmp實(shí)現(xiàn)類似,這里就不實(shí)現(xiàn)了
上述內(nèi)容就是C語言的一些重要字符串與內(nèi)存函數(shù)是怎樣的,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。