干貨分享 | C語(yǔ)言的聯(lián)合體

原創(chuàng) 2021-03-05 09:04:00 C語(yǔ)言聯(lián)合體

1、聯(lián)合體介紹

我們知道結(jié)構(gòu)體(Struct)是一種構(gòu)造類型或復(fù)雜類型,它可以包含多個(gè)類型不同的成員。在C語(yǔ)言 中,還有另外一種和結(jié)構(gòu)體非常類似的語(yǔ)法,叫做共用體(Union),它的定義格式為:

union 共用體名{ 
成員列表
 };

共用體有時(shí)也被稱為聯(lián)合或者聯(lián)合體,這也是 Union 這個(gè)單詞的本意。

結(jié)構(gòu)體和共用體的區(qū)別在于:結(jié)構(gòu)體的各個(gè)成員會(huì)占用不同的內(nèi)存,互相之間沒(méi)有影響;而共用體的所

有成員占用同一段內(nèi)存,修改一個(gè)成員會(huì)影響其余所有成員。

結(jié)構(gòu)體占用的內(nèi)存大于等于所有成員占用的內(nèi)存的總和(成員之間可能會(huì)存在縫隙),共用體占用的內(nèi)

存等于最長(zhǎng)的成員占用的內(nèi)存。共用體使用了內(nèi)存覆蓋技術(shù),同一時(shí)刻只能保存一個(gè)成員的值,如果對(duì)

新的成員賦值,就會(huì)把原來(lái)成員的值覆蓋掉。


#include "stdafx.h"
#include <stdlib.h>
#include <stdio.h> 
union data{ 
   int n; 
   char ch; 
   short m; 
};
int main(){ 
union data a; 
printf("sizeof(a) = %d\n", sizeof(a));
a.n = 0x11;
printf("n=%X, ch=%X, m=%X\n", a.n, a.ch, a.m);
a.ch = 0x66;
printf("n=%X, ch=%X, m=%X\n", a.n, a.ch, a.m); 
a.m = 0x5577; 
printf("n=%X, ch=%X, m=%X\n", a.n, a.ch, a.m); 
a.n = 0x11226677; 
printf("n=%X, ch=%X, m=%X\n", a.n, a.ch, a.m); 
system("pause"); 
return 0; }

運(yùn)行結(jié)果:

運(yùn)行結(jié)果


2 、結(jié)構(gòu)體冒號(hào)的用法

結(jié)構(gòu)體中的冒號(hào)表示位域,位域出現(xiàn)的原因是由于某些信息的存儲(chǔ)表示只需要幾個(gè)bit位就可以表示而不 需要一個(gè)完整的字節(jié),同時(shí)也是為了節(jié)省存儲(chǔ)空間和方便處理。

其表示形式為:



struct 位域結(jié)構(gòu)名
{
類型說(shuō)明符	位域名:位域長(zhǎng)度
}

其表示形式為:

struct  bit_struct
{
int  bit1:3;
int  bit2:5;
int  bit3:7;
}data;

其中bit_struct表示位域結(jié)構(gòu)體,bit1、bit2、bit3表示對(duì)應(yīng)的位域,data表示位域結(jié)構(gòu)體定義的變量。整個(gè)位域結(jié)構(gòu)體占用2個(gè)字節(jié),bit1占3位,bit2占5位,bit1和bit2共用一個(gè)字節(jié),bit3占7位,獨(dú)占一個(gè)字節(jié)


#include "stdafx.h" 
#include <stdlib.h> 
#include <stdio.h>
struct  bit_struct 
{
  char  bit1:3;
  char  bit2:5;
  char  bit3:7;
}data;
int main()
{
   data.bit1 = 0xAA;
  data.bit2 = 0xAA;
  data.bit3 = 0xAA;
 printf("sizeof(data) = %x \n data.bit1 = %x \n data.bit2 = %x \n data.bit3 = %x \n",sizeof(bit_struct),data.bit1,data.bit2,data.bit3);
  system("pause");
  return 0;
}


運(yùn)行結(jié)果:

查看運(yùn)行結(jié)果

3、聯(lián)合使用

舉一個(gè)MCP2518FD芯片的例子:先看一下CAN幀格式:

CAN幀格式


