<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 牛人業(yè)話 > C語言的那些小秘密之變參函數(shù)的實現(xiàn)

          C語言的那些小秘密之變參函數(shù)的實現(xiàn)

          作者: 時間:2015-03-06 來源:網(wǎng)絡 收藏

            在學習C語言的過程中我們可能很少會去寫變參,印象中大學老師好像也沒有提及過,但我發(fā)現(xiàn)變參的實現(xiàn)很巧妙,所以還是特地在此分析下變參的實現(xiàn)原理。無需標準C的支持,我們自己寫代碼來實現(xiàn)。

          本文引用地址:http://www.ex-cimer.com/article/270588.htm

            先來看看一個實現(xiàn)代碼:

            #include

            #define va_list void*

            #define va_arg(arg, type) *(type*)arg; arg = (char*)arg + sizeof(type);

            #define va_start(arg, start) arg = (va_list)(((char*)&(start)) + sizeof(start))

            int sum(int nr, ...)

            {

            int i = 0;

            int result = 0;

            va_list arg = NULL;

            va_start(arg, nr);

            for(i = 0; i < nr; i++)

            {

            result += va_arg(arg, int);

            }

            return result;

            }

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

            {

            printf("%dn", sum(4, 100,100,100,100));

            printf("%dn", sum(3, 200, 200, 200));

            return 0;

            }

            運行結果如下:



            #define va_list void*通過這句代碼我們實現(xiàn)了定義va_list是一個指針,參數(shù)類型不定,它可以指向任意類型的指針。為了讓arg指向第一個可變參數(shù),我們用nr的地址加上nr的數(shù)據(jù)類型大小就行了,采用如下的定義可以實現(xiàn)。

            #define va_start(arg, start) arg = (va_list)(((char*)&(start)) + sizeof(start)) 。

            通過(((char*)&(start)) + sizeof(start)) 可以得到第一個可變參數(shù)的地址,再將其強制轉換為va_list類型。

            成功取出了第一個可變參數(shù)后,接下來的任務就是繼續(xù)取出可變參數(shù),方法跟上面求第一個可變參數(shù)的方法一樣,通過arg = (char*)arg + sizeof(type);來實現(xiàn)讓arg指向下一個可變參數(shù),type為可變參數(shù)的類型,通過這種方法可以一一取出可變參數(shù)。

            在這里順便給出上面實現(xiàn)代碼的匯編代碼,有興趣的可以讀讀,加深下對于底層匯編代碼的閱讀能力。

            .file "varargs.c"

            .text

            .globl sum

            .type sum, @function

            sum:

            pushl %ebp

            movl %esp, %ebp

            subl $16, %esp

            movl $0, -4(%ebp)

            movl $0, -8(%ebp)

            movl $0, -12(%ebp)

            leal 12(%ebp), %eax

            movl %eax, -12(%ebp)

            movl $0, -4(%ebp)

            jmp .L2

            .L3:

            movl -12(%ebp), %eax

            movl (%eax), %eax

            addl %eax, -8(%ebp)

            addl $4, -12(%ebp)

            addl $1, -4(%ebp)

            .L2:

            movl 8(%ebp), %eax

            cmpl %eax, -4(%ebp)

            jl .L3

            movl -8(%ebp), %eax

            leave

            ret

            .size sum, .-sum

            .section .rodata

            .LC0:

            .string "%dn"

            .text

            .globl main

            .type main, @function

            main:

            pushl %ebp

            movl %esp, %ebp

            andl $-16, %esp

            subl $32, %esp

            movl $100, 16(%esp)

            movl $100, 12(%esp)

            movl $100, 8(%esp)

            movl $100, 4(%esp)

            movl $4, (%esp)

            call sum

            movl $.LC0, %edx

            movl %eax, 4(%esp)

            movl %edx, (%esp)

            call printf

            movl $200, 12(%esp)

            movl $200, 8(%esp)

            movl $200, 4(%esp)

            movl $3, (%esp)

            call sum

            movl $.LC0, %edx

            movl %eax, 4(%esp)

            movl %edx, (%esp)

            call printf

            movl $0, %eax

            leave

            ret

            .size main, .-main

            .ident "GCC: (Ubuntu/Linaro 4.5.2-8ubuntu4) 4.5.2"

            .section .note.GNU-stack,"",@progbits

          樹莓派文章專題:樹莓派是什么?你不知道樹莓派的知識和應用

          c語言相關文章:c語言教程



          上一頁 1 2 下一頁

          關鍵詞: C語言 函數(shù)

          評論


          相關推薦

          技術專區(qū)

          關閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();