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

            Welcome to ErranLi's Blog!

              C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
              106 Posts :: 1 Stories :: 97 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(12)

            搜索

            •  

            積分與排名

            • 積分 - 175054
            • 排名 - 151

            最新評論

            閱讀排行榜

             

            怎樣實現數目不定的函數參數

             

            2006-12-28

            erran

             

            實現方法一:自定義指針找到函數參數的下一個參數;

             

            // 例子 A

            template<class T>

            T umin(T _a, ...)

            {           

                   T _r = _a;

                   T* _p = &_a;

                  

                   int count = 1;

                  

                   while ((*_p) != -1)

                   {    

                          if (_r > (*_p)) _r = (*_p);

                          cout << "No." << count << ": " << (*_p) << endl;


                         
            //
            計算第二個參數的地址,并且輸出該地址

            // 注: 在32位(4字節)機器上表現為所有的變量地址都能被4整除

            // 相當于 _INTSIZEOF(n):  ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

            // 問題是用這個式子去算,是無法知道哪個才是最后哪個參數。所以用- 1 做結束參數。

                          _p = (T*)((char *)_p + (sizeof(T) + sizeof(int) - 1) / sizeof(int) * sizeof(int));


                   
                   count++;

                   }

                  

                   return _r;

            }

            // 調用:

            int main(int argc, char* argv[])

            {

                   int a, b, c, d, e;      

                   a = 1;    b = 3;       c = 2;       d = 12;  e = 34;

                  

                   cout << umin<int>( a, b, c, d, e, -1) << endl;      

            cout << umin<int>(1, 4, 2, 5, 2, -1) << endl;

            cout << umin<char>('a', 'b', 'c', 'd', 'e', -1) << endl;

             

                   return 0;

            }

             

             

             

            實現方法二:利用 ANSIC提供的三個宏 va_arg, va_end, va_start 來實現 ;

             

            這三個宏定義在頭文件 stdarg.h , 主要有: va_arg, va_end, va_start va_list


            _WIN32 下:

            typedef char * va_list;


            其它的三個
            ( 不同的平臺會有不太一樣 )


            /* A c at the proper definitions for other platforms */

            #define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

            #define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )

            #define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

            #define va_end(ap) ( ap = (va_list)0 )


            其實這個方法的實現原理同方法一是一樣的。


            //
            例子 B

            template<class T>

            T umin(T _a, ...)

            {    

                   T _p = _a;

                   T _r = _a;

             

                   int count = 1;

                  

                   va_list arg_ptr;

                   va_start(arg_ptr, _a);

                  

                   while (_p != -1)// 所以用- 1 做結束參數。

                   {    

                          if (_r > _p) _r = _p;

                          cout << "No." << count << ": " << _p << endl;                                                       

                          _p = va_arg(arg_ptr, int);

                          count++;

                   }

             

                   va_end(arg_ptr);

                  

                   return _r;

            }


            //
            調用:

            int main(int argc, char* argv[])

            {

                   int a, b, c, d, e;      

                   a = 1;    b = 3;       c = 2;       d = 12;  e = 34;

                  

                   cout << umin<int>( a, b, c, d, e, -1) << endl;      

            cout << umin<int>(1, 4, 2, 5, 2, -1) << endl;

            cout << umin<char>('a', 'b', 'c', 'd', 'e', -1) << endl;

             

                   return 0;

            }

             

             

            方法一和方法二都會遇到同一個問題:那就是沒有辦法知道該函數中一共到底有多少個參數。因為你 沒有辦法去判斷到底哪個指針才是最后的哪個參數的地址。

             

            想到一個解決方法是在該函數最后添加一個標志參數,以標志該參數是最后一個參數,但是應該注意的是:作為標志參數的條件是,該標志是不可能出現在此前的參數表中。如以上函數調用的最后那個參數(- 1 )。

             

            我想 c 語言里面的 scanf printf 等函數應該做過一些特殊處理。

             

            函數的參數地址是連續的,但是 可變參數列表提供了很大的靈活性,且對可變部分的參數C語言編譯器不會進行類型檢查,所以程序中要特別小心,必須確保參數的傳遞和接受是正確的;而且C語言的實現也依賴于cpu和操作系統,所以并不適用于所有的計算機和系統,而且隨機器的不同會由很大的區別。所以 stdarg.h 定義很多宏。如:

            #if     !defined(_WIN32) && !defined(_MAC)

            #error ERROR: Only Mac or Win32 targets supported!

            #endif

             

             

            l          關于函數參數的地址

             

            寫了一下測試代碼,來測試函數的參數地址是不是連續的。

            測試代碼如下 :


            //
            例子 C

            #include <iostream>

            using namespace std;

             

            template<class T>

            T umin(T _a, T _b, T _c, T _d, T _e)

            {    

            // 輸出各個參數的地址

                   cout << "_a : " << hex << (int*)(&_a) << endl;

                   cout << "_b : " << hex << (int*)(&_b) << endl;

                   cout << "_c : " << hex << (int*)(&_c) << endl;

                   cout << "_d : " << hex << (int*)(&_d) << endl;

                   cout << "_e : " << hex << (int*)(&_e) << endl;

                  

                   // 取得最小值

                   T _r = _a;

                 if (_r > _b) _r = _b;

                 if (_r > _c) _r = _c;

                 if (_r > _d) _r = _d;

                  

                   return _r;

            }

             

            int main(int argc, char* argv[])

            {

                   int a, b, c, d, e;      

                   a = 1;    b = 3;       c = 2;       d = 12;  e = 34;

             

                   cout << umin<int>(a, b, c, d, e) << endl;// 調用一

                   cout << umin<int>(1, 4, 2, 5, 2) << endl; // 調用二

                   cout << umin<char>('a', 'b', 'c', 'd', 'e') << endl;// 調用三

             

                   return 0;

            }

             

            VC6.0 下編譯,發現:

            對于例子 C 在編譯器不作任何優化的時候(及編譯選項選擇 Disable(Debug) ),對于調用一、二、三的參數 _a, _b, _c, _d, _e 都是連續的。在選擇優化運行速度 ( 及編譯選項選擇 Maximize Speed), min 的參數地址并不都是連續的。而用對于例子 A 和例子 B :兩種情況都是連續的 ( 測試的時候要更改相應的代碼 ).

             

            C Disable(Debug)

            C Maximize Speed

            A Maximize Speed

            A Disable(Debug)

            _a : 0012FF58

            _b : 0012FF5C

            _c : 0012FF60

            _d : 0012FF64

            _e : 0012FF68

             

            _a : 0012FF58

            _b : 0012FF5C

            _c : 0012FF60

            _d : 0012FF64

            _e : 0012FF68

             

            _a : 0012FF58

            _b : 0012FF5C

            _c : 0012FF60

            _d : 0012FF64

            _e : 0012FF68

            _a : 0012FF70

            _b : 0012FF74

            _c : 0012FF78

            _d : 0012FF7C

            _e : 0012FF80

             

            _a : 0012FF70

            _b : 0012FF80

            _c : 0012FF7C

            _d : 0012FF78

            _e : 0012FF74

             

            _a : 0012FF58

            _b : 0012FF5C

            _c : 0012FF60

            _d : 0012FF64

            _e : 0012FF68

            No.1: 0012FF50

            No.2: 0012FF54

            No.3: 0012FF58

            No.4: 0012FF5C

            No.5: 0012FF60

             

            No.1: 0012FF50

            No.2: 0012FF54

            No.3: 0012FF58

            No.4: 0012FF5C

            No.5: 0012FF60

             

            No.1: 0012FF50

            No.2: 0012FF54

            No.3: 0012FF58

            No.4: 0012FF5C

            No.5: 0012FF60

            No.1: 0012FF50

            No.2: 0012FF54

            No.3: 0012FF58

            No.4: 0012FF5C

            No.5: 0012FF60

             

            No.1: 0012FF50

            No.2: 0012FF54

            No.3: 0012FF58

            No.4: 0012FF5C

            No.5: 0012FF60

             

            No.1: 0012FF50

            No.2: 0012FF54

            No.3: 0012FF58

            No.4: 0012FF5C

            No.5: 0012FF60

             

             

             

             

            posted on 2006-12-28 20:21 erran 閱讀(1820) 評論(2)  編輯 收藏 引用 所屬分類: C & C++

            Feedback

            # re: 怎樣實現數目不定的函數參數 2006-12-29 13:02 Dain
            bucuo  回復  更多評論
              

            # re: 怎樣實現數目不定的函數參數 2007-06-09 00:03 星夢情緣
            謝謝樓主,我找了好久,我轉回去了,希望不要介意  回復  更多評論
              

            青青草原1769久久免费播放| 国产精品久久亚洲不卡动漫| 狠狠色丁香久久婷婷综合图片| 久久SE精品一区二区| 99国产精品久久| 91精品国产综合久久四虎久久无码一级| 亚洲精品乱码久久久久久不卡| 日产精品久久久久久久性色| 一本一道久久精品综合| 久久国语露脸国产精品电影| 久久精品免费一区二区三区| 久久亚洲精品无码aⅴ大香| 亚洲天堂久久精品| 无码日韩人妻精品久久蜜桃| 精品乱码久久久久久夜夜嗨| 99久久99久久| 亚洲国产精品无码久久久不卡| 久久久久99精品成人片牛牛影视| 久久精品亚洲精品国产色婷| 一级做a爰片久久毛片免费陪| 国产精品久久亚洲不卡动漫| 人妻无码αv中文字幕久久| 亚洲&#228;v永久无码精品天堂久久 | 狠狠色丁香久久婷婷综合图片| 国产AV影片久久久久久| 国产精品一区二区久久不卡| 大香伊人久久精品一区二区| 欧美伊人久久大香线蕉综合69| 香港aa三级久久三级| 久久精品中文騷妇女内射| 精品久久久无码人妻中文字幕| 色综合合久久天天给综看| 国产精品免费看久久久香蕉 | 国产精品美女久久久免费| 精品久久久久久亚洲精品| 欧美丰满熟妇BBB久久久| 中文字幕无码久久人妻| 精品伊人久久久| 久久久无码精品亚洲日韩京东传媒| 中文字幕久久精品| 亚洲精品国精品久久99热|