成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

C語(yǔ)言-常用內(nèi)存函數(shù)詳解+模擬實(shí)現(xiàn)

cucumber / 1458人閱讀

前言:博主之前有已經(jīng)寫(xiě)過(guò)了C語(yǔ)言常用字符函數(shù)詳解+模擬實(shí)現(xiàn),感興趣的同學(xué)可以去圍觀一下哦!


目錄

前言:

1.內(nèi)存函數(shù)

memcpy()

?memmove()

memcmp()

memset()

2.錯(cuò)誤信息報(bào)告函數(shù)

strerror()

?perror()


1.內(nèi)存函數(shù)

memcpy()

作用:內(nèi)存拷貝

函數(shù)原型:

注意:count:要拷貝的字節(jié)數(shù)

  • 函數(shù)memcpy從src位置開(kāi)始向后賦值count個(gè)字節(jié)的數(shù)據(jù)到dest的內(nèi)存位置,

  • 遇到 ‘/0’ 的時(shí)候不停下來(lái)

  • 如果source和destination有任何的重疊,復(fù)制的結(jié)果都是未定義的

使用樣例:

int main(){	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };	int arr2[20] = { 0 };	memcpy(arr2, arr1, sizeof(int) * 10);	int i = 0;	for (i = 0;i < 20; i++)	{		printf("%d ", arr2[i]);	}	return 0;}

模擬實(shí)現(xiàn):

  • 1.只知道要拷貝的字節(jié)數(shù),不知道要拷貝的元素是什么類型。所以可以強(qiáng)轉(zhuǎn)為char*指針->1個(gè)字節(jié)1個(gè)字節(jié)的拷貝
  • 2.返回目標(biāo)空間的起始地址,注意返回類型是void*,是指針,所以可以返回 若返回類型是void,則不可以返回,
  • 3.目標(biāo)空間不修改,所以可以使用const修飾
  • 4.要拷貝的字節(jié)數(shù)不可能為負(fù)數(shù),所以接收類型可以為無(wú)符號(hào)整數(shù)
