欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語(yǔ)言柔性數(shù)組詳解

 更新時(shí)間:2021年10月10日 17:10:33   作者:高郵吳少  
這篇文章主要介紹了C語(yǔ)言柔性數(shù)組,通過(guò)實(shí)例分析了不完整類型、結(jié)構(gòu)體及柔性數(shù)組等概念,需要的朋友可以參考下,希望能夠給你帶來(lái)幫助

前言

可能大家第一眼看到這個(gè)標(biāo)題會(huì)有點(diǎn)懵,到底什么是柔性數(shù)組,我怎么從來(lái)沒(méi)聽說(shuō)過(guò)?但柔性數(shù)組確實(shí)是存在的,也經(jīng)常會(huì)出現(xiàn)在一些公司的面試題中,今天就跟著筆者來(lái)學(xué)習(xí)一下柔性數(shù)組吧。

提示:以下是本篇文章正文內(nèi)容,下面案例可供參考

在這里插入圖片描述

一、柔性數(shù)組是什么?

C99中,結(jié)構(gòu)體中的最后一個(gè)元素允許是未知大小的數(shù)組,這就叫作柔性數(shù)組,for example:

 struct st_type
{
	int i;
	int a[0];//柔性數(shù)組成員,也可以寫int a[];
};

結(jié)構(gòu)體成員a數(shù)組,它的數(shù)組大小是沒(méi)有確定的,將來(lái)如果需要可以大也可以小。
有些編譯器支持a[0]這種寫法,有些編譯器支持a[ ]這種寫法,具體取決編譯器。

二、柔性數(shù)組的特點(diǎn)

1.結(jié)構(gòu)體中柔性數(shù)組成員前面必須至少有一個(gè)其他成員

示例如下:

 struct st_type
{
	int i;
	int a[0];//柔性數(shù)組成員,也可以寫int a[];
};

比如上面這段代碼,如果你要?jiǎng)?chuàng)建一個(gè)柔性數(shù)組a,前面必須創(chuàng)建一些別的成員

2.sizeof返回的這種結(jié)構(gòu)大小不包括柔性數(shù)組的內(nèi)存

示例如下:

 struct st_type
{
	int i;//4字節(jié)
	int a[0];//柔性數(shù)組成員,也可以寫int a[];
	//因?yàn)槭侨嵝詳?shù)組,無(wú)法確認(rèn)a占幾個(gè)字節(jié)
};
 int main()
 {
	 printf("%d\n", sizeof(struct st_type));//打印4
	 return 0;
 }

這里計(jì)算包含柔性數(shù)組的結(jié)構(gòu)體大小,因?yàn)槿嵝詳?shù)組本身是無(wú)法確定有幾個(gè)字節(jié)的,所以計(jì)算整體結(jié)構(gòu)體大小時(shí),會(huì)省略柔性數(shù)組的計(jì)算。

3.包含柔性數(shù)組成員的結(jié)構(gòu)用malloc()函數(shù)進(jìn)行內(nèi)存的動(dòng)態(tài)分配,并且分配的內(nèi)存應(yīng)該大于結(jié)構(gòu)的大小,以適應(yīng)柔性數(shù)組的預(yù)期大小

ps:除了malloc函數(shù),realloc、calloc等動(dòng)態(tài)內(nèi)存開辟的函數(shù)也需要類似的操作

比如說(shuō)我現(xiàn)在要數(shù)組a里面有10個(gè)元素,現(xiàn)在進(jìn)行malloc一下

示例如下:

#include<string.h>
#include<errno.h>
struct st_type
{
	int i;//4字節(jié)
	int a[0];//柔性數(shù)組成員,也可以寫int a[];
};
int main()
{
    //假設(shè)我現(xiàn)在需要a里有10個(gè)元素
	struct st_type*ps=(struct st_type*)malloc(sizeof(struct st_type) + 10 * sizeof(int));
	if (ps == NULL)//由于空間可能不夠開辟導(dǎo)致malloc開辟失敗,開辟失敗會(huì)返回空指針
	{
		printf("%s\n", strerror(errno));
		return -1;//程序出問(wèn)題后,跳出程序
	}
	//開辟成功
	int j = 0;
	for (j = 0;j < 10;j++)
	{
		ps->a[j] = j;
	}
	for (j = 0;j < 10;j++)
	{
		printf("%d ", ps->a[j]);//打印0-9
	}
	printf("\n");
	//如果想繼續(xù)用柔性數(shù)組a進(jìn)行打印
	//比如現(xiàn)在a里只有10個(gè)元素,我用完10個(gè)了,我還要繼續(xù)來(lái)10個(gè),用realloc追加
	struct st_type*ptr=realloc(ps, sizeof(struct st_type) + 20 * sizeof(int));//ps:realloc第二個(gè)參數(shù)是調(diào)整后的整體大小
	if (ptr == NULL)
	{
		printf("擴(kuò)容失敗\n");
		return -1;
	}
	else
	{
		ps = ptr;
	}
	//擴(kuò)容成功
	int k = 0;
	for (k = 10;k < 20;k++)
	{
		ps->a[k] = k;
	}
	for (j = 0;j < 20;j++)
	{
		printf("%d ", ps->a[j]);//打印0-19
	}
	//釋放空間
	free(ps);
	ps = NULL;
	return 0;
}

