日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 人文社科 > 生活经验 >内容正文

生活经验

Typedef用法(转载)

發(fā)布時間:2023/11/27 生活经验 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Typedef用法(转载) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

在C的學(xué)習(xí)過程中,現(xiàn)在才發(fā)現(xiàn),以前有那么多被忽略的重點;現(xiàn)在是慢慢拾起這些重點的時候,通過百度和博客,我感覺我學(xué)到了很多東西,自己只是在別人說的基礎(chǔ)上,按照自己學(xué)習(xí)的過程在這里記錄一下,以后有時間回過頭反復(fù)看,才不會忘記。加油!

1、typedef 聲明,簡稱 typedef,為現(xiàn)有類型創(chuàng)建一個新的名字,或稱為類型別名,在結(jié)構(gòu)體定義,還有一些數(shù)組等地方都大量的用到。

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

?

2、typedef與結(jié)構(gòu)結(jié)合使用   

typedef struct tagMyStruct   

{   

int iNum;   

long lLength;   

} MyStruct;   

這語句實際上完成兩個操作:   

1)?????? 定義一個新的結(jié)構(gòu)類型   

struct tagMyStruct   

{   

int iNum;   

long lLength;   

};   

分析:tagMyStruct稱為“tag”,即“標(biāo)簽”,實際上是一個臨時名字,struct 關(guān)鍵字和tagMyStruct一起,構(gòu)成了這個結(jié)構(gòu)類型,不論是否有typedef,這個結(jié)構(gòu)都存在。   

我們可以用struct tagMyStruct varName來定義變量,但要注意,使用tagMyStruct varName來定義變量是不對的,因為struct 和tagMyStruct合在一起才能表示一個結(jié)構(gòu)類型。   

2)?????? typedef為這個新的結(jié)構(gòu)起了一個名字,叫MyStruct。   

typedef struct tagMyStruct MyStruct;   

因此,MyStruct實際上相當(dāng)于struct tagMyStruct,我們可以使用MyStruct varName來定義變量。

?

3、typedef & #define的問題

(1)#define:簡單的文本替換

(2)typedef char * pStr;   

char string[4] = "abc";   

const char *p1 = string;   

const pStr p2 = string;   

p1++;   

p2++;

是p2++出錯了。這個問題再一次提醒我們:typedef和#define不同,它不是簡單的文本替換。上述代碼中const pStr p2并不等于const char * p2。const pStr p2和const long x本質(zhì)上沒有區(qū)別,都是對變量進行只讀限制,只不過此處變量p2的數(shù)據(jù)類型是我們自己定義的而不是系統(tǒng)固有類型而已。因此,const pStr p2的含義是:限定數(shù)據(jù)類型為char *的變量p2為只讀,因此p2++錯誤。

(3)#define與typedef引申談

1) #define宏定義有一個特別的長處:可以使用 #ifdef ,#ifndef等來進行邏輯判斷,還可以使用#undef來取消定義。   

2) typedef也有一個特別的長處:它符合范圍規(guī)則,使用typedef定義的變量類型其作用范圍限制在所定義的函數(shù)或者文件內(nèi)(取決于此變量定義的位置),而宏定義則沒有這種特性。

?

4、別人總結(jié)的用途:

用途一:

定義一種類型的別名,而不只是簡單的宏替換。可以用作同時聲明指針型的多個對象。比如:

char* pa, pb;?// 這多數(shù)不符合我們的意圖,它只聲明了一個指向字符變量的指針,

// 和一個字符變量;

以下則可行:

typedef char* PCHAR;

PCHAR pa, pb;??

這種用法很有用,特別是char* pa, pb的定義,初學(xué)者往往認為是定義了兩個字符型指針,其實不是,而用typedef char* PCHAR就不會出現(xiàn)這樣的問題,減少了錯誤的發(fā)生。

用途二:
用在舊的C代碼中,幫助struct。以前的代碼中,聲明struct新對象時,必須要帶上struct,即形式為: struct 結(jié)構(gòu)名對象名,如:

struct tagPOINT1

?{
??? int x;

??? int y;
};

struct tagPOINT1 p1;

而在C++中,則可以直接寫:結(jié)構(gòu)名對象名,即:tagPOINT1 p1;

typedef struct tagPOINT
{
??? int x;

??? int y;
}POINT;