void* my_memcpy(void* dest,const void* src,size_t count ){	assert(dest&&src);	void* ret = dest;	//為了返回目標(biāo)空間起始地址	//共拷貝n個(gè)字節(jié)	while(count--)	{		*(char*)dest = *(char*)src;		dest = (char*)dest + 1;//這樣寫(xiě)是防止編譯器報(bào)錯(cuò)		src = (char*)src + 1;	}	//若寫(xiě)成 dest = (char*)dest++; //有些編譯器可能跑不過(guò)去	return ret;}int main(){	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };	int arr2[20] = { 0 };	my_memcpy(arr2, arr1, sizeof(int) * 10);	int i = 0;	for (i = 0;i < 10; i++)	{		printf("%d ", arr2[i]);	}	return 0;}

注意:我們這種模擬實(shí)現(xiàn)的方法,不能處理重疊的內(nèi)存拷貝,

如:想要將數(shù)組的1234拷貝到3456中,

void* my_memcpy(void* dest, const void* src, size_t count){	assert(dest && src);	void* ret = dest;	//為了返回目標(biāo)空間起始地址	//共拷貝n個(gè)字節(jié)	while (count--)	{		*(char*)dest = *(char*)src;		dest = (char*)dest + 1;//這樣寫(xiě)是防止編譯器報(bào)錯(cuò)		src = (char*)src + 1;	}	//若寫(xiě)成 dest = (char*)dest++; //有些編譯器可能跑不過(guò)去	return ret;}int main(){	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };	my_memcpy(arr+2, arr, 16);	int i = 0;	for (i = 0;i < 10; i++)	{		printf("%d ", arr[i]);	}	return 0;}

?想要的效果:1212347890

使用my_memcpy得到的結(jié)果:1 2 1 2 1 2 7 8 9 10

?

但是使用庫(kù)函數(shù)memcpy函數(shù),可以實(shí)現(xiàn)拷貝

但是C語(yǔ)言只要求memcpy函數(shù)可以實(shí)現(xiàn)不重疊的內(nèi)存拷貝即可,要實(shí)現(xiàn)重疊內(nèi)存拷貝的話,實(shí)現(xiàn)memmove函數(shù) ?


?memmove()

作用:內(nèi)存拷貝-可處理重疊的內(nèi)存拷貝

函數(shù)原型:

  • 和memcpy的差別就是memmove函數(shù)處理的源內(nèi)存塊和目標(biāo)內(nèi)存塊是可以重疊的。

  • 如果源空間和目標(biāo)空間出現(xiàn)重疊,就得使用memmove函數(shù)處理。


使用樣例:

int main(){    int arr[] ={1,2,3,4,5,6,7,8,9,10};    memmove(arr+2,arr,16);    int i = 0;    for(i = 0;i <10;i++)    {        printf("%d ",arr[i]);    }    return 0;}

打印結(jié)果:1 2 1 2 3 4 7 8 9 10


?模擬實(shí)現(xiàn):

以dest

注意:由于不知道傳過(guò)來(lái)的是什么類型的數(shù)據(jù),所以可以使用void*類型接收參數(shù)

返回目標(biāo)空間的起始地址

由于只知道拷貝的字節(jié)數(shù),不知道拷貝的類型->使用char*類型,每次拷貝一個(gè)字節(jié)

從前向后拷貝:dest和src強(qiáng)轉(zhuǎn)為char*類型,然后+1,共拷貝n次即可


如何從后向前拷貝

->原數(shù)據(jù)的最后拷貝到目標(biāo)位置的最后,然后二者不斷往前拷貝

從后往前拷貝:dest和src強(qiáng)轉(zhuǎn)為char*類型后,+count,跳過(guò)count個(gè)字節(jié),指向要拷貝的最后一個(gè)字節(jié),不斷拷貝,count--

?

?

void* my_memmove(void* dest, void* src, size_t count){	assert(dest && src);	void* tmp = dest;//后序返回目標(biāo)空間的地址	if (dest < src)	{        //共拷貝count次		while (count--)		{			//從前往后拷貝			*(char*)dest = *(char*)src;			dest = (char*)dest + 1;			src = (char*)src + 1;		}	}	else	{		//從后往前拷貝		while (count--)		{			*((char*)dest + count) = *((char*)src + count);		}	}	return tmp;}int main(){	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };	//my_memmove(arr+2,arr,16);	my_memmove(arr , arr+2, 16);	int i = 0;	for (i = 0; i < 10; i++)	{		printf("%d ", arr[i]);	}	return 0;}


memcmp()

作用:內(nèi)存比較

函數(shù)原型:


?


使用strcmp()只能比較字符串,而memcmp()可以比較整形,也可以比較字符串 ?

使用樣例:

int main(){    int arr1[] = {1,2,3,4};    //01 00 00 00 02 00 00 00 03 00 00 00 04 00 00 00    int arr2[] = {1,2,3,5};    //01 00 00 00 02 00 00 00 03 00 00 00 05 00 00 00    int ret = memcmp(arr1,arr2,12);    //比較前12個(gè)字節(jié),二者相同,返回0    printf("%d/n",ret);//0    ret = memcmp(arr1,arr2,13);    //比較前13個(gè)字節(jié),arr1中的04比arr2中的05小,返回-1    printf("%d/n",ret);	//-1}

?模擬實(shí)現(xiàn)memcmp()

注意:由于不知道傳過(guò)來(lái)的是什么類型->使用void*類型接收,由于要比較的字節(jié)數(shù)恒大于0,所以用size_t接收,

由于只知道要比較的字節(jié)數(shù),不知道是什么類型->所以使用char* 一個(gè)字節(jié)一個(gè)字節(jié)的向后比較,直到找到二者中字節(jié)對(duì)應(yīng)內(nèi)容不相等時(shí),比較此時(shí)二者字節(jié)內(nèi)容

int my_memcmp(const void* p1, const void* p2, size_t count){    assert(p1 && p2);    while (count--)    {        //如果二者指向的一個(gè)字節(jié)內(nèi)容相等->指向下一個(gè)字節(jié)        if (*(char*) p1 == *(char*)p2)        {            p1 = (char*)p1+1;            p2 = (char*)p2+1;        }        else        {            //此時(shí)二者指向的字節(jié)內(nèi)容不相同            //通過(guò)強(qiáng)轉(zhuǎn)為char* 比較此時(shí)二者指向的一個(gè)字節(jié)內(nèi)容            if ((*(char*)p1 - *(char*)p2) > 0)            {                return 1;            }            else            {                return -1;            }        }    }    //跳出循環(huán)時(shí),count = 0,說(shuō)明二者要比較的字節(jié)的內(nèi)容都相同        return 0;}int main(){    int arr1[] = { 1,2,3,4 };    //01 00 00 00 02 00 00 00 03 00 00 00 04 00 00 00    int arr2[] = { 1,2,3,99 };    //01 00 00 00 02 00 00 00 03 00 00 00 05 00 00 00    int ret = my_memcmp(arr1, arr2, 12);    //比較前12個(gè)字節(jié),二者相同,返回0    printf("%d/n", ret);    //0    ret = my_memcmp(arr1, arr2, 13);    //比較前13個(gè)字節(jié),arr1中的04比arr2中的05小,返回-1    printf("%d/n", ret);    //-1    char* p1 = "Mangopp";    char* p2 = "Mangoppp";    ret = memcmp(p1, p2, 8);    printf("%d/n", ret);    //-1}

memset()

作用:內(nèi)存設(shè)置函數(shù)

函數(shù)原型:

?

將緩沖區(qū)設(shè)置為指定的字符

字節(jié)為單位進(jìn)行初始化

-----------------

使用樣例:

int main(){    int arr[] = {1,2,3,4,5};        memset(arr,0,20);	//初始化20個(gè)字節(jié)內(nèi)容為0        memset(arr,1,20);	//初始化20個(gè)字節(jié)內(nèi)容為1    return 0;}

?

?

?


?

?注意點(diǎn):以字節(jié)為單位進(jìn)行初始化


?模擬實(shí)現(xiàn)

注意:由于不知道傳過(guò)來(lái)的是什么類型->使用void*類型接收,由于要設(shè)置的字節(jié)數(shù)恒大于0,所以用size_t接收,

由于只知道要設(shè)置的字節(jié)數(shù),不知道是什么類型->所以使用char* 一個(gè)字節(jié)一個(gè)字節(jié)的向后設(shè)置

返回目標(biāo)空間的起始地址


寫(xiě)法1:接收類型為整形

memset:將緩沖區(qū)設(shè)置為指定的字符,所以要把整形強(qiáng)轉(zhuǎn)為字符型,再把每一個(gè)字節(jié)設(shè)置為整形對(duì)應(yīng)ascii的字符

void* my_memset(void* dest, int c, size_t count){    assert(dest);    void* tmp = dest;    char set_c = (char)c;    //內(nèi)存設(shè)置,共設(shè)置count個(gè)字節(jié)    while (count--)    {        *(char*)dest = set_c;        dest = (char*)dest + 1;    }    return tmp;}int main(){    int arr[] = { 1,2,3,4,5 };    my_memset(arr, 0, 20);	//初始化20個(gè)字節(jié)內(nèi)容為0    my_memset(arr, 1, 20);//初始化20個(gè)字節(jié)的內(nèi)容為1    return 0;}

?

寫(xiě)法2:接收類型為字符

若直接設(shè)置字符,要設(shè)置的是其對(duì)應(yīng)的值 1 而不是其對(duì)應(yīng)的ascii碼值。所以要減去‘0’

‘1’ - ‘0’ -> 1 所以接收的字符還要減上字符0

void* my_memset(void* dest, char c, size_t count){    assert(dest);    void* tmp = dest;    //內(nèi)存設(shè)置,共設(shè)置count個(gè)字節(jié)    while (count--)    {        *(char*)dest = c - "0";        dest = (char*)dest + 1;    }    return tmp;}int main(){    int arr[] = { 1,2,3,4,5 };    my_memset(arr, "0", 20);	//初始化20個(gè)字節(jié)內(nèi)容為0    my_memset(arr, "1", 20);    return 0;}


2.錯(cuò)誤信息報(bào)告函數(shù)


strerror()

作用:可以返回C語(yǔ)言內(nèi)置的錯(cuò)誤碼對(duì)應(yīng)的錯(cuò)誤信息

函數(shù)原型:

返回錯(cuò)誤碼對(duì)應(yīng)的錯(cuò)誤信息

參數(shù)傳錯(cuò)誤碼,翻譯成錯(cuò)誤信息,返回的是錯(cuò)誤碼對(duì)應(yīng)的錯(cuò)誤信息的字符串的首字符地址

C語(yǔ)言庫(kù)函數(shù)調(diào)用失敗的時(shí)候,會(huì)把錯(cuò)誤碼存儲(chǔ)到errno變量中


?使用樣例:

#includeint main(){     printf("%s/n",strerror(0));     printf("%s/n",strerror(1));     printf("%s/n",strerror(2));     printf("%s/n",strerror(3));    return 0;}

打印結(jié)果:
No error
Operation not permitted
No such file or directory
No such process?


?應(yīng)用:

int main(){    FILE* pf = fopen("test.txt","r");    if(pf == NULL)    {        printf("%s/n",strerror(errno));	//C會(huì)把錯(cuò)誤碼放到errno變量,打印錯(cuò)誤碼對(duì)應(yīng)的錯(cuò)誤信息            }    else    {        printf("打開(kāi)成功/n");    }    return 0;}//fopen():打開(kāi)文件的函數(shù),如果打開(kāi)成功,返回指向該文件的文件指針FILE* 如否則返回NULL//r:只讀方式打開(kāi)

由于我們沒(méi)有此文件:所以打印結(jié)果

No such file or directory


?perror()

作用:

打印自定義內(nèi)容+strerror

函數(shù)原型:

?

string:自定義字符


使用樣例:

int main(){    FILE* pf = fopen("test.txt", "r");    if (pf == NULL)    {        perror("測(cè)試");        perror("測(cè)試:/n");        perror("/n");    }    else    {        printf("打開(kāi)成功/n");    }    return 0;}

?測(cè)試: No such file or directory
測(cè)試:
: No such file or directory

: No such file or directory


如果perror中有內(nèi)容,先打印內(nèi)容,然后自動(dòng)在后面加上冒號(hào): 然后打印錯(cuò)誤信息

缺點(diǎn):不想打印時(shí)也打印了


最后,感謝你能看到這里!如果感覺(jué)此文對(duì)你有所幫助的,歡迎留個(gè)三連呀!感謝

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/122566.html

相關(guān)文章

  • [C/C++]詳解STL容器1--string的功能和模擬實(shí)現(xiàn)(深淺拷貝問(wèn)題)

    摘要:本文介紹了類的常用接口的使用,并對(duì)其進(jìn)行了模擬實(shí)現(xiàn),對(duì)模擬實(shí)現(xiàn)中涉及到的深淺拷貝問(wèn)題進(jìn)行了解析。在此之前,必須提到一個(gè)經(jīng)典問(wèn)題。為了解決淺拷貝問(wèn)題,所以中引入了深拷貝。但是實(shí)際使用中需要是第一個(gè)形參對(duì)象,才能正常使用。 本文介紹了string類的常用接口的使用,并對(duì)其進(jìn)行了模擬實(shí)現(xiàn),對(duì)模擬實(shí)...

    tianren124 評(píng)論0 收藏0
  • 手撕C語(yǔ)言進(jìn)階---字符串和內(nèi)存函數(shù)(詳解+實(shí)現(xiàn)+原碼)

    摘要:自己實(shí)現(xiàn)時(shí)返回值可根據(jù)實(shí)際情況而定源字符串必須以結(jié)束。語(yǔ)言中給了一些長(zhǎng)度受限的字符串函數(shù),而前面的函數(shù)是長(zhǎng)度不受限的字符串函數(shù)??截悅€(gè)字符從源字符串到目標(biāo)空間。 目錄 字符函數(shù)和字符串函數(shù) 函數(shù)介紹 strlen strcpy strcat strcmp strncpy ?strncat s...

    vslam 評(píng)論0 收藏0
  • [C/C++]詳解STL容器3--list的功能和模擬實(shí)現(xiàn)(迭代器失效問(wèn)題)

    摘要:本文介紹了的常用接口的使用,并對(duì)其進(jìn)行了模擬實(shí)現(xiàn),包括迭代器的實(shí)現(xiàn)。與為反向迭代器,對(duì)迭代器執(zhí)行操作,迭代器向前移動(dòng)。 本文介紹了list的常用接口的使用,并對(duì)其進(jìn)行了模擬實(shí)現(xiàn),包括list迭代器的實(shí)現(xiàn)。 目錄 一、list的介紹 二、list的常用接口的使用 1. list的構(gòu)造 2. l...

    amc 評(píng)論0 收藏0
  • 刨析C語(yǔ)言常用的字符串函數(shù)內(nèi)存函數(shù)模擬實(shí)現(xiàn)

    摘要:四函數(shù)字符串追加函數(shù)介紹函數(shù)的返回值類型為,可以返回被追加的字符串的起始地址。利用函數(shù)所需的頭文件函數(shù)的使用代碼運(yùn)行結(jié)果為函數(shù)的特點(diǎn)及注意事項(xiàng)源字符串必須以結(jié)束。目標(biāo)空間必須有足夠的大,能容納下源字符串的內(nèi)容。 ...

    XBaron 評(píng)論0 收藏0
  • C語(yǔ)言-常用字符函數(shù)詳解+模擬實(shí)現(xiàn)

    摘要:源字符串不修改,所以用修飾找到目標(biāo)空間的位置跳出循環(huán)時(shí),指向的就是目標(biāo)空間的位置,從該位置向后追加,相當(dāng)于作用字符串比較函數(shù),是比較字符串的庫(kù)函數(shù),比較的是字符串的內(nèi)容,不是長(zhǎng)度。 ?今天是2021年9月21日,首先祝大家中秋節(jié)快樂(lè)啦!但愿人長(zhǎng)久,千里共嬋娟。又是一年月圓,祝大家在中秋節(jié)都能...

    elliott_hu 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<