C語言 淺談可變參數

1.可變參數產生原因

首先來看一個簡單的例子。

int Add(int x, int y)
{return x + y;
}
int main()
{int sum = 0;sum = Add(1, 2);//sum = Add(1, 2, 3);//sum = Add(1);system("pause");return 0;
}

我們可以看到,對于這個代碼只可以計算兩個數的加法。
這樣我們引入一個新的知識點!

可變參數實現求任意數的和

#define _CRT_SECURE_NO_WARNINGS 
#include <stdio.h>
#include <stdarg.h>
#include <stdlib.h>int Add(int num, ...)
{int sum = 0;va_list arg;va_start(arg, num);for (int i = 0; i < num; i++){sum += va_arg(arg, int);}va_end(arg);return sum;
}
int main()
{printf("%d\n", Add(4, 2, 3, 4, 5));system("pause");return 0;
}

詳解

1.main()函數第一句
printf("%d\n", Add(4,     2, 3, 4, 5));

第一個4表示元素個數剩余表示元素內容

2.

操作符的含義:

1.

    va_list arg;
宏操作符實際含義詳解
va_listchar *va_list arg 創建一個字符指針arg,用于訪問參數列表的未確定部分。

2.

    va_start(arg, num);

(ap=(va_list)&v + _INTSIZEOF(v))

宏操作符實際含義詳解
va_start (ap,v)(ap=(va_list)&v + _INTSIZEOF(v))初始化變量arg ,va_start有兩個參數,它的第一個參數是va_list所創建的字符指針。第2個參數是“…”之前的那個參數;(不可以讀省略號)
va_start (arg,num)(arg=(char *)&v + _INTSIZEOF(num))
_INTSIZEOF(sizeof(n) + sizeof(int)-1 & ~(sizeof(int)-1)跳過num,來到位置參數的第一個參數。向上取四的整數倍

3.

    for (int i = 0; i < num; i++){sum += va_arg(arg, int);}
宏操作符實際含義詳解
va_arg(ap,t) (*(t*)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)))(*(t*)((arg += _INTSIZEOF(int)) - _INTSIZEOF(int)))
(*(t*)((arg += _INTSIZEOF(int)) - _INTSIZEOF(int)))(*(int *)((arg += 4) - 4))
(arg += 4) - 4)arg; arg+=4;先給arg加4再減,arg這次產生的結果沒變,但是arg指向了下一個位置。一個代碼做了兩個動作
    for (int i = 0; i < num; i++){sum += va_arg(arg, int);}

取出所有未知參數,然后加給sum

宏操作符實際含義詳解
va_end(ap)(ap = (va_list)0)給指針賦值空指針。不用的時候釋放
va_end(arg);

防止指針亂指用完一定要釋放。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/382889.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/382889.shtml
英文地址,請注明出處:http://en.pswp.cn/news/382889.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

有兩個鏈表a,b,設結點包括學號,姓名。從a鏈表中刪去與b鏈表中有相同學號的那些結點。

#include <stdio.h> #include <string.h> #include <stdlib.h> typedef struct linknode { int num;char name[20];struct linknode *next; }node; node *creat() { node *h NULL,*s,*t;int d;int i 1; char name1[20];while(1) { printf("輸入第…

C語言模擬實現標準庫函數之strlen()

strlen() strlen所作的僅僅是一個計數器的工作&#xff0c;它從內存的某個位置 &#xff08;可以是字符串開頭&#xff0c;中間某個位置&#xff0c;甚至是某個不確定的內存區域&#xff09; 開始掃描&#xff0c;直到碰到第一個字符串結束符\0為止&#xff0c;然后返回計數…

C語言模擬實現標準庫函數之strcpy()

strcpy(dest,src) strcpy是一種C語言的標準庫函數&#xff0c;strcpy把從src地址開始且含有\0結束符的字符串復制到以dest開始的地址空間&#xff0c;返回值的類型為char*。 char * my_strcpy(char *str2, char *str1) {assert(*str2);assert(*str1);while(*str1!0){ *str2 …

C語言模擬實現標準庫函數之strcat()

