青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

C++中使用union的幾點思考

大衛注:
這段時間整理舊資料,看到一些文章,雖然講的都是些小問題,不大可能用到,但也算是一個知識點,特整理出來與大家共享.與此相關的那篇文章的作者的有些理解是錯誤的,我寫此文,也是糾正為了作者的一些錯誤認識.當然,如果我的理解有任何錯誤,也懇請大家批評指正.

C++
雖說被B.S.稱作一門新語言,但它畢竟與C有著千絲萬縷的聯系,雖然B.S.一再堅持,但我還是愿意把C++看作是C ++.
我們應該按照C中的convention去使用union,這是我這篇文章要給出的觀點.雖然C++使得我們可以擴展一些新的東西進去,但是,我建議你不要那樣去做,看完這篇文章之后,我想你大概也是這么想的.

C
由于沒有類的概念,所有類型其實都可以看作是基本類型的組合,因此在union中包含struct也就是一件很自然的事情了,到了C++之后,既然普遍認為C++中的structclass基本等價,那么union中是否可以有類成員呢?先來看看如下的代碼:

struct
TestUnion
{

    TestUnion() {}
};


typedef union

{

    TestUnion obj;
}
UT;

int
main (void)
{

    return
0;
}


編譯該程序,我們將被告知:
error C2620: union '__unnamed' : member 'obj' has user-defined constructor or non-trivial default constructor
而如果去掉那個什么也沒干的構造函數,則一切OK.

為什么編譯器不允許我們的union成員有構造函數呢?我無法找到關于這個問題的比較權威的解釋,對這個問題,我的解釋是:
如果C++標準允許我們的union有構造函數,那么,在進行空間分配的時候要不要執行這個構造函數呢?如果答案是yes,那么如果TestUnion的構造函數中包含了一些內存分配操作,或者其它對整個application狀態的修改,那么,如果我今后要用到obj的話,事情可能還比較合理,但是如果我根本就不使用obj這個成員呢?由于obj的引入造成的對系統狀態的修改顯然是不合理的;反之,如果答案是no,那么一旦我們今后選中了obj來進行操作,則所有信息都沒有初始化(如果是普通的struct,沒什么問題,但是,如果有虛函數呢?).更進一步,假設現在我們的union不是只有一個TestUnion obj,還有一個TestUnion2 obj2,二者均有構造函數,并且都在構造函數中執行了一些內存分配的工作(甚至干了很多其它事情),那么,如果先構造obj,后構造obj2,則執行的結果幾乎可以肯定會造成內存的泄漏.
鑒于以上諸多麻煩(可能還有更多麻煩),在構造union,編譯器只負責分配空間,而不負責去執行附加的初始化工作,為了簡化工作,只要我們提供了構造函數,就會收到上面的error.
同理,除了不能加構造函數,析構函數/拷貝構造函數/賦值運算符也是不可以加.

此外,如果我們的類中包含了任何virtual函數,編譯時,我們將收到如下的錯誤信息:
error C2621: union '__unnamed' : member 'obj' has copy constructor

所以,打消在union中包含有構造函數/析構函數/拷貝構造函數/賦值運算符/虛函數的類成員變量的念頭,老老實實用你的C風格struct!
不過,定義普通的成員函數是OK,因為這不會使得classC風格的struct有任何本質區別,你完全可以將這樣的class理解為一個C風格的struct + n個全局函數.

現在,再看看在類中包含內部union時會有什么不同.看看下面的程序,并請注意閱讀程序提示:

class
TestUnion
{

    union
DataUnion
    {

        DataUnion(const char*);
        DataUnion(long);
        const
char* ch_;
        long
       l_;
    }
data_;

public
:
    TestUnion(const char* ch);
    TestUnion(long l);
};


TestUnion::TestUnion(const char* ch) : data_(ch) // if you want to use initialzing list to initiate a nested-union member, the union must not be anonymous and must have a constructor.
{
}


TestUnion::TestUnion(long l) : data_(l)
{
}


TestUnion::DataUnion::DataUnion(const char* ch) : ch_(ch)
{
}


TestUnion::DataUnion::DataUnion(long l) : l_(l)
{
}


int
main (void)
{

    return
0;
}


正如上面程序所示,C++中的union也可以包含構造函數,但是,這雖然被語言所支持,但實在是一種不佳的編程習慣,因此,我不打算對上面的程序進行過多的說明.我更推薦如下的編程風格:

class
TestUnion
{

    union
DataUnion
    {

        const
char* ch_;
        long
       l_;
    }
data_;
   
public
:
    TestUnion(const char* ch);
    TestUnion(long l);
};


