c語(yǔ)言重要的字符串與內(nèi)存函數(shù)
一.字符串函數(shù)
1. 求字符串長(zhǎng)度的strlen
size_t strlen ( const char * str );
- 字符串以 ‘\0' 作為結(jié)束標(biāo)志,strlen函數(shù)返回的是在字符串中 ‘\0' 前面出現(xiàn)的字符個(gè)數(shù)(不包含 ‘\0' )。
- 參數(shù)指向的字符串必須要以 ‘\0' 結(jié)束。
- 注意函數(shù)的返回值為size_t,是無(wú)符號(hào)的。
模擬實(shí)現(xiàn)strlen
size_t my_strlen(const char*str)
{
size_t count=0;
while(*str)
{
str++;
count++;
}
2.比較字符串大小的strcmp
int strcmp ( const char * str1, const char * str2 );
- 從str1和str2指向的位置開始比較,如果遇到兩個(gè)不相等的字符或者\(yùn)0函數(shù)結(jié)束并且返回值。
- 第一個(gè)字符串的字符大于第二個(gè)字符串的字符返回 >0的數(shù)字。
- 第一個(gè)字符串的字符等于第二個(gè)字符串的字符返回 =0的數(shù)字。
- 第一個(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;
}
3.復(fù)制字符串的strcpy
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;
}
4.追加字符串的strcat
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++) {
;
}
}
5.查找字符串函數(shù)的strstr
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;
}
二、內(nèi)存函數(shù)
1.復(fù)制 memcpy,memmove
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。
也就是說(shuō)被復(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é)果沒有問(wèn)題。
對(duì)于這種情況,c語(yǔ)言有一個(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;
}
2.比較 memcmp
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;
到此這篇關(guān)于c語(yǔ)言重要的字符串與內(nèi)存函數(shù)的文章就介紹到這了,更多相關(guān)c語(yǔ)言的字符串與內(nèi)存函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C語(yǔ)言的字符串函數(shù),內(nèi)存函數(shù)筆記詳解
- C語(yǔ)言內(nèi)存函數(shù)的使用及其模擬實(shí)現(xiàn)
- C語(yǔ)言詳解如何應(yīng)用模擬字符串和內(nèi)存函數(shù)
- C語(yǔ)言超詳細(xì)講解字符串函數(shù)和內(nèi)存函數(shù)
- C語(yǔ)言深入詳解四大內(nèi)存函數(shù)的使用
- C語(yǔ)言字符串函數(shù),字符函數(shù),內(nèi)存函數(shù)使用及模擬實(shí)現(xiàn)
- 深入了解C語(yǔ)言中的字符串和內(nèi)存函數(shù)
- 深度解析三個(gè)常見的C語(yǔ)言內(nèi)存函數(shù)
- C語(yǔ)言內(nèi)存函數(shù)的實(shí)現(xiàn)示例
- C語(yǔ)言內(nèi)存函數(shù)的具體使用
- C語(yǔ)言實(shí)現(xiàn)內(nèi)存函數(shù)的示例代碼
相關(guān)文章
C++中opencv4.1.0環(huán)境配置的詳細(xì)過(guò)程
這篇文章主要介紹了C++中opencv4.1.0環(huán)境配置的詳細(xì)過(guò)程,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-10-10
深入了解C++優(yōu)先隊(duì)列(priority_queue)的使用方法
在計(jì)算機(jī)科學(xué)中,優(yōu)先隊(duì)列是一種抽象數(shù)據(jù)類型,它與隊(duì)列相似,但是每個(gè)元素都有一個(gè)相關(guān)的優(yōu)先級(jí)。C++中的優(yōu)先隊(duì)列是一個(gè)容器適配器(container adapter),它提供了一種在元素之間維護(hù)優(yōu)先級(jí)的方法。本文帶你深入了解C++優(yōu)先隊(duì)列的使用方法,需要的可以參考下2023-05-05
C++基于socket編程實(shí)現(xiàn)聊天室功能
這篇文章主要介紹了C++基于socket編程實(shí)現(xiàn)聊天室功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07
如何使用visual studio2019創(chuàng)建簡(jiǎn)單的MFC窗口(使用C++)
這篇文章主要介紹了如何使用visual studio2019創(chuàng)建簡(jiǎn)單的MFC窗口(使用C++),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
編寫C語(yǔ)言程序進(jìn)行進(jìn)制轉(zhuǎn)換的問(wèn)題實(shí)例
這篇文章主要介紹了編寫C語(yǔ)言程序進(jìn)行進(jìn)制轉(zhuǎn)換的問(wèn)題實(shí)例,文中附錄了一個(gè)各種進(jìn)制間的轉(zhuǎn)換程序代碼,需要的朋友可以參考下2015-08-08
C語(yǔ)言鏈表實(shí)現(xiàn)歌手評(píng)分系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言鏈表實(shí)現(xiàn)歌手評(píng)分系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-03-03

