C語言struct關鍵字詳解—結構體

struct 是個神奇的關鍵字,它將一些相關聯的數據打包成一個整體,方便使用。

在網絡協議、通信控制、嵌入式系統、驅動開發等地方,我們經常要傳送的不是簡單的字節流(char 型數組),而是多種數據組合起來的一個整體,其表現形式是一個結構體。

經驗不足的開發人員往往將所有需要傳送的內容依順序保存在char 型數組中,通過指針偏移的方法傳送網絡報文等信息。這樣做編程復雜,易出錯,而且一旦控制方式及通信協議有所變化,程序就要進行非常細致的修改,非常容易出錯。這個時候只需要一個結構體就能搞定。平時我們要求函數的參數盡量不多于4 個,如果函數的參數多于4 個使用起來非常容易出錯(包括每個參數的意義和順序都容易弄錯),效率也會降低(與具體CPU 有關,ARM芯片對于超過4 個參數的處理就有講究,具體請參考相關資料)。這個時候,可以用結構體壓縮參數個數。

一、空結構體多大?

結構體所占的內存大小是其成員所占內存之和(關于結構體的內存對齊,請參考預處理那章)。這點很容易理解,但是下面的這種情況呢?
struct student
{
}stu;

sizeof(stu)的值是多少呢?在Visual C++ 6.0 上測試一下。

很遺憾,不是0,而是1。為什么呢?你想想,如果我們把struct student 看成一個模子的話,你能造出一個沒有任何容積的模子嗎?

顯然不行。編譯器也是如此認為。 編譯器認為任何一種數據類型都有其大小,用它來定義一個變量能夠分配確定大小的空間。既然如此,編譯器就理所當然的認為任何一個結構體都是有大小的,哪怕這個結構體為空。那萬一結構體真的為空,它的大小為什么值比較合適呢?

假設結構體內只有一個char 型的數據成員,那其大小為1byte(這里先不考慮內存對齊的情況).也就是說非空結構體類型數據最少需要占一個字節的空間,而空結構體類型數據總不能比最小的非空結構體類型數據所占的空間大吧。這就麻煩了,空結構體的大小既不能為0,也不能大于1,怎么辦?定義為0.5個byte?但是內存地址的最小單位是1 個byte,0.5 個byte 怎么處理?解決這個問題的最好辦法就是折中,編譯器理所當然的認為你構造一個結構體數據類型是用來打包一些數據成員的,而最小的數據成員需要1 個byte,編譯器為每個結構體類型數據至少預留1 個byte的空間。所以,空結構體的大小就定位1 個byte。

二、柔性數組

也許你從來沒有聽說過柔性數組(flexible array)這個概念,但是它確實是存在的。

C99 中,結構中的最后一個元素允許是未知大小的數組,這就叫做柔性數組成員,但結構中的柔性數組成員前面必須至少一個其他成員。柔性數組成員允許結構中包含一個大小可變的數組。sizeof 返回的這種結構大小不包括柔性數組的內存。包含柔性數組成員的結構用malloc ()函數進行內存的動態分配,并且分配的內存應該大于結構的大小,以適應柔性數組的預期大小。

柔性數組到底如何使用呢?看下面例子:
typedef struct st_type
{
? ?int i;
? ?int a[0];
}type_a;

有些編譯器會報錯無法編譯可以改成:
typedef struct st_type
{
? ?int i;
? ?int a[];
}type_a;

這樣我們就可以定義一個可變長的結構體, 用sizeof(type_a) 得到的只有4 , 就是sizeof(i)=sizeof(int)。那個0 個元素的數組沒有占用空間,而后我們可以進行變長操作了。通過如下表達式給結構體分配內存:
? ?type_a *p = (type_a*)malloc(sizeof(type_a)+100*sizeof(int));
這樣我們為結構體指針p 分配了一塊內存。用p->item[n]就能簡單地訪問可變長元素。

但是這時候我們再用sizeof(*p)測試結構體的大小,發現仍然為4。是不是很詭異?我們不是給這個數組分配了空間么?