在這里插入圖片描述

在這里插入圖片描述

我們這里需要數(shù)組a里有10個(gè)元素,那我們malloc的時(shí)候要對(duì)結(jié)構(gòu)體里的整形i先開辟4個(gè)字節(jié),然后為整形數(shù)組a再開辟40個(gè)字節(jié),然后malloc函數(shù)返回開辟空間的起始地址,賦給truct st_type * 類型的ps指針。

malloc(sizeof(struct st_type) + 10 * sizeof(int))這個(gè)操作等價(jià)于struct st_type類型創(chuàng)建一個(gè)變量所占空間,只不過(guò)是用malloc來(lái)開辟

你改變數(shù)組a大小,追加空間時(shí),realloc(ps, sizeof(struct st_type) + 20 * sizeof(int)),realloc的第一個(gè)參數(shù)仍然是ps,因?yàn)槟惝?dāng)時(shí)是用malloc一次開辟出的一塊空間,你是不能單獨(dú)調(diào)整數(shù)組a的空間的

三、柔性數(shù)組的優(yōu)點(diǎn)

柔性數(shù)組就是對(duì)一塊空間實(shí)現(xiàn)動(dòng)態(tài)開辟嘛,那我們之前也講過(guò)指針來(lái)動(dòng)態(tài)內(nèi)存開辟,我們來(lái)看一段代碼來(lái)對(duì)比一下這兩種方法:

//用指針也可以做到a指向的空間動(dòng)態(tài)變化
struct st_type
{
	int i;//4字節(jié)
	int *a;//4字節(jié),這里計(jì)算結(jié)構(gòu)體大小恰好是8字節(jié)
};
int main()
{
	struct st_type*ps = (struct st_type*)malloc(sizeof(struct st_type));
	ps->i = 100;
	ps->a = (int*)malloc(10 * sizeof(int));//a指向40個(gè)字節(jié)的空間,該空間由int*進(jìn)行管理
	int j = 0;
	for (j = 0;j < 10;j++)
	{
		ps->a[j] = j;//a[j]=*(a+j)
	}
	for (j = 0;j < 10;j++)
	{
		printf("%d", ps->a[j]);
	}
	//a指向的空間不夠了,希望調(diào)整大小
	int *ptr = (int*)realloc(ps->a, 20 * sizeof(int));
	if (ptr == NULL)
	{
		printf("擴(kuò)容失敗");
		return -1;
	}
	else
	{
		ps->a = ptr;
	}
	//使用...
	//釋放
	free(ps->a);
	ps->a = NULL;
	free(ps);
	ps = NULL;
}

這里需要注意的是,在釋放空間時(shí),你要先釋放指針a指向的空間,然后釋放結(jié)構(gòu)體指針

在這里插入圖片描述

如上圖,我們結(jié)構(gòu)體指針ps開辟一塊空間,空間里存放整形i和整形指針a,a又malloc(后續(xù)如果需要還可以realloc追加)一塊空間,如果你先釋放掉ps,a就沒(méi)了,你就沒(méi)法找到a指向的那塊空間了。

還是那個(gè)生動(dòng)的例子 就比如a是一個(gè)警察頭子,malloc開辟的空間是臥底,只有a知道那個(gè)臥底,你現(xiàn)在警察頭子死了,再也沒(méi)法證明臥底是臥底了,也就是說(shuō)a消失后,沒(méi)辦法再對(duì)開辟的空間進(jìn)行釋放,這時(shí)就會(huì)造成內(nèi)存泄露,指針實(shí)現(xiàn)動(dòng)態(tài)內(nèi)存調(diào)整是需要對(duì)指針釋放講解一定的順序性的

這里對(duì)比柔性數(shù)組,柔性數(shù)組和上述的指針都可以實(shí)現(xiàn)一塊空間大小的調(diào)整,但是柔性數(shù)組有兩個(gè)好處:
第一個(gè)好處是:方便內(nèi)存釋放

如果我們的代碼是在一個(gè)給別人用的函數(shù)中,你在里面做了二次內(nèi)存分配,并把整個(gè)結(jié)構(gòu)體返回給用戶。用戶調(diào)用free可以釋放結(jié)構(gòu)體,但是用戶并不知道這個(gè)結(jié)構(gòu)體內(nèi)的成員也需要free,所以你不能指望用戶來(lái)發(fā)現(xiàn)這個(gè)事。以上,如果我把結(jié)構(gòu)體的內(nèi)存及其成員要的內(nèi)存一次性分配好,并返回給用戶一個(gè)結(jié)構(gòu)體指針,用戶做一次free就可以把所有內(nèi)存都釋放掉,并且不用考慮前面說(shuō)的釋放的順序。