POINT p1; // 這樣就比原來的方式少寫了一個struct,比較省事,尤其在大量使用的時

候,或許,在C++中,typedef的這種用途二不是很大,但是理解了它,對掌握以前的舊代

碼還是有幫助的,畢竟我們在項目中有可能會遇到較早些年代遺留下來的代碼。

用途三

用typedef來定義與平臺無關(guān)的類型。

比如定義一個叫 REAL 的浮點類型,在目標(biāo)平臺一上,讓它表示最高精度的類型為:

typedef long double REAL;

在不支持 long double 的平臺二上,改為:

typedef double REAL;

在連 double 都不支持的平臺三上,改為:

typedef float REAL;

也就是說,當(dāng)跨平臺時,只要改下 typedef 本身就行,不用對其他源碼做任何修改。

標(biāo)準(zhǔn)庫就廣泛使用了這個技巧,比如size_t。另外,因為typedef是定義了一種類型的新別名,不是簡單的字符串替換,所以它比宏來得穩(wěn)健。
?用途四:

為復(fù)雜的聲明定義一個新的簡單的別名。方法是:在原來的聲明里逐步用別名替換一部分復(fù)雜聲明,如此循環(huán),把帶變量名的部分留到最后替換,得到的就是原聲明的最簡化

版。舉例:?

?原聲明:void (*b[10]) (void (*)());

變量名為b,先替換右邊部分括號里的,pFunParam為別名一:

typedef void (*pFunParam)();

再替換左邊的變量b,pFunx為別名二:

typedef void (*pFunx)(pFunParam);

原聲明的最簡化版:

pFunx b[10];
?
原聲明:doube(*)() (*e)[9];

變量名為e,先替換左邊部分,pFuny為別名一:

typedef double(*pFuny)();

再替換右邊的變量e,pFunParamy為別名二

typedef pFuny (*pFunParamy)[9];

原聲明的最簡化版:

pFunParamy e;

理解復(fù)雜聲明可用的“右左法則”:從變量名看起,先往右,再往左,碰到一個圓括號就調(diào)轉(zhuǎn)閱讀的方向;括號內(nèi)分析完就跳出括號,還是按先右后左的順序,如此循環(huán),直

到整個聲明分析完。舉例:

int (*func)(int *p);

首先找到變量名func,外面有一對圓括號,而且左邊是一個*號,這說明func是一個指針;然后跳出這個圓括號,先看右邊,又遇到圓括號,這說明(*func)是一個函數(shù),所以

func是一個指向這類函數(shù)的指針,即函數(shù)指針,這類函數(shù)具有int*類型的形參,返回值類型是int。

int (*func[5])(int *);

func右邊是一個[]運算符,說明func是具有5個元素的數(shù)組;func的左邊有一個*,說明func的元素是指針(注意這里的*不是修飾func,而是修飾func[5]的,原因是[]運算符

優(yōu)先級比*高,func先跟[]結(jié)合)。跳出這個括號,看右邊,又遇到圓括號,說明func數(shù)組的元素是函數(shù)類型的指針,它指向的函數(shù)具有int*類型的形參,返回值類型為int。

這種用法是比較復(fù)雜的,出現(xiàn)的頻率也不少,往往在看到這樣的用法卻不能理解,相信以上的解釋能有所幫助。

5、后補充

陷阱一:?
??????? 記住,typedef是定義了一種類型的新別名,不同于宏,它不是簡單的字符串替換。比如:
先定義:
typedef?char*?PSTR;
然后:
int?mystrcmp(const?PSTR,?const?PSTR);

const?PSTR實際上相當(dāng)于const?char*嗎?不是的,它實際上相當(dāng)于char*?const。 原因在于const給予了整個指針本身以常量性,也就是形成了常量指針char*?const。
簡單來說,記住當(dāng)const和typedef一起出現(xiàn)時,typedef不會是簡單的字符串替換就行。

陷阱二:?
??typedef在語法上是一個存儲類的關(guān)鍵字(如auto、extern、mutable、static、register等一樣),雖然它并不真正影響對象的存儲特性,如:
typedef?static?int?INT2;?//不可行
編譯將失敗,會提示“指定了一個以上的存儲類”。

轉(zhuǎn)載于:https://www.cnblogs.com/xyl-share-happy/archive/2012/05/17/2506928.html

總結(jié)

以上是生活随笔為你收集整理的Typedef用法(转载)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。