別急,先回憶一下我們前面講過的“模子”。在定義這個結構體的時候,模子的大小就已經確定不包含柔性數組的內存大小。柔性數組只是編外人員,不占結構體的編制。只是說在使用柔性數組時需要把它當作結構體的一個成員,僅此而已。再說白點,柔性數組其實與結構體沒什么關系,只是“掛羊頭賣狗肉”而已,算不得結構體的正式成員。

需要說明的是:C89 不支持這種東西,C99 把它作為一種特例加入了標準。但是,C99所支持的是incomplete type,而不是zero array,形同int item[0];這種形式是非法的,C99 支持的形式是形同int item[];只不過有些編譯器把int item[0];作為非標準擴展來支持,而且在C99 發布之前已經有了這種非標準擴展了,C99 發布之后,有些編譯器把兩者合而為一了。

當然,上面既然用malloc 函數分配了內存,肯定就需要用free 函數來釋放內存:
? ?free(p);
經過上面的講解,相信你已經掌握了這個看起來似乎很神秘的東西。不過實在要是沒掌握也無所謂,這個東西實在很少用。

三、struct 與class 的區別

在C++里struct 關鍵字與class 關鍵字一般可以通用,只有一個很小的區別。struct 的成員默認情況下屬性是public 的,而class 成員卻是private 的。很多人覺得不好記,其實很容易。你平時用結構體時用public 修飾它的成員了嗎?既然struct 關鍵字與class 關鍵字可以通用,你也不要認為結構體內不能放函數了。

當然,關于結構體的討論遠沒有結束,在指針與數組那一章,你還會要和它打交道的。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/450400.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/450400.shtml
英文地址,請注明出處:http://en.pswp.cn/news/450400.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

JDK1.8使用Dubbo時需注意

2019獨角獸企業重金招聘Python工程師標準>>> Dubbo自帶的很多包都比較舊了,其中的javassist在JDK1.8上運行會報錯 而且錯誤通常比較詭異,javassist是編輯和創建Java字節碼的類庫,常見的錯誤會從spring中報出 解決辦法:…

Java 11 已發布,String 還能這樣玩!

在文章《Java 11 正式發布,這 8 個逆天新特性教你寫出更牛逼的代碼》中,我有介紹到 Java 11 的八個新特性,其中關于 String 加強部分,我覺得有點意思,這里單獨再拉出來講。 Java 11 增加了一系列的字符串處理方法&…

Redis1 晨考題

Redis1 晨考題 1.Redis 是什么 ? redis 是一個開源的使用 ANSI C 語言編寫、支持網絡、可基于內存亦可持久化的日志型、Key-Value 的內存數據庫,并提供多種語言的 API。 2.NOSQL 是什么 ?出現的目的和意義是什么 ? NoSQL 泛指…

C語言void關鍵字

void 有什么好講的呢?如果你認為沒有,那就沒有;但如果你認為有,那就真的有。有點像“色即是空,空即是色”。一、void a? void 的字面意思是“空類型”,void *則為“空類型指針”,vo…

深入了解RabbitMQ工作原理及簡單使用

深入了解RabbitMQ工作原理及簡單使用 RabbitMQ系列文章 RabbitMQ在Ubuntu上的環境搭建深入了解RabbitMQ工作原理及簡單使用RabbitMQ交換器Exchange介紹與實踐RabbitMQ事務和Confirm發送方消息確認——深入解讀使用Docker部署RabbitMQ集群你不知道的RabbitMQ集群架構全解RabbitM…

使用el-checkbox實現全選,點擊失效沒有反應

最近在公司接收到了一個需求,給收藏夾的書籍添加批量、全選刪除實現思路:點擊全選改變item的checked,改變item的checked,重新便利一下所有item的checked來改變全選的selectAll1)該組件基本功能已經實現,che…

Spring3.2新注解@ControllerAdvice

前些天發現了一個巨牛的人工智能學習網站,通俗易懂,風趣幽默,忍不住分享一下給大家。點擊跳轉到教程。 ControllerAdvice,是spring3.2提供的新注解,從名字上可以看出大體意思是控制器增強。讓我們先看看ControllerAdv…

Mysql1 晨考題

Mysql1 晨考題 1.描述主鍵、外鍵、候選主鍵、超鍵分別是什么 ? (1)主鍵:數據庫表中對存儲數據對象給予唯一完整標識的數據列或屬性的組合。一個數據列只能有一個主 鍵,且主鍵的取值不能缺失,即不能為空值…