TestUnion::TestUnion(const char* ch)
{

    data_.ch_ = ch;
}


TestUnion::TestUnion(long l)
{

    data_.l_ = l;
}


int
main (void)
{

    return
0;
}


它完全是C風格的.

所以,接受這個結論吧:
請按照C中的convention去使用union,盡量不要嘗試使用任何C++附加特性.

posted on 2004-11-12 09:08 大衛的思維空間 閱讀(8069) 評論(16)  編輯 收藏

# "比較權威的解釋", B.S權威不?可惜你不看他的書。

以下摘自《 the c++ programming language 3rd 》,對為什么union不支持構造函數和析構函數,推薦怎么使用unions都有說明。

10.4.12
Unions [class.union]
A named union is defined as a struct, where every member has the same address (see §C.8.2). A
union can have member functions but not static members.
In general, a compiler cannot know what member of a union is used; that is, the type of the
object stored in a union is unknown. Consequently, a union may not have members with constructors
or destructors. It wouldn’t be possible to protect that object against corruption or to guarantee
that the right destructor is called when the union goes out of scope.
Unions are best used in lowlevel
code, or as part of the implementation of classes that keep
track of what is stored in the union (see §10.6[20]).

其實正如在vckbase/c++論壇上我對你說的,想想union的用途就可以了。union的出現是為了解決一組不定類型數據的空間緊張問題,它是為省空間用的,不是為了封裝數據和方法。所以我覺得大衛兄太過于刨根究底了,而忽略了C++的實用性(實用性是C/C++的首要設計思想)

2004-11-12 10:18 | 一笑

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

謝謝笑兄給出的CPL中的相關論述.我現在主要關注的內容并非C++(雖然C++是我最喜歡的語言),我在blog上發布這些文章也只是希望談一下自己對于某些語言特性或新技術的認識,以期對后來者有些許幫助,同時,也希望在與大家的交流中提高自己.
^_^,
也許我真的有點刨根究底,不過笑兄的關于"union是為省空間用的"的觀點我不認同,雖然,我也看到過類似的說法,但是,union怎么可能節省空間呢?:
union uT
{
int i;
char c;
};
當他被作為char使用時,明明浪費了空間.
只是從另一角度出發:把它當成一種特殊的struct,struct相比時才可以勉強說是節省了空間.
我認為union的引入,更多地是提供了一種解決多種型別共同使用同一空間(這與節省空間是兩回事)的機制,借助這種機制,我們可以提供更豐富的語言表現力.
為此,我決定再寫一篇文章,談一些比較有意思/有意義的union的使用,請關注 & 批評指正.

2004-11-13 05:11 | Bill David

# union的進一步認識與一些深層應用[TrackBack]

Ping Back來自:blog.csdn.net
Bill David
引用了該文章,地址:http://blog.csdn.net/billdavid/archive/2004/11/26/195006.aspx

2004-11-26 18:58 | Bill David

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

呵呵,仁者見仁,智者見智。
但是union的確應該是節省空間使用的。當時的初衷是這樣的。但是后來就不一定了。3ks 二位。

2005-04-19 17:38 | 漫天飛舞

# 今日經論壇匿名大俠指點,union還有一個重要的特性

union不支持繼承。

特記錄于此。

2005-04-19 17:59 | Bill David

# re: union不支持繼承。

我突然覺得如果union設計為可繼承的也未嘗不可。
union base
{
    int mem1;
    char mem2;
};
這個假設被原實現者設計好了,突然,某天我想在base的基礎上添加一些特性,于是我寫
union derive : base
{
    short mem 3;
};
或許可能實際意義不大吧!但,如果這么說,我reinterpret_cast成員mem1呢?union的實際意義又有多大呢?

不知道你怎么看?

2006-05-16 13:35 | 清風雨

# re: union的幾點思考

又看了你的另一篇union的,可能我比較懶惰的緣故,我只得到一個結論: union僅僅只是為了方便。

2006-05-16 13:49 | 清風雨

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

這都是些什么?

2006-05-18 21:35 | 一二三

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

看不懂

2006-05-18 21:36 | 一二三

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

union
{
    struct
           {
                string name;
           }info;
    int  age;
}information;

上面的語句可能出錯,has copy constructor.   難道只有把string -->char *??

2006-07-07 15:38 | Tmin

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