第二個(gè)好處是:加快訪問(wèn)速度

連續(xù)的內(nèi)存有益于提高訪問(wèn)速度,也有益于減少內(nèi)存碎片。

ps:內(nèi)存碎片如下圖

在這里插入圖片描述

操作系統(tǒng)給我們一塊內(nèi)存,我們?cè)谶M(jìn)行malloc時(shí),不一定就是一塊連著一塊的,上圖的空白部分就是內(nèi)存碎片,有些類似我們裁剪布料,一些剩余的邊角料這樣。

總結(jié)

本文介紹了柔性數(shù)組的定義、其三個(gè)使用特點(diǎn),及其對(duì)比指針實(shí)現(xiàn)動(dòng)態(tài)內(nèi)存的優(yōu)勢(shì),并對(duì)其進(jìn)行了具體舉例

本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!

相關(guān)文章

  • 基于C++全局變量的聲明與定義的詳解

    基于C++全局變量的聲明與定義的詳解

    本篇文章是對(duì)C++全局變量的聲明與定義進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C語(yǔ)言詳解無(wú)頭單向非循環(huán)鏈表各種操作方法

    C語(yǔ)言詳解無(wú)頭單向非循環(huán)鏈表各種操作方法

    無(wú)頭單向非循環(huán)鏈表:結(jié)構(gòu)簡(jiǎn)單,一般不會(huì)單獨(dú)用來(lái)存數(shù)據(jù)。實(shí)際中更多是作為其他數(shù)據(jù)結(jié)構(gòu)的子結(jié)構(gòu),如哈希桶、圖的鄰接表等等。另外這種結(jié)構(gòu)在筆試面試中出現(xiàn)很多
    2022-04-04
  • C++簡(jiǎn)單QQ程序服務(wù)器端的實(shí)現(xiàn)代碼

    C++簡(jiǎn)單QQ程序服務(wù)器端的實(shí)現(xiàn)代碼

    這篇文章主要為大家詳細(xì)介紹了C++簡(jiǎn)單QQ程序服務(wù)器端的實(shí)現(xiàn)代碼,感興趣的朋友可以參考一下
    2016-05-05
  • C++ 實(shí)現(xiàn)稀疏矩陣的壓縮存儲(chǔ)的實(shí)例

    C++ 實(shí)現(xiàn)稀疏矩陣的壓縮存儲(chǔ)的實(shí)例

    這篇文章主要介紹了C++ 實(shí)現(xiàn)稀疏矩陣的壓縮存儲(chǔ)的實(shí)例的相關(guān)資料,M*N的矩陣,矩陣中有效值的個(gè)數(shù)遠(yuǎn)小于無(wú)效值的個(gè)數(shù),且這些數(shù)據(jù)的分布沒(méi)有規(guī)律,需要的朋友可以參考下
    2017-07-07
  • C++超詳細(xì)講解運(yùn)算符重載

    C++超詳細(xì)講解運(yùn)算符重載

    本文包括了對(duì)C++類的6個(gè)默認(rèn)成員函數(shù)中的賦值運(yùn)算符重載和取地址和const對(duì)象取地址操作符的重載。運(yùn)算符是程序中最最常見的操作,例如對(duì)于內(nèi)置類型的賦值我們直接使用=賦值即可,因?yàn)檫@些編譯器已經(jīng)幫我們做好了,但是對(duì)象的賦值呢?能直接賦值嗎
    2022-06-06
  • C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇游戲

    C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇游戲

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-07-07
  • C++Primer筆記之關(guān)聯(lián)容器的使用詳解

    C++Primer筆記之關(guān)聯(lián)容器的使用詳解

    本篇文章對(duì)C++Primer 關(guān)聯(lián)容器的使用進(jìn)行了詳細(xì)的分析介紹。需要的朋友參考下
    2013-05-05
  • C++中priority_queue模擬實(shí)現(xiàn)的代碼示例

    C++中priority_queue模擬實(shí)現(xiàn)的代碼示例

    在c++語(yǔ)言中數(shù)據(jù)結(jié)構(gòu)中的堆結(jié)構(gòu)可以通過(guò)STL庫(kù)中的priority_queue 優(yōu)先隊(duì)列來(lái)實(shí)現(xiàn),這樣做極大地簡(jiǎn)化了我們的工作量,這篇文章主要給大家介紹了關(guān)于C++中priority_queue模擬實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下
    2021-08-08
  • C++實(shí)現(xiàn)猜數(shù)字小游戲

    C++實(shí)現(xiàn)猜數(shù)字小游戲

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)猜數(shù)字小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-10-10
  • C++實(shí)現(xiàn)寢室衛(wèi)生管理系統(tǒng)

    C++實(shí)現(xiàn)寢室衛(wèi)生管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)寢室衛(wèi)生管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03

最新評(píng)論