當前位置:首頁 » 編程語言 » c語言datype的用法
擴展閱讀
webinf下怎麼引入js 2023-08-31 21:54:13
堡壘機怎麼打開web 2023-08-31 21:54:11

c語言datype的用法

發布時間: 2023-06-30 03:15:49

c語言...用法

C語言變參技術
概述

C語言中有一種長度不確定的參數,形如:"…",它主要用在參數個數不確定的函數中,我們最容易想到的例子是printf函數。

原型:

int printf( const char *format [, argument]... );

使用例:

printf("Enjoy yourself everyday!\\n");

printf("The value is %d!\\n", value);

這種可變參數可以說是C語言一個比較難理解的部分,這里會由幾個問題引發一些對它的分析。

注意:在C++中有函數重載(overload)可以用來區別不同函數參數的調用,但它還是不能表示任意數量的函數參數。

問題:printf的實現

請問,如何自己實現printf函數,如何處理其中的可變參數問題? 答案與分析:

在標准C語言中定義了一個頭文件<stdarg.h>專門用來對付可變參數列表,它包含了一組宏,和一個va_list的typedef聲明。一個典型實現如下:

typedef char* va_list;

#define va_start(list) list = (char*)&va_alist

#define va_end(list)

#define va_arg(list, mode)\\

((mode*) (list += sizeof(mode)))[-1]

自己實現printf:

#include <stdarg.h>

int printf(char* format, …)

{

va_list ap;

va_start(ap, format);

int n = vprintf(format, ap);

va_end(ap);

return n;

}

問題:運行時才確定的參數

有沒有辦法寫一個函數,這個函數參數的具體形式可以在運行時才確定?

答案與分析:

目前沒有"正規"的解決辦法,不過獨門偏方倒是有一個,因為有一個函數已經給我們做出了這方面的榜樣,那就是main(),它的原型是:

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

函數的參數是argc和argv。

深入想一下,"只能在運行時確定參數形式",也就是說你沒辦法從聲明中看到所接受的參數,也即是參數根本就沒有固定的形式。常用的辦法是你可以通過定
義一個void
*類型的參數,用它來指向實際的參數區,然後在函數中根據根據需要任意解釋它們的含義。這就是main函數中argv的含義,而argc,則用來表明實際
的參數個數,這為我們使用提供了進一步的方便,當然,這個參數不是必需的。

雖然參數沒有固定形式,但我們必然要在函數中解析參數的意義,因此,理所當然會有一個要求,就是調用者和被調者之間要對參數區內容的格式,大小,有效性等所有方面達成一致,否則南轅北轍各說各話就慘了。

問題:可變長參數的傳遞

有時候,需要編寫一個函數,將它的可變長參數直接傳遞給另外的函數,請問,這個要求能否實現?

答案與分析:

目前,你尚無辦法直接做到這一點,但是我們可以迂迴前進,首先,我們定義被調用函數的參數為va_list類型,同時在調用函數中將可變長參數列表轉換為va_list,這樣就可以進行變長參數的傳遞了。看如下所示:

void subfunc (char *fmt, va_list argp)

{

...

arg = va_arg (fmt, argp); /* 從argp中逐一取出所要的參數 */

...

}

void mainfunc (char *fmt, ...)

{

va_list argp;

va_start (argp, fmt); /* 將可變長參數轉換為va_list */

subfunc (fmt, argp); /* 將va_list傳遞給子函數 */

va_end (argp);

...

}

問題:可變長參數中類型為函數指針

我想使用va_arg來提取出可變長參數中類型為函數指針的參數,結果卻總是不正確,為什麼?

答案與分析:

這個與va_arg的實現有關。一個簡單的、演示版的va_arg實現如下:

#define va_arg(argp, type) \\

(*(type *)(((argp) += sizeof(type)) - sizeof(type)))

其中,argp的類型是char *。

如果你想用va_arg從可變參數列表中提取出函數指針類型的參數,例如

int (*)(),則va_arg(argp, int (*)())被擴展為:

(*(int (*)() *)(((argp) += sizeof (int (*)())) -sizeof (int (*)())))

顯然,(int (*)() *)是無意義的。

解決這個問題的辦法是將函數指針用typedef定義成一個獨立的數據類型,例如:

typedef int (*funcptr)();

這時候再調用va_arg(argp, funcptr)將被擴展為:

(* (funcptr *)(((argp) += sizeof (funcptr)) - sizeof (funcptr)))

這樣就可以通過編譯檢查了。

問題:可變長參數的獲取

有這樣一個具有可變長參數的函數,其中有下列代碼用來獲取類型為float的實參:

va_arg (argp, float);

這樣做可以嗎?

答案與分析:

不可以。在可變長參數中,應用的是"加寬"原則。也就是float類型被擴展成double;char,
short被擴展成int。因此,如果你要去可變長參數列表中原來為float類型的參數,需要用va_arg(argp,
double)。對char和short類型的則用va_arg(argp, int)。

問題:定義可變長參數的一個限制