//占用4個(gè)字節(jié)
typedef struct _CAN_MSGOBJ_ID {
uint32_t SID : 11;
uint32_t EID : 18;uint32_t SID11 : 1;
uint32_t unimplemented1 : 2;
} CAN_MSGOBJ_ID;
//占用4個(gè)字節(jié)
typedef struct _CAN_TX_MSGOBJ_CTRL {
uint32_t DLC : 4;
uint32_t IDE : 1;
uint32_t RTR : 1;
uint32_t BRS : 1;
uint32_t FDF : 1;
uint32_t ESI : 1;
#ifdef MCP2517FD
uint32_t SEQ : 7;
uint32_t unimplemented1 : 16;
#else
uint32_t SEQ : 23;
#endif
} CAN_TX_MSGOBJ_CTRL;
//占用4個(gè)字節(jié)
typedef uint32_t CAN_MSG_TIMESTAMP;//沒(méi)有用到
typedef union _CAN_TX_MSGOBJ {
struct {
CAN_MSGOBJ_ID id; //占4個(gè)字節(jié)
CAN_TX_MSGOBJ_CTRL ctrl; //占4個(gè)字節(jié)
CAN_MSG_TIMESTAMP timeStamp;//占4個(gè)字節(jié)
} bF; //共享12個(gè)字節(jié)
uint32_t word[3];//共享12個(gè)字節(jié)
uint8_t byte[12];//共享12個(gè)字節(jié)
} CAN_TX_MSGOBJ;
txObj.bF.id.SID = CAN_TX_ID;
txObj.bF.ctrl.DLC = CAN_DLC_4;//發(fā)送的數(shù)據(jù)長(zhǎng)度
txObj.bF.ctrl.IDE = 0;//標(biāo)識(shí)符擴(kuò)展位,在擴(kuò)展幀中恒為隱性1,在標(biāo)準(zhǔn)幀中,IDE位于控制段,且
恒為顯性0
txObj.bF.ctrl.BRS = 0;//BRS(Bit Rate Switch)位速率轉(zhuǎn)換開(kāi)關(guān),當(dāng)BRS為顯性位時(shí)數(shù)據(jù)段的位
速率與仲裁段的位速率一致,當(dāng)BRS為隱性位時(shí)數(shù)據(jù)段的位速率高于仲裁段的位速率
txObj.bF.ctrl.FDF = 0;//擴(kuò)展數(shù)據(jù)長(zhǎng)度,在標(biāo)準(zhǔn)的CAN幀中,控制場(chǎng)包含的保留位被指定為顯性位發(fā)
送,但是在CAN-FD幀中以隱性位發(fā)送,主要用于區(qū)分標(biāo)準(zhǔn)CAN幀格式和CAN-FD的幀格式
n = DRV_CANFDSPI_DlcToDataBytes(CAN_DLC_4);
for (i = 0; i < n; i++)
{
txd[i] = Count;
Count++;
}uint8_t txBuffer[MAX_MSG_SIZE];
txBuffer[0] = txObj->byte[0]; //not using 'for' to reduce no of instructions
txBuffer[1] = txObj->byte[1];
txBuffer[2] = txObj->byte[2];
txBuffer[3] = txObj->byte[3];
txBuffer[4] = txObj->byte[4];
txBuffer[5] = txObj->byte[5];
txBuffer[6] = txObj->byte[6];
txBuffer[7] = txObj->byte[7];
uint8_t i;
for (i = 0; i < txdNumBytes; i++)
{
txBuffer[i + 8] = txd[i];
}
// Make sure we write a multiple of 4 bytes to RAM
uint16_t n = 0;
uint8_t j = 0;
if (txdNumBytes % 4)
{
// Need to add bytes
n = 4 - (txdNumBytes % 4);
i = txdNumBytes + 8;
for (j = 0; j < n; j++)
{
txBuffer[i + 8 + j] = 0;
}
}
spiTransferError = DRV_CANFDSPI_WriteByteArray(index, a, txBuffer,
txdNumBytes + 8 + n);
if (spiTransferError)
{
return -4;
}
// Set UINC and TXREQ
spiTransferError = DRV_CANFDSPI_TransmitChannelUpdate(index, channel,
flush);
if (spiTransferError)
{
return -5;
}
return spiTransferError;

相關(guān)產(chǎn)品 >

  • OKT507-C 開(kāi)發(fā)板

    CPU: T507 工業(yè)級(jí)處理器,超低功耗,10年+生命周期|推薦全志工業(yè)級(jí)T507開(kāi)發(fā)板系列,飛凌OKT507-C開(kāi)發(fā)板采用全志T507 四核工業(yè)級(jí)處理器 T507設(shè)計(jì)開(kāi)發(fā),Cortex-A53架構(gòu),工業(yè)級(jí)寬溫,性能強(qiáng),低功耗,是一款高性價(jià)比的工業(yè)級(jí)產(chǎn)品,提供豐富的開(kāi)發(fā)設(shè)計(jì)資料,提供產(chǎn)品規(guī)格書,軟硬件手冊(cè)等,全志的T507適用于車載電子、電力、醫(yī)療、工業(yè)控制、物聯(lián)網(wǎng)、智能終端等領(lǐng)域。

    了解詳情
    OKT507-C 開(kāi)發(fā)板
  • OK3568-C開(kāi)發(fā)板

    強(qiáng)而穩(wěn),國(guó)產(chǎn)芯,1Tops算力,多路高速接口|飛凌RK3568系列RK3568開(kāi)發(fā)板基于國(guó)產(chǎn)工業(yè)級(jí)AI處理器RK3568四核64位Cortex-A55 處理器設(shè)計(jì)。RK3568作為國(guó)產(chǎn)化高性能處理器,瑞芯微RK3568芯片是一款定位中高端的通用型SoC,瑞芯微RK3568芯片是一款定位中高端的通用型SoC,NPU達(dá)到1Tops,飛凌RK3568系列核心板提供瑞芯微RK3568規(guī)格書_datasheet_數(shù)據(jù)手冊(cè)_原理圖等,


    了解詳情
    OK3568-C開(kāi)發(fā)板

推薦閱讀 換一批 換一批