C++11中union的使用方法示例
前言
union即為聯(lián)合,它是一種特殊的類。通過關(guān)鍵字union進(jìn)行定義,一個(gè)union可以有多個(gè)數(shù)據(jù)成員。例如
union Token{
char cval;
int ival;
double dval;
};
c++11中union除了繼承c語言的數(shù)據(jù)共享內(nèi)存之外,行為上越來越像一個(gè)類,比如成員默認(rèn)是public類型。
在C++11以后,很多基礎(chǔ)語法都進(jìn)行了修正。其中 union 的行為向類對象進(jìn)行了發(fā)展,在兼容原有語法定義的基礎(chǔ)上進(jìn)行了擴(kuò)充:
- union可以擁有成員函數(shù)(包含構(gòu)造函數(shù)和析構(gòu)函數(shù)),但是不能有虛函數(shù)
- union不能參與繼承,不能成為基類也不能成為子類
- union的成員對象不能為引用類型
對于全部成員都是 build-in 的 union 類,還可以向以往一樣進(jìn)行使用:
#include <iostream>
#include <cstdint>
union S
{
std::int32_t n; // 占用4字節(jié)
std::uint16_t s[2]; // 占用4字節(jié)
std::uint8_t c; // 占用1字節(jié)
}; // 整體占用4字節(jié)
int main()
{
S s = {0x12345678}; // 初始化第一個(gè)成員,當(dāng)前s.n為活躍成員
// 于此點(diǎn),讀取 s.s 或 s.c 是未定義行為
std::cout << std::hex << "s.n = " << s.n << '\n';
s.s[0] = 0x0011; // s.s 現(xiàn)在是活躍成員
// 在此點(diǎn),讀取 s.n 或 s.c 是未定義行為
std::cout << "s.c is now " << +s.c << '\n' // 11 or 00, 依賴平臺(tái)實(shí)現(xiàn)
<< "s.n is now " << s.n << '\n'; // 12340011 or 00115678
}
對于全部包含非 built-in 的 union 類,則:
- 如果非靜態(tài)(non-static)成員帶有非平凡的特殊(non-trivial special)成員函數(shù)(自定義的:復(fù)制/移動(dòng)構(gòu)造函數(shù),復(fù)制/移動(dòng)賦值函數(shù),析構(gòu)函數(shù)),則該 union 類的默認(rèn)相關(guān)的復(fù)制/移動(dòng)構(gòu)造函數(shù)、復(fù)制/移動(dòng)賦值函數(shù)、析構(gòu)函數(shù)都會(huì)被刪除,如果需要的話,要求用戶自己定義實(shí)現(xiàn),如果用戶沒有自己定義,則不能進(jìn)行相關(guān)的復(fù)制/移動(dòng)操作
- 如果非靜態(tài)(non-static)成員帶有非平凡的特殊(non-trivial special)構(gòu)造函數(shù)(自定義的構(gòu)造函數(shù)),則該 union 類的默認(rèn)構(gòu)造函數(shù)會(huì)被刪除,如果需要的話,要求用戶自己定義實(shí)現(xiàn),如果用戶沒有自己定義,則不能該 union 類不能進(jìn)行實(shí)例化。
- 至多一個(gè)變體成員能擁有默認(rèn)成員初始化值
- 不能有靜態(tài)成員數(shù)據(jù)(這個(gè)很奇怪,在clang上會(huì)報(bào)鏈接錯(cuò)誤而不是語法錯(cuò)誤),但是可以有靜態(tài)成員函數(shù)
- 全部成員的訪問控制都是 public
第1、2點(diǎn)的意思是,如果成員數(shù)據(jù)類型是非平凡的(non-trivial),則 union 類需要定義相關(guān)的構(gòu)造函數(shù)、復(fù)制函數(shù)、移動(dòng)構(gòu)造、移動(dòng)賦值函數(shù)、析構(gòu)函數(shù)等。
union A {
int a;
double b;
std::string c;
A() : c("111") {} // 因?yàn)閟td::string擁有是非平凡的的數(shù)據(jù)類型,
~A() {} // 則A必須自定義構(gòu)造函數(shù)和析構(gòu)函數(shù),否則無法進(jìn)行實(shí)例化
// 如果想實(shí)現(xiàn)復(fù)制語義,還得自定義復(fù)制(構(gòu)造)函數(shù)
};
第3點(diǎn)的意思是:
union A {
int a;
double b;
std::string c = "abc"; // 只有一個(gè)成員數(shù)據(jù)能擁有這種初始化值
~A(){};
};
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
C語言基于graphics.h實(shí)現(xiàn)圣誕樹
這篇文章主要介紹了圣誕樹代碼,c語言編程,基于graphics.h實(shí)現(xiàn),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12
C# interface與delegate效能比較的深入解析
本篇文章是對C#中interface與delegate的效能比較進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
C++實(shí)現(xiàn)多線程并發(fā)場景下的同步方法
在C++中實(shí)現(xiàn)多線程并發(fā)場景下的同步方法,包括使用互斥鎖、獨(dú)占鎖、共享鎖和原子操作的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2025-02-02
C語言深入刨析數(shù)據(jù)結(jié)構(gòu)之棧與鏈棧的設(shè)計(jì)與應(yīng)用
棧是限定僅在表尾進(jìn)行插入或刪除操作的線性表,表尾稱為棧頂(top),表頭稱為棧底(bottom)。棧的最主要特點(diǎn)就是“先進(jìn)后出”(FILO),或“后進(jìn)先出”(LIFO)。用鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)表示的棧稱為“鏈?!保湕?yīng)于鏈表2022-05-05
C++實(shí)現(xiàn)字符串類型相互轉(zhuǎn)換的代碼示例
在C/C++編程中,字符串是非?;A(chǔ)且常用的數(shù)據(jù)類型,但是由于不同的編程語言或標(biāo)準(zhǔn)庫可能采用不同的字符串類型,因此在不同的應(yīng)用場景下可能需要進(jìn)行字符串類型的相互轉(zhuǎn)換,本文將介紹如何在C/C++中將char*,std::string,QString,CString/MFC?String相互轉(zhuǎn)換2023-06-06