為什麼我的編譯器不允許我定義如下的函數,也就是可變長參數,但是沒有任何的固定參數?

int f (...)

{

...

}

答案與分析:

不可以。這是ANSI C 所要求的,你至少得定義一個固定參數。

這個參數將被傳遞給va_start(),然後用va_arg()和va_end()來確定所有實際調用時可變長參數的類型和值。

② C語言中typedef struct什麼意思

類型定義。

它就和define相對應,define是把一個自己起的名字的常量定義為代替它的另一個常量來同。

typedef是把一個自己起的名字的類型用已經有的類型代替使用,如:typedef int Type1;之後如果有Type1 i ;系統則理解為int i;。

在編程中使用typedef目的一般有兩個,一個是給變數一個易記且意義明確的新名字,另一個是簡化一些比較復雜的類型聲明。

輸出為:token 9 = 10

詳見網路typedef(相似)。

③ C語言中的datatype是什麼

datatype是數據類型。C的數據類型包括:整型、字元型、實型或浮點型(單精度和雙精度)、枚舉類型、數組類型、結構體類型、共用體類型、指針類型和空類型。

數據類型關鍵字:

1、short:修飾int,短整型數據,可省略被修飾的int。(K&R時期引入)

2、long:修飾int,長整型數據,可省略被修飾的int。(K&R時期引入)

3、long long:修飾int,超長整型數據,可省略被修飾的int。(C99標准新增)

4、signed:修飾整型數據,有符號數據類型。(C89標准新增)

5、unsigned:修飾整型數據,無符號數據類型。(K&R時期引入)

6、restrict:用於限定和約束指針,並表明指針是訪問一個數據對象的唯一且初始的方式。(C99標准新增)

(3)c語言datype的用法擴展閱讀

轉換:

在算術運算和關系運算中如果參與運算的操作數類型不一樣,則系統會對其進行類型轉換,這是隱含轉換,轉換的原則就是將低類型的數據轉換為高類型數據。

各類型從低到高依次為char,short,int,unsigned int,long,unsigned long,float,double。類型越高范圍越大,精度也越高。隱含轉換是安全的,因為沒有精度損失。

邏輯運算符的操作數必須是bool型,如果不是就需要將其轉換為bool型,非0數據轉換為true,0轉換為false。位運算操作數必須是整數,如果不是也會自動進行類型轉換,也是低類型數據轉換為高類型數據。

賦值運算要求賦值運算符左邊的值和右邊的值類型相同,不同的話也要進行自動轉換,但這個時候不會遵從上面的原則而是一律將右值轉換為左值的類型。

比如,int iVal; float fVal; double dVal;則dVal=iVal*fVal;計算時先將iVal轉換為跟fVal一樣的float型,乘法的結果再轉換為double型。

④ C語言中的typedef是什麼意思啊

typedef為C語言的關鍵字,作用是為一種數據類型定義一個新名字。這里的數據類型包括內部數據類型(int,char等)和自定義的數據類型(struct等)。

在編程中使用typedef目的一般有兩個,一個是給變數一個易記且意義明確的新名字,另一個是簡化一些比較復雜的類型聲明。

(4)c語言datype的用法擴展閱讀:

C語言允許用戶使用 typedef 關鍵字來定義自己習慣的數據類型名稱,來替代系統默認的基本類型名稱、數組類型名稱、指針類型名稱與用戶自定義的結構型名稱、共用型名稱、枚舉型名稱等。

一旦用戶在程序中定義了自己的數據類型名稱,就可以在該程序中用自己的數據類型名稱來定義變數的類型、數組的類型、指針變數的類型與函數的類型等。

例如,C 語言在 C99 之前並未提供布爾類型,但可以使用 typedef 關鍵字來定義一個簡單的布爾類型。

參考資料來源:網路—typedef

⑤ typedef用法

typedef用法:

1、用typedef為現有類型創建別名,定義易於記憶的類型名

(5)c語言datype的用法擴展閱讀

typedef 有另外一個重要的用途,那就是定義機器無關的類型,例如,你可以定義一個叫 REAL 的浮點類型,在目標機器上它可以獲得最高的精度:

typedef long double REAL;

在不支持 long double 的機器上,該 typedef 看起來會是下面這樣:

typedef double REAL;

並且,在連 double 都不支持的機器上,該 typedef 看起來會是這樣:

typedef float REAL;

你不用對源代碼做任何修改,便可以在每一種平台上編譯這個使用 REAL 類型的應用程序。唯一要改的是 typedef 本身。

在大多數情況下,甚至這個微小的變動完全都可以通過奇妙的條件編譯來自動實現。不是嗎?

標准庫廣泛地使用 typedef 來創建這樣的平台無關類型:size_t,ptrdiff 和 fpos_t 就是其中的例子。

此外,象 std::string 和 std::ofstream 這樣的 typedef 還隱藏了長長的,難以理解的模板特化語法,例如:basic_string,allocator> 和 basic_ofstream>。