strcat() strcat用于將兩個char類型鏈接的函數。 char * my_strcat(char *str1, char *str2) {assert(str2);assert(str1);char * p str1;while (*str1 ! 0){str1;}while (*str1 *str2){;}return p; } int main() {char str1[10] "abc";char str2[10] "de…

模板

模板是泛型編程的基礎&#xff0c;即與類型無關的邏輯代碼。 利用模板機制可以顯著減少冗余信息&#xff0c;能大幅度地節約程序代碼&#xff0c;進一步提高面向對象程序的可重用性和可維護性。 模板是實現代碼重用機制的一種工具&#xff0c;它可以實現類型參數化&#xff1b;…

C語言模擬實現標準庫函數之strstr()

strstr() strstr(str1,str2) 函數用于判斷字符串str2是否是str1的子串。如果是&#xff0c;則該函數返回str2在str1中首次出現的地址&#xff1b;否則&#xff0c;返回NULL。 char* my_strstr(const char* dest, const char* src) {char * str1 dest;char * str2 src;char …

linux-----強大的find

我又回來了。哈哈。今天我們來說一下linux中的另一個強大的find命令&#xff0c;灰常重要&#xff0c;灰常重要&#xff0c;灰常重要。顯而易見&#xff0c;find就是對某一個文件或者目錄的查找嘍。但是它的一個顯著的特點就是&#xff1a;一般放在后臺執行&#xff0c;從整個文…

C語言模擬實現標準庫函數之strchr()

strchr() 查找字符串s中首次出現字符c的位置 char * my_strchr(char *str1, char str2) {while (*str1 ! str2 && *str1 ! NULL){str1;}return str1; }int main() {char string[17];char *ptr, c r;strcpy(string, "Thisisastring");ptr my_strchr(string…

python 多人連接mysql 進行事務操作 對mysql加鎖與釋放鎖

python 多人連接mysql 對mysql進行事務操作 對mysql加鎖與釋放鎖 下面這個是user1代碼塊 # -*- coding: utf-8 -*- # user1 import pymysql import timeconn pymysql.connections.Connection(host"localhost", userdebian-sys-maint, passwordwL5wsDKDub4gT2EU…

C語言模擬實現標準庫函數之strcmp()

strcmp() C/C函數&#xff0c;比較兩個字符串 設這兩個字符串為str1&#xff0c;str2&#xff0c; 若str1str2&#xff0c;則返回零&#xff1b; 若str1<str2&#xff0c;則返回負數&#xff1b; 若str1>str2&#xff0c;則返回正數。 char * my_strcmp(char *key,…

linux之task_struct

每個進程中都有一個進程控制塊--PCB。PCB--維護進程相關的信息。然而&#xff0c;linux內核的進程控制塊就是task_struct結構體&#xff0c;它可以保存進程的信息。 所有運行在系統里的進程都以task_struct鏈表的形式存在內核里。 每個進程都將它的信息放在task_struct結構體…

python 用元類 type 實現對數據庫的ORM 映射

python 實現對數據庫的ORM 映射 如果使用pymysql 操作數據庫 不借助框架的話&#xff0c;頻繁寫sql語句, 的確比較麻煩 這里借助 type 元類 對 數據表類實現了 與mysql之間的 映射 直接上代碼 import pymysqldef conn_database_execute(sql_str):conn pymysql.connect(host…

C語言模擬實現標準庫函數之memcpy()

memcpy&#xff08;&#xff09; 1.如果我們需要對一個數組初始化&#xff0c;把數組的內容全部置0&#xff0c;那么能不能用strcpy() int main() {char arr1[10] { 0 };char arr2[10] " abcdefg ";strcpy(arr2, arr1);system("pause");return 0; } 我…

說說堆及堆排序

堆&#xff1a;是一種數組對象&#xff0c;它可以被看成是一種二叉樹結構。 我們把堆的二叉樹存儲方式分為兩種&#xff1a;即大堆和小堆。那么問題來了&#xff0c;什么大堆&#xff1f;什么是小堆&#xff1f; 大堆&#xff1a;讓每個父節點的值都大于孩子節點的值。 小堆…

運算符優先級 速查表

運算符優先級 優先級【高到低】&#xff1a; 第一級&#xff1a; 圓括號【&#xff08;&#xff09;】、下標運算符【[]】、分量運算符的指向結構體成員運算符【->】、結構體成員運算符【.】 第二級&#xff1a; 邏輯非運算符【!】、按位取反運算符【~】、自增自減運…

linux--幾種常見的進程調度算法

進程調度&#xff1a;在操作系統中調度是指一種資源分配&#xff0c;因而調度算法是指:根據系統的資源分配策略所規定的資源分配算法。操作系統管理了系統的有限資源&#xff0c;當有多個進程(或多個進程發出的請求)要使用這些資源時&#xff0c;因為資源的有限性&#xff0c;必…

指針數組和數組指針和函數指針

文章目錄1.指針數組和數組指針1.int *p1[10];2.int (*p2)[10];2.函數指針char *(*fun1)(char * p1,char *p2)函數指針的概念函數指針的作用&#xff1a;例子1 .調用方式例子2&#xff1a;&#xff08;帶注釋&#xff09;例子33.做題的小技巧1.指針數組和數組指針 1.int *p1[10…

使用虛擬環境virtualenv 創建虛擬環境出現PermissionError: [Errno 13] Permission denied:

使用虛擬環境virtualenv 創建虛擬環境出現PermissionError: [Errno 13] Permission denied: 原因&#xff1a;虛擬環境安裝的目錄所屬用戶非當前用戶 解決辦法&#xff1a;將目錄及其文件的所有者改為當前用戶 解決命令&#xff1a;sudo chown -R 當前用戶 待更改用戶的目錄/ …

linux之父子進程的輸出

首先&#xff0c;我們來回憶一下父進程與子進程&#xff0c;前幾節講了如何創建子進程&#xff0c;像這樣的&#xff0c;pid_t id fork(); 這樣我們就創建好了一個子進程&#xff0c;然而fork()函數的返回值是什么呢&#xff1f;這里要記住&#xff1a;子進程返回0&#xff0c…

linux---談談vfork和fork的區別及exit與return

fork()&#xff1a;創建子進程的函數&#xff0c;是大家比較熟悉的吧。pid_t id fork(); 這里的vfork();也是創建子進程的函數。現在我們來剖析一下它們吧。 第一例&#xff1a; 先看一個fork()的例子哦。 對于fork()而言&#xff0c;創建子進程成功后直接打印出父子進程執…