• <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>

            JACKY_ZZ[貓貓愛吃魚]

            春風拂面兩頰紅,秋葉灑地一片金。 夏荷搖曳一身輕,冬雪覆蓋大地銀。
            posts - 30, comments - 123, trackbacks - 0, articles - 0

            [C/C++] union初探

            Posted on 2007-08-06 17:31 jacky_zz 閱讀(1225) 評論(4)  編輯 收藏 引用 所屬分類: C/C++
            這幾天在看MFC深入淺出,看到一個union的定義,代碼如下:
              1union MessageMapFunctions
              2{
              3    AFX_PMSG pfn;   // generic member function pointer
              4
              5    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_D)(CDC*);
              6    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_b)(BOOL);
              7    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_u)(UINT);
              8    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_h)(HANDLE);
              9    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_W_u_u)(CWnd*, UINT, UINT);
             10    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_W_COPYDATASTRUCT)(CWnd*, COPYDATASTRUCT*);
             11    BOOL (AFX_MSG_CALL CCmdTarget::*pfn_b_HELPINFO)(LPHELPINFO);
             12    HBRUSH (AFX_MSG_CALL CCmdTarget::*pfn_B_D_W_u)(CDC*, CWnd*, UINT);
             13    HBRUSH (AFX_MSG_CALL CCmdTarget::*pfn_B_D_u)(CDC*, UINT);
             14    int (AFX_MSG_CALL CCmdTarget::*pfn_i_u_W_u)(UINT, CWnd*, UINT);
             15    int (AFX_MSG_CALL CCmdTarget::*pfn_i_u_u)(UINT, UINT);
             16    int (AFX_MSG_CALL CCmdTarget::*pfn_i_W_u_u)(CWnd*, UINT, UINT);
             17    int (AFX_MSG_CALL CWnd::*pfn_i_s)(LPTSTR);
             18    LRESULT (AFX_MSG_CALL CWnd::*pfn_l_w_l)(WPARAM, LPARAM);
             19    LRESULT (AFX_MSG_CALL CWnd::*pfn_l_u_u_M)(UINT, UINT, CMenu*);
             20    void (AFX_MSG_CALL CWnd::*pfn_v_v)();
             21    int (AFX_MSG_CALL CWnd::*pfn_i_u)(UINT);
             22    HCURSOR (AFX_MSG_CALL CWnd::*pfn_C_v)();
             23    UINT (AFX_MSG_CALL CWnd::*pfn_u_u)(UINT);
             24    BOOL (AFX_MSG_CALL CWnd::*pfn_b_v)();
             25    void (AFX_MSG_CALL CWnd::*pfn_v_u)(UINT);
             26    void (AFX_MSG_CALL CWnd::*pfn_v_u_u)(UINT, UINT);
             27    void (AFX_MSG_CALL CWnd::*pfn_v_i_i)(intint);
             28    void (AFX_MSG_CALL CWnd::*pfn_v_u_u_u)(UINT, UINT, UINT);
             29    void (AFX_MSG_CALL CWnd::*pfn_v_u_i_i)(UINT, intint);
             30    void (AFX_MSG_CALL CWnd::*pfn_v_w_l)(WPARAM, LPARAM);
             31    void (AFX_MSG_CALL CWnd::*pfn_v_b_W_W)(BOOL, CWnd*, CWnd*);
             32    void (AFX_MSG_CALL CWnd::*pfn_v_D)(CDC*);
             33    void (AFX_MSG_CALL CWnd::*pfn_v_M)(CMenu*);
             34    void (AFX_MSG_CALL CWnd::*pfn_v_M_u_b)(CMenu*, UINT, BOOL);
             35    void (AFX_MSG_CALL CWnd::*pfn_v_W)(CWnd*);
             36    void (AFX_MSG_CALL CWnd::*pfn_v_W_u_u)(CWnd*, UINT, UINT);
             37    void (AFX_MSG_CALL CWnd::*pfn_v_W_p)(CWnd*, CPoint);
             38    void (AFX_MSG_CALL CWnd::*pfn_v_W_h)(CWnd*, HANDLE);
             39    void (AFX_MSG_CALL CWnd::*pfn_v_u_W)(UINT, CWnd*);
             40    void (AFX_MSG_CALL CWnd::*pfn_v_u_W_b)(UINT, CWnd*, BOOL);
             41    void (AFX_MSG_CALL CWnd::*pfn_v_u_u_W)(UINT, UINT, CWnd*);
             42    void (AFX_MSG_CALL CWnd::*pfn_v_s)(LPTSTR);
             43    void (AFX_MSG_CALL CWnd::*pfn_v_u_cs)(UINT, LPCTSTR);
             44    void (AFX_MSG_CALL CWnd::*pfn_v_i_s)(int, LPTSTR);
             45    int (AFX_MSG_CALL CWnd::*pfn_i_i_s)(int, LPTSTR);
             46    UINT (AFX_MSG_CALL CWnd::*pfn_u_p)(CPoint);
             47    UINT (AFX_MSG_CALL CWnd::*pfn_u_v)();
             48    void (AFX_MSG_CALL CWnd::*pfn_v_b_NCCALCSIZEPARAMS)(BOOL, NCCALCSIZE_PARAMS*);
             49    void (AFX_MSG_CALL CWnd::*pfn_v_v_WINDOWPOS)(WINDOWPOS*);
             50    void (AFX_MSG_CALL CWnd::*pfn_v_u_u_M)(UINT, UINT, HMENU);
             51    void (AFX_MSG_CALL CWnd::*pfn_v_u_p)(UINT, CPoint);
             52    void (AFX_MSG_CALL CWnd::*pfn_v_u_pr)(UINT, LPRECT);
             53    BOOL (AFX_MSG_CALL CWnd::*pfn_b_u_s_p)(UINT, short, CPoint);
             54    LRESULT (AFX_MSG_CALL CWnd::*pfn_l_v)();
             55
             56    // type safe variant for thread messages
             57    void (AFX_MSG_CALL CWinThread::*pfn_THREAD)(WPARAM, LPARAM);
             58
             59    // specific type safe variants for WM_COMMAND and WM_NOTIFY messages
             60    void (AFX_MSG_CALL CCmdTarget::*pfnCmd_v_v)();
             61    BOOL (AFX_MSG_CALL CCmdTarget::*pfnCmd_b_v)();
             62    void (AFX_MSG_CALL CCmdTarget::*pfnCmd_v_u)(UINT);
             63    BOOL (AFX_MSG_CALL CCmdTarget::*pfnCmd_b_u)(UINT);
             64
             65    void (AFX_MSG_CALL CCmdTarget::*pfnNotify_v_NMHDR_pl)(NMHDR*, LRESULT*);
             66    BOOL (AFX_MSG_CALL CCmdTarget::*pfnNotify_b_NMHDR_pl)(NMHDR*, LRESULT*);
             67    void (AFX_MSG_CALL CCmdTarget::*pfnNotify_v_u_NMHDR_pl)(UINT, NMHDR*, LRESULT*);
             68    BOOL (AFX_MSG_CALL CCmdTarget::*pfnNotify_b_u_NMHDR_pl)(UINT, NMHDR*, LRESULT*);
             69    void (AFX_MSG_CALL CCmdTarget::*pfnCmdUI_v_C)(CCmdUI*);
             70    void (AFX_MSG_CALL CCmdTarget::*pfnCmdUI_v_C_u)(CCmdUI*, UINT);
             71
             72    void (AFX_MSG_CALL CCmdTarget::*pfnCmd_v_pv)(void*);
             73    BOOL (AFX_MSG_CALL CCmdTarget::*pfnCmd_b_pv)(void*);
             74
             75//OLD
             76    // specific type safe variants for WM-style messages
             77//    BOOL    (AFX_MSG_CALL CWnd::*pfn_bD)(CDC*);
             78//    BOOL    (AFX_MSG_CALL CWnd::*pfn_bb)(BOOL);
             79//    BOOL    (AFX_MSG_CALL CWnd::*pfn_bWww)(CWnd*, UINT, UINT);
             80//    BOOL    (AFX_MSG_CALL CWnd::*pfn_bHELPINFO)(HELPINFO*);
             81//    BOOL    (AFX_MSG_CALL CWnd::*pfn_bWCDS)(CWnd*, COPYDATASTRUCT*);
             82//    HBRUSH  (AFX_MSG_CALL CWnd::*pfn_hDWw)(CDC*, CWnd*, UINT);
             83//    HBRUSH  (AFX_MSG_CALL CWnd::*pfn_hDw)(CDC*, UINT);
             84//    int     (AFX_MSG_CALL CWnd::*pfn_iwWw)(UINT, CWnd*, UINT);
             85//    int     (AFX_MSG_CALL CWnd::*pfn_iww)(UINT, UINT);
             86//    int     (AFX_MSG_CALL CWnd::*pfn_iWww)(CWnd*, UINT, UINT);
             87//    int     (AFX_MSG_CALL CWnd::*pfn_is)(LPTSTR);
             88//    LRESULT (AFX_MSG_CALL CWnd::*pfn_lwl)(WPARAM, LPARAM);
             89//    LRESULT (AFX_MSG_CALL CWnd::*pfn_lwwM)(UINT, UINT, CMenu*);
             90//    void    (AFX_MSG_CALL CWnd::*pfn_vv)(void);
             91
             92//    void    (AFX_MSG_CALL CWnd::*pfn_vw)(UINT);
             93//    void    (AFX_MSG_CALL CWnd::*pfn_vww)(UINT, UINT);
             94//    void    (AFX_MSG_CALL CWnd::*pfn_vvii)(intint);
             95//    void    (AFX_MSG_CALL CWnd::*pfn_vwww)(UINT, UINT, UINT);
             96//    void    (AFX_MSG_CALL CWnd::*pfn_vwii)(UINT, intint);
             97//    void    (AFX_MSG_CALL CWnd::*pfn_vwl)(WPARAM, LPARAM);
             98//    void    (AFX_MSG_CALL CWnd::*pfn_vbWW)(BOOL, CWnd*, CWnd*);
             99//    void    (AFX_MSG_CALL CWnd::*pfn_vD)(CDC*);
            100//    void    (AFX_MSG_CALL CWnd::*pfn_vM)(CMenu*);
            101//    void    (AFX_MSG_CALL CWnd::*pfn_vMwb)(CMenu*, UINT, BOOL);
            102
            103//    void    (AFX_MSG_CALL CWnd::*pfn_vW)(CWnd*);
            104//    void    (AFX_MSG_CALL CWnd::*pfn_vWww)(CWnd*, UINT, UINT);
            105//    void    (AFX_MSG_CALL CWnd::*pfn_vWp)(CWnd*, CPoint);
            106//    void    (AFX_MSG_CALL CWnd::*pfn_vWh)(CWnd*, HANDLE);
            107//    void    (AFX_MSG_CALL CWnd::*pfn_vwW)(UINT, CWnd*);
            108//    void    (AFX_MSG_CALL CWnd::*pfn_vwWb)(UINT, CWnd*, BOOL);
            109//    void    (AFX_MSG_CALL CWnd::*pfn_vwwW)(UINT, UINT, CWnd*);
            110//    void    (AFX_MSG_CALL CWnd::*pfn_vwwx)(UINT, UINT);
            111//    void    (AFX_MSG_CALL CWnd::*pfn_vs)(LPTSTR);
            112//    void    (AFX_MSG_CALL CWnd::*pfn_vOWNER)(int, LPTSTR);   // force return TRUE
            113//    int     (AFX_MSG_CALL CWnd::*pfn_iis)(int, LPTSTR);
            114//    UINT    (AFX_MSG_CALL CWnd::*pfn_wp)(CPoint);
            115//    UINT    (AFX_MSG_CALL CWnd::*pfn_wv)(void);
            116    void    (AFX_MSG_CALL CWnd::*pfn_vPOS)(WINDOWPOS*);
            117    void    (AFX_MSG_CALL CWnd::*pfn_vCALC)(BOOL, NCCALCSIZE_PARAMS*);
            118    void    (AFX_MSG_CALL CWnd::*pfn_vwp)(UINT, CPoint);
            119    void    (AFX_MSG_CALL CWnd::*pfn_vwwh)(UINT, UINT, HANDLE);
            120    BOOL    (AFX_MSG_CALL CWnd::*pfn_bwsp)(UINT, short, CPoint);
            121//    void    (AFX_MSG_CALL CWnd::*pfn_vws)(UINT, LPCTSTR);
            122};

            而調用的方法只是指定了一個函數指針,而函數指針的定義為:
            typedef void (AFX_MSG_CALL CCmdTarget::*AFX_PMSG)(void);
            起初一點也不明白這樣的定義怎么指向實際執行的函數?懷著這樣的疑問,我自己做了個測試,才明白其中的真實含義。union與struct的區別是:union為每個成員共享一個地址空間,而struct為每個成員都分配一個地址空間。這樣就好理解了,不管union里定義了多少個成員,每個成員都使用同一個內存地址,而struct的每個成員的內存地址卻都不相同。
            測試代碼如下:

             1#include "stdafx.h"
             2
             3typedef void (*FUNC)(void);
             4
             5typedef void (*Func01)(intint);
             6typedef double (*Func02)(doubledouble);
             7typedef int (*Func03)(intint);
             8
             9void Test01(intint);
            10double Test02(doubledouble);
            11int Test03(intint);
            12
            13union UnionTest {
            14    FUNC pFunc;
            15
            16    Func01 func01;
            17    Func02 func02;
            18    Func03 func03;
            19};
            20
            21int _tmain(int argc, char* argv[])
            22{
            23    system("cls");
            24    printf("此程序模仿了MFC中的消息函數分配機制\n");
            25    union UnionTest test;
            26
            27    printf("(FUNC)(static_cast< void (*)(int, int) > (Test01))\n");
            28    test.pFunc = (FUNC)(static_cast< void (*)(intint> (Test01));
            29    printf("%p, %p, %p\n", test.func01, test.func02, test.func03);
            30
            31    test.func01(10,10);
            32    test.func02(10.00120.002);
            33    test.func03(10,10);
            34    printf("\n");
            35
            36    printf("(FUNC)(static_cast< double (*)(double, double) > (Test02))\n");
            37    test.pFunc = (FUNC)(static_cast< double (*)(doubledouble> (Test02));
            38    printf("%p, %p, %p\n", test.func01, test.func02, test.func03);
            39
            40    test.func01(10,10);
            41    double xx = test.func02(10.00120.002);
            42    test.func03(10,10);
            43    printf("\n");
            44
            45    printf("(FUNC)(static_cast< int (*)(int, int) > (Test03))\n");
            46    test.pFunc = (FUNC)(static_cast< int (*)(intint> (Test03));
            47    printf("%p, %p, %p\n", test.func01, test.func02, test.func03);
            48
            49    test.func01(10,10);
            50    double y01 = test.func02(10.00120.002);
            51    int yy = test.func03(10,10);
            52
            53    return 0;
            54}
            55
            56void Test01(int x, int y) {
            57    printf("hehe, test it. x=%d, y=%d\n", x, y);
            58}
            59
            60double Test02(double x, double y) {
            61    double sum =x+y;
            62    printf("sum double: %10.10f+%10.10f=%10.10f\nx address:%p\ny address:%p\n",x,y,sum, &x, &y);
            63    return sum;
            64}
            65
            66int Test03(int x, int y) {
            67    int sum =x+y;
            68    printf("sum int: %d+%d=%d\n",x,y,sum);
            69    return sum;
            70}
            輸出結果如下:
             1此程序模仿了MFC中的消息函數分配機制
             2(FUNC)(static_cast< void (*)(intint> (Test01))
             3004010000040100000401000
             4hehe, test it. x=10, y=10
             5hehe, test it. x=309237645, y=1076101251
             6hehe, test it. x=10, y=10
             7
             8(FUNC)(static_cast< double (*)(doubledouble> (Test02))
             9004010200040102000401020
            10sum double0.0000000000+0.0000000000=0.0000000000
            11x address:0012FEB0
            12y address:0012FEB8
            13sum double10.0010000000+20.0020000000=30.0030000000
            14x address:0012FED8
            15y address:0012FEE0
            16sum double0.0000000000+10.0010000000=10.0010000000
            17x address:0012FED0
            18y address:0012FED8
            19
            20(FUNC)(static_cast< int (*)(intint> (Test03))
            21004010700040107000401070
            22sum int10+10=20
            23sum int309237645+1076101251=1385338896
            24sum int10+10=20
            從輸出結果可以看出,union的每個成員的內存地址都是一樣的。 這樣就好理解union MessageMapFunctions的真實作用了。

            Feedback

            # re: union初探  回復  更多評論   

            2007-11-25 10:36 by aspmaster
            不錯,雖然候Sir在深入淺出中說了,但當時還是沒有看明白

            # re: union初探  回復  更多評論   

            2012-07-09 10:34 by HooperStephanie21
            The <a href="http://goodfinance-blog.com">loans</a> are useful for people, which would like to ground their own business. As a fact, this is comfortable to get a short term loan.

            # re: union初探  回復  更多評論   

            2013-03-25 14:10 by site
            Look over EssaysLeader rewiew (best-essay-sites.com), select top-notch firm and purchase professionally written essays from competent writers.
            中文字幕无码久久久| 99久久99久久精品国产| 精品久久久中文字幕人妻| 久久久亚洲裙底偷窥综合| 久久精品九九亚洲精品天堂| 国产综合成人久久大片91| 久久久久免费精品国产| 久久精品国产69国产精品亚洲| 久久免费国产精品| 久久精品国产亚洲AV无码娇色| 久久亚洲欧洲国产综合| 日韩人妻无码精品久久免费一| 国内精品久久久久久麻豆| 午夜不卡久久精品无码免费| 久久成人精品| 一本色道久久88加勒比—综合| 久久精品一本到99热免费| 久久人人爽人人爽人人片AV麻豆 | 亚洲国产精品热久久| 一级做a爰片久久毛片看看| 91精品国产综合久久香蕉| 久久综合给久久狠狠97色| 狠狠色丁香久久婷婷综合图片| 久久精品成人免费观看97| 狠狠狠色丁香婷婷综合久久五月 | 久久er国产精品免费观看8| 久久99亚洲网美利坚合众国| 99久久免费国产精品特黄| 欧美麻豆久久久久久中文| 欧美粉嫩小泬久久久久久久| 久久综合九色综合精品| 精品免费tv久久久久久久| 久久精品国产亚洲AV大全| 久久婷婷五月综合97色一本一本| 精品久久久久久无码不卡| 色综合久久中文字幕综合网| 亚洲国产成人精品无码久久久久久综合 | 久久久久久伊人高潮影院| 久久人妻少妇嫩草AV蜜桃| 伊人 久久 精品| 国产69精品久久久久久人妻精品|