亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C語言如何實現(xiàn)可變參數(shù)詳解

 更新時間:2021年07月26日 11:41:28   作者:東垂小夫  
這種可變參數(shù)可以說是C語言一個比較難理解的部分,下面這篇文章主要給大家介紹了關(guān)于C語言如何實現(xiàn)可變參數(shù)的相關(guān)資料,需要的朋友可以參考下

可變參數(shù)

可變參數(shù)是指函數(shù)的參數(shù)的數(shù)據(jù)類型和數(shù)量都是不固定的。

printf函數(shù)的參數(shù)就是可變的。這個函數(shù)的原型是:int printf(const char *format, ...)。

用一段代碼演示printf的用法。

// code-A
#include <stdio.h>
int main(int argc, char **argv)
{
   printf("a is %d, str is %s, c is %c\n", 23, "Hello, World;", 'A');
   printf("T is %d\n", 78);
   return 0;
}

在code-A中,第一條printf語句有4個參數(shù),第二條printf語句有2個參數(shù)。顯然,printf的參數(shù)是可變的。

實現(xiàn)

代碼

code-A

先看兩段代碼,分別是code-A和code-B。

// file stack-demo.c

#include <stdio.h>

// int f(char *fmt, int a, char *str);
int f(char *fmt, ...);
int f2(char *fmt, void *next_arg);
int main(int argc, char *argv)
{
        char fmt[20] = "hello, world!";
        int a = 10;
        char str[10] = "hi";
        f(fmt, a, str);
        return 0;
}

// int f(char *fmt, int a, char *str)
int f(char *fmt, ...)
{
        char c = *fmt;
        void *next_arg = (void *)((char *)&fmt + 4);
        f2(fmt, next_arg);
        return 0;
}


int f2(char *fmt, void *next_arg)
{
        printf(fmt);
        printf("a is %d\n", *((int *)next_arg));
        printf("str is %s\n", *((char **)(next_arg + 4)));

        return 0;
}

編譯執(zhí)行,結(jié)果如下:

# 編譯
[root@localhost c]# gcc -o stack-demo stack-demo.c -g -m32
# 反匯編并把匯編代碼寫入dis-stack.asm中
[root@localhost c]# objdump -d stack-demo>dis-stack.asm
[root@localhost c]# ./stack-demo
hello, world!a is 10
str is hi

code-B

// file stack-demo.c

#include <stdio.h>

// int f(char *fmt, int a, char *str);
int f(char *fmt, ...);
int f2(char *fmt, void *next_arg);
int main(int argc, char *argv)
{
        char fmt[20] = "hello, world!";
        int a = 10;
        char str[10] = "hi";
     char str2[10] = "hello";
        f(fmt, a, str, str2);
        return 0;
}

// int f(char *fmt, int a, char *str)
int f(char *fmt, ...)
{
        char c = *fmt;
        void *next_arg = (void *)((char *)&fmt + 4);
        f2(fmt, next_arg);
        return 0;
}


int f2(char *fmt, void *next_arg)
{
        printf(fmt);
        printf("a is %d\n", *((int *)next_arg));
        printf("str is %s\n", *((char **)(next_arg + 4)));
     printf("str2 is %s\n", *((char **)(next_arg + 8)));

        return 0;
}

編譯執(zhí)行,結(jié)果如下:

# 編譯
[root@localhost c]# gcc -o stack-demo stack-demo.c -g -m32
# 反匯編并把匯編代碼寫入dis-stack.asm中
[root@localhost c]# objdump -d stack-demo>dis-stack.asm
[root@localhost c]# ./stack-demo
hello, world!a is 10
str is hi
str2 is hello

分析

在code-A中,調(diào)用f的語句是f(fmt, a, str);;在code-B中,調(diào)用f的語句是f(fmt, a, str, str2);。

很容易看出,int f(char *fmt, ...);就是參數(shù)可變的函數(shù)。

關(guān)鍵語句

實現(xiàn)可變參數(shù)的關(guān)鍵語句是:

char c = *fmt;
void *next_arg = (void *)((char *)&fmt + 4);
printf("a is %d\n", *((int *)next_arg));
printf("str is %s\n", *((char **)(next_arg + 4)));
printf("str2 is %s\n", *((char **)(next_arg + 8)));
  • &fmt是第一個參數(shù)的內(nèi)存地址。
  • next_arg是第二個參數(shù)的內(nèi)存地址。
  • next_arg+4、next_arg+8分別是第三個、第四個參數(shù)的內(nèi)存地址。