看了文章和回帖,我非常同意一笑兄的看法。我覺得作者沒有完全理解節省空間的指的是如何節省,例如我們獲取一些信息時,需要把信息數據填充到一個結構中,但是我們需要的信息的數據的類型會根據情況不同而不同,如果我們為了通用的話,可能就需要為所有可能出現的類型都預留相應的空間來存儲,但是每次只能使用其中一個有效。其他預留無效。這顯然是浪費空間的。所以union使得不同類型數據存放在內存的同一個位置,當然這個空間的大小是能容納所有類型的數據的。因為unicon的數據類型是運行時決定的,事先并不知道,所以為union做一個構造函數是沒有意義的。

2006-10-17 11:48 | hoodlum

# re: (大衛的閱讀筆記)C++中使用union的幾點思考

剛才回的倉促,可能有論述不是很恰當。但是看了作者的另一篇文章里面引用的英文,論述的也和我本來的想法是一致的。也就是說,可以使用union.memberData來訪問成員,但是應該注意的是,使用時只有其中一個成員才是有意義的。union占用的空間大小是所有成員數據大小的最小公倍數。(因此,轉為托管代碼中就可以使用一個能夠容納union大小的數據類型來代替,或者用通用的字節數組)

例如下面的例子(來自Windows API):
union
{    
   DWORD dwOemId;    
   struct {
                 WORD wProcessorArchitecture; 
                 WORD wReserved; 
              };
  };

它通常占用4個字節,因此可以使用托管代碼中的Int32來代替。
其中dwOemId是為了向前兼容某些系統版本,當前已經棄用。這樣,一個程序對API的調用可以在不同版本的系統中運轉良好,處理各自自己需要的數據。

 

posted on 2008-11-21 22:07 肥仔 閱讀(2303) 評論(1)  編輯 收藏 引用 所屬分類: C++ 基礎

評論

# re: C++中使用union的幾點思考  回復  更多評論   

# re: (大衛的閱讀筆記)C++中使用union的幾點思考
看了文章和回帖,我非常同意一笑兄的看法。我覺得作者沒有完全理解“節省空間”的指的是如何節省,例如我們獲取一些信息時,需要把信息數據填充到一個結構中,但是我們需要的信息的數據的類型會根據情況不同而不同,如果我們為了通用的話,可能就需要為所有可能出現的類型都預留相應的空間來存儲,但是每次只能使用其中一個有效。其他預留無效。這顯然是浪費空間的。所以union使得不同類型數據存放在內存的同一個位置,當然這個空間的大小是能容納所有類型的數據的。因為unicon的數據類型是運行時決定的,事先并不知道,所以為union做一個構造函數是沒有意義的。