C語言關鍵字

C語言do、while、for關鍵字—循環 C 語言中循環語句有三種:while 循環、do-while 循環、for 循環。while 循環:先判斷while 后面括號里的值,如果為真則執行其后面的代碼;否則不執行。while(1)表示死循環。…

C語言字符篇(五)內存函數

memcpy不可以把目的地址寫成本身但是memmove可以,因為它是先保存到臨時空間 #include <string.h> void *memcpy(void *dest, const void *src, size_t n);將內存src拷貝n個字符到內存destvoid *memmove(void *dest, const void *src, size_t n);將內存src的前n個數據拷貝…

GMQ交易平臺大力探索區塊鏈技術,進一步推動產業繁榮

近年來&#xff0c;區塊鏈技術作為金融科技的中堅力量&#xff0c;受到了產業界的熱切關注&#xff0c;其實驗開展和應用研發正在如火如荼的進行。 在此背景下&#xff0c;各地涌現出一大批優秀的企業投入到區塊鏈產業中&#xff0c;各類企業投融 資活動十分活躍&#xff0c;充…

java 筆試題

JAVA-2003筆試題 一、選擇題&#xff08;每小題2&#xff0c;共10分&#xff09; 下列語句序列執行后&#xff0c;m 的值是&#xff08; C &#xff09; int a10, b3, m5; if( ab ) ma; else ma*m; A.15 B.50 C.55 D.5若已定義byte[]x{11,22,33,-66}其中0≤k≤3&#xff0c;則…

objectdatasouce的溫故

在做ecxel的時候&#xff0c;需要前臺做一個聯動的效果。 記錄一下這個數據源的用法&#xff0c;大學時候用的&#xff0c;忘得差不多了 首先就是往頁面拖拽一個objectdatasouce的控件 然后配置數據源&#xff1a; 選擇業務對象(其實就是選擇你要用的哪個類&#xff0c;如果下拉…

都會五星回評,歡迎留下地址-博客之星

歡迎五星回評地址https://bbs.csdn.net/topics/603961857

jQuery核心

jQuery(selector) jQuery 的核心功能都是通過這個函數實現的。 jQuery中的一切都基于這個函數&#xff0c;或者說都是在以某種方式使用這個函數。這個函數最基本的用法就是向它傳遞一個表達式&#xff08;通常由 CSS 選擇器組成&#xff09;&#xff0c;然后根據這個表達式來查…

Feign api調用方式

Feign使用簡介 基本用法 基本的使用如下所示&#xff0c;一個對于canonical Retrofit sample的適配。 interface GitHub {// RequestLine注解聲明請求方法和請求地址,可以允許有查詢參數RequestLine("GET /repos/{owner}/{repo}/contributors")List<Contributor&g…

預處理

C語言##預算符 和#運算符一樣&#xff0c;##運算符可以用于宏函數的替換部分。這個運算符把兩個語言符號組合成單個語言符號。看例子&#xff1a;#define XNAME(n) x ## n如果這樣使用宏&#xff1a;XNAME(8)則會被展開成這樣&#xff1a;x8看明白了沒&#xff1f; ##就是個粘合…

Lambda表達式使用2

1.概述    本篇主要介紹lambda中常用的收集器&#xff0c;收集器的作用就是從數據流中生成需要的數據接口。    最常用的就是Collectors.toList()&#xff0c;只要將它傳遞給collect()函數&#xff0c;就能夠使用它了。    在我們使用收集器的時候經常會用到“方法…

notepad++ 使用去掉自動檢查紅線

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 notepad新升級了之后就有自動判斷的紅線&#xff0c;單詞拼錯了就給提示&#xff0c;看著這紅線實在難受 在 菜單選項&#xff1a;[插件…

cAdvisor+InfluxDB+Grafana 監控Docker

容器的監控方案其實有很多&#xff0c;有docker自身的docker stats命令、有Scout、有Data Dog等等&#xff0c;本文主要和大家分享一下比較經典的容器開源監控方案組合&#xff1a;cAdvisorInfluxDBGrafan 一、概念 1). InfluxDB是什么nfluxDB是用GO語言編寫的一個開源分布式時…