為什么

內(nèi)存地址的計算方法

先看一段偽代碼。這段偽代碼是f函數(shù)的對應(yīng)的匯編代碼。假設(shè)f有三個參數(shù)。當(dāng)然f也可以有四個參數(shù)或2個參數(shù)。我們用三個參數(shù)的情況來觀察一下f。

f:

 ; 入棧ebp

 ; 把ebp設(shè)置為esp

 

 ; ebp + 0 存儲的是 eip,由call f入棧

 ; ebp + 4 存儲的是 舊ebp

 ; 第一個參數(shù)是 ebp + 8

 ; 第二個參數(shù)是 ebp + 12

 ; 第三個參數(shù)是 ebp + 16

 

 ; 函數(shù)f的邏輯

 

 ; 出棧ebp。ebp恢復(fù)成了剛進入函數(shù)之前的舊ebp

 ; ret

調(diào)用f的偽代碼是:

; 入棧第三個參數(shù)

; 入棧第二個參數(shù)

; 入棧第一個參數(shù)

; 調(diào)用f,把eip入棧

在匯編代碼中,第一個參數(shù)的內(nèi)存地址很容易確定,第二個、第三個還有第N個參數(shù)的內(nèi)存地址也非常容易確定。無法是在ebp的基礎(chǔ)上增加特定長度而已。

可是,我們只能確定,必定存在第一個參數(shù),不能確定是否存在的二個、第三個還有第N個參數(shù)。沒有理由使用一個可能不存在的參數(shù)作為參照物、并且還要用它卻計算其他參數(shù)的地址。

第一個參數(shù)必定存在,所以,我們用它作為確定其他參數(shù)的內(nèi)存地址的參照物。

內(nèi)存地址

在f函數(shù)的C代碼中,&fmt是第一個參數(shù)占用的f的棧的元素的內(nèi)存地址,換句話說,是一個局部變量的內(nèi)存地址。

局部變量的內(nèi)存地址不能作為函數(shù)的返回值,卻能夠在本函數(shù)執(zhí)行結(jié)束前使用,包括在本函數(shù)調(diào)用的其他函數(shù)中使用。這就是在f2中仍然能夠使用fmt計算出來的內(nèi)存地址的原因。

難點

當(dāng)參數(shù)是int類型時,獲取參數(shù)的值使用*(int *)(next_arg)。

當(dāng)參數(shù)是char str[20]時,獲取參數(shù)的值使用*(char **)(next_arg + 4)。

為什么不直接使用next_arg、(next_arg + 4)呢?

分析*(int *)(next_arg)。

在32位操作系統(tǒng)中,任何內(nèi)存地址的值看起來都是一個32位的正整數(shù)??墒沁@個正整數(shù)的值的類型并不是unsigned int,而是int *。

關(guān)于這點,我們可以在gdb中使用ptype確認(rèn)一下。例如,有一小段代碼int *a;*a = 5;,執(zhí)行ptype a,結(jié)果會是int *。

next_arg只是一個正整數(shù),損失了它的數(shù)據(jù)類型,我們需要把數(shù)據(jù)類型補充進來。我們能夠把這個操作理解成”強制類型轉(zhuǎn)換“。

至于*(int *)(next_arg)前面的*,很容易理解,獲取一個指針指向的內(nèi)存中的值。

用通用的方式分析*(char **)(next_arg+4)。

  1. 因為是第三個參數(shù),因此next_arg+4。
  2. 因為第三個參數(shù)的數(shù)據(jù)類型是char str[20]。根據(jù)經(jīng)驗,char str[20]對應(yīng)的指針是char *。
  3. 因為next_arg+4只是函數(shù)的棧的元素的內(nèi)存地址,在目標(biāo)元素中存儲的是一個指針。也就是說,next_arg+4是一個雙指針類型的指針。它最終又指向字符串,根據(jù)經(jīng)驗,next_arg+4的數(shù)據(jù)類型是char **。沒必要太糾結(jié)這一點。自己寫一個簡單的指向字符串的雙指針,使用gdb的ptype查看這種類型的數(shù)據(jù)類型就能驗證這一點。
  4. 最前面的*,獲取指針指向的數(shù)據(jù)。

給出一段驗證第3點的代碼。