2006-10-17 11:48 | hoodlum
------------------------------------------------
個人覺得hoodlum說得非常好!
2010-05-11 10:55 | Coastline
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国内精品久久久久久久影视麻豆 | 一本色道婷婷久久欧美| 国产人成精品一区二区三| 国产欧美va欧美va香蕉在| 国产伦精品一区二区三| 国内精品一区二区三区| 亚洲国产成人在线播放| 亚洲精品美女在线观看播放| 亚洲精品久久视频| 亚洲一区二区免费视频| 午夜日韩在线| 免费成人av在线| 日韩视频精品在线| 午夜在线一区二区| 欧美国产日韩精品| 国产欧美日韩精品专区| 精品动漫3d一区二区三区| 日韩视频免费观看| 欧美专区在线观看一区| 亚洲高清久久久| 中文无字幕一区二区三区| 久久se精品一区二区| 欧美日韩免费一区| 国产综合自拍| 亚洲综合国产激情另类一区| 女生裸体视频一区二区三区| 国产欧美精品xxxx另类| 免费观看一区| 国产欧美日韩综合| 亚洲毛片播放| 老司机aⅴ在线精品导航| 亚洲日本激情| 久久一二三区| 国产欧美在线视频| 一区二区免费看| 蜜桃av综合| 亚洲一区二区三区视频播放| 欧美成人精品福利| 韩国自拍一区| 欧美一区视频在线| 一区二区三区高清不卡| 免费亚洲一区二区| 好吊妞**欧美| 久久久久欧美精品| 午夜伦欧美伦电影理论片| 欧美视频在线观看免费| 99国产精品久久久久久久久久 | 在线日本欧美| 久久国产视频网站| 亚洲欧美日韩一区| 国产精品一二三四| 午夜精品久久久久久久久| 日韩视频一区二区三区| 欧美激情一区二区三区全黄| 亚洲国产精品一区二区尤物区| 久久深夜福利| 久久久蜜桃一区二区人| 经典三级久久| 麻豆av一区二区三区久久| 欧美一区二视频| 国产一区二区三区在线观看免费视频 | 99国产精品自拍| 欧美精品九九99久久| 亚洲国产一区二区视频| 欧美成年人视频网站| 久久性色av| 亚洲国产裸拍裸体视频在线观看乱了中文| 久久国产精品亚洲77777| 亚洲午夜一区二区| 国产一区二区三区av电影| 久久这里有精品视频| 久久久久一区二区三区四区| 亚洲电影自拍| 亚洲另类一区二区| 久久精品视频播放| 欧美国产精品v| 亚洲人成免费| 亚洲人成网站色ww在线| 久久九九久久九九| 久久精品视频99| 一区二区三区在线免费观看 | 国产女同一区二区| 久久综合福利| 欧美精品aa| 午夜久久一区| 久久久久久穴| 亚洲视频精品在线| 欧美在线播放视频| 夜夜狂射影院欧美极品| 亚洲一区二区毛片| 在线欧美日韩| 亚洲一区免费看| 亚洲国产日韩欧美在线图片| 日韩亚洲视频在线| 狠狠色丁香婷综合久久| 亚洲三级毛片| 国内视频精品| 日韩亚洲在线| 亚洲成人在线视频播放| aa成人免费视频| 尤物精品在线| 亚洲永久精品国产| 亚洲欧洲在线一区| 欧美一区二区黄色| 亚洲免费小视频| 欧美黑人在线播放| 久久午夜激情| 国产日产亚洲精品系列| 99精品国产高清一区二区| 伊人婷婷久久| 性欧美大战久久久久久久免费观看| 亚洲裸体视频| 麻豆精品精华液| 久久嫩草精品久久久精品| 国产精品久久999| 日韩视频不卡中文| 亚洲精品视频在线观看免费| 欧美中文字幕不卡| 午夜亚洲伦理| 国产精品久久久久久久电影| 亚洲国产成人在线视频| 精品成人a区在线观看| 午夜精品亚洲| 亚洲欧美日韩精品久久奇米色影视| 欧美激情精品久久久久久免费印度 | 久久久久久九九九九| 国产精品高潮呻吟久久| 亚洲精品乱码久久久久久蜜桃麻豆 | 欧美激情在线狂野欧美精品| 激情国产一区二区| 欧美一区二区大片| 久久久久这里只有精品| 国产麻豆精品在线观看| 亚洲桃色在线一区| 亚洲欧美在线另类| 国产免费成人av| 欧美一区二视频| 久久全球大尺度高清视频| 国内外成人免费激情在线视频| 性欧美大战久久久久久久免费观看 | 欧美激情久久久久久| 欧美肥婆在线| 日韩视频一区二区| 欧美日韩午夜精品| 亚洲一区二区三区涩| 欧美一区二区三区免费视| 国产偷久久久精品专区| 久久精品国产免费看久久精品| 久久一二三区| 亚洲三级观看| 欧美午夜不卡在线观看免费| 亚洲视频观看| 久久中文字幕一区二区三区| 亚洲国产精品成人综合| 欧美激情精品久久久久久黑人| 日韩亚洲成人av在线| 久久国产精品久久精品国产| 在线观看国产成人av片| 欧美日韩国产区| 欧美一区二区福利在线| 亚洲国产成人av在线| 亚洲女女做受ⅹxx高潮| 国产视频久久久久| 欧美国产精品人人做人人爱| 一区二区福利| 久久亚洲精品视频| 亚洲精品影院| 国产区亚洲区欧美区| 久久综合中文色婷婷| 亚洲视频在线观看三级| 久久综合色88| 亚洲私人影吧| 在线欧美一区| 国产农村妇女精品一区二区| 欧美成人性网| 欧美在线视频播放| 日韩视频一区二区三区在线播放免费观看 | 男人插女人欧美| 亚洲一区在线观看免费观看电影高清 | 久久婷婷国产综合精品青草| 亚洲毛片一区| 国内精品免费在线观看| 欧美午夜宅男影院| 欧美成人自拍视频| 久久精品成人一区二区三区蜜臀 | 亚洲免费视频观看| 亚洲美女毛片| 亚洲成人在线| 国产主播喷水一区二区| 国产精品女主播在线观看| 麻豆成人91精品二区三区| 亚洲一区二区三区乱码aⅴ| 91久久极品少妇xxxxⅹ软件| 久久久综合香蕉尹人综合网| 亚洲影院免费| 亚洲欧美日韩精品久久亚洲区| 一区二区三区www| 一区二区日韩精品| 亚洲美女精品久久| 亚洲区一区二|