char str[20] = "hello";
char *ptr = str;
// 使用gdb的ptype 打印 ptype &ptr

打印結(jié)果如下:

Breakpoint 1, main (argc=1, argv=0xffffd3f4) at point.c:13
13  char str7[20] = "hello";
(gdb) s
14  char *ptr = str7;
(gdb) s
19  int b = 7;
(gdb) p &str
$1 = (char **) 0xffffd2fc

優(yōu)化

在code-A和code-B中,我們?nèi)斯じ鶕?jù)參數(shù)的類型來獲取參數(shù),使用*(int *)(next_arg)或*(char **)(next_arg + 4)。

庫函數(shù)printf顯然不是人工識別參數(shù)的類型。

這個函數(shù)的第一個參數(shù)中包含%d、%x、%s等占位符。遍歷第一個參數(shù),識別出%d,就用*(int *)next_arg替換%d。識別出

%s,就用*(char **)next_arg。

實現(xiàn)了識別占位符并且根據(jù)占位符選擇指針類型的功能,就能實現(xiàn)一個完成度很高的可變參數(shù)了。

總結(jié)

到此這篇關(guān)于C語言如何實現(xiàn)可變參數(shù)的文章就介紹到這了,更多相關(guān)C語言可變參數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Qt自定義實現(xiàn)一個等待提示Ui控件

    Qt自定義實現(xiàn)一個等待提示Ui控件

    等待樣式控件是我們在做UI時出場率還挺高的控件之一,所以這篇文章主要為大家介紹了Qt如何自定義一個好看的等待提示Ui控件,感興趣的可以了解下
    2024-01-01
  • 使用pthread庫實現(xiàn)openssl多線程ssl服務(wù)端和客戶端

    使用pthread庫實現(xiàn)openssl多線程ssl服務(wù)端和客戶端

    使用pthread庫實現(xiàn)openssl多線程ssl服務(wù)端和客戶端,大家參考使用吧
    2014-01-01
  • 關(guān)于VS2022不能使用<bits/stdc++.h>的解決方案(萬能頭文件)

    關(guān)于VS2022不能使用<bits/stdc++.h>的解決方案(萬能頭文件)

    #include<bits/stdc++.h>包含了目前 C++ 所包含的所有頭文件,又稱萬能頭文件,那么如何在VS2022中使用萬能頭呢?下面小編給大家代理了關(guān)于VS2022不能使用<bits/stdc++.h>的解決方案(萬能頭文件),感興趣的朋友一起看看吧
    2022-03-03
  • OpenCV實現(xiàn)高斯噪聲

    OpenCV實現(xiàn)高斯噪聲

    這篇文章主要為大家詳細(xì)介紹了OpenCV實現(xiàn)高斯噪聲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • C語言手寫多級時間輪定時器

    C語言手寫多級時間輪定時器

    這篇文章主要為大家詳細(xì)介紹了如何利用C語言實現(xiàn)手寫多級時間輪定時器,文中的示例代碼講解詳細(xì),具有一定的借鑒價值,需要的可以參考一下
    2022-09-09
  • C++中用substr()函數(shù)消除前后空格的解決方法詳解

    C++中用substr()函數(shù)消除前后空格的解決方法詳解

    本篇文章是對C++中用substr()函數(shù)消除前后空格的方法進行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • Define,const,static用法總結(jié)

    Define,const,static用法總結(jié)

    const定義的全局?jǐn)?shù)據(jù)變量,其基本作用和define相同,但又在define的基礎(chǔ)上增加了好多功能
    2013-10-10
  • C++中的string類型

    C++中的string類型

    這篇文章主要介紹了C++中的string類型,在C++當(dāng)中,除了char 類型,還有專門的字符串類型,就叫做string,下面文字將圍繞其相關(guān)資料展開詳細(xì)內(nèi)容,需要的朋友可以參考一下,希望對你有所幫助
    2021-11-11
  • C語言實現(xiàn)輸入ascii碼,輸出對應(yīng)的字符方式

    C語言實現(xiàn)輸入ascii碼,輸出對應(yīng)的字符方式

    這篇文章主要介紹了C語言實現(xiàn)輸入ascii碼,輸出對應(yīng)的字符方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • 淺談C++中字符串輸入get與getline的區(qū)別

    淺談C++中字符串輸入get與getline的區(qū)別

    這篇文章主要介紹了C++中字符串輸入get與getline的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03

最新評論