1. 程式人生 > >關於typedef,指標陣列和陣列指標,指標函式的說明

關於typedef,指標陣列和陣列指標,指標函式的說明

用途一:
定義一種型別的別名,而不只是簡單的巨集替換。可以用作同時宣告指標型的多個物件。比如:
char* pa, pb; // 這多數不符合我們的意圖,它只聲明瞭一個指向字元變數的指標, 
// 和一個字元變數;
以下則可行:
typedef char* PCHAR; // 一般用大寫
PCHAR pa, pb; // 可行,同時聲明瞭兩個指向字元變數的指標
雖然:
char *pa, *pb;
也可行,但相對來說沒有用typedef的形式直觀,尤其在需要大量指標的地方,typedef的方式更省事。

用途二:
用在舊的C程式碼中(具體多舊沒有查),幫助struct。以前的程式碼中,宣告struct新物件時,必須要帶上struct,即形式為: struct 結構名 物件名,如:
struct tagPOINT1
{
int x;
int y;
};
struct tagPOINT1 p1; 

而在C++中,則可以直接寫:結構名 物件名,即:
tagPOINT1 p1;

估計某人覺得經常多寫一個struct太麻煩了,於是就發明了:
typedef struct tagPOINT
{
int x;
int y;
}POINT;

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

或許,在C++中,typedef的這種用途二不是很大,但是理解了它,對掌握以前的舊程式碼還是有幫助的,畢竟我們在專案中有可能會遇到較早些年代遺留下來的程式碼。

用途三:
用typedef來定義與平臺無關的型別。
比如定義一個叫 REAL 的浮點型別,在目標平臺一上,讓它表示最高精度的型別為:
typedef long double REAL; 
在不支援 long double 的平臺二上,改為:
typedef double REAL; 
在連 double 都不支援的平臺三上,改為:
typedef float REAL; 
也就是說,當跨平臺時,只要改下 typedef 本身就行,不用對其他原始碼做任何修改。
標準庫就廣泛使用了這個技巧,比如size_t。
另外,因為typedef是定義了一種型別的新別名,不是簡單的字串替換,所以它比巨集來得穩健(雖然用巨集有時也可以完成以上的用途)。

用途四:
為複雜的宣告定義一個新的簡單的別名。方法是:在原來的聲明裡逐步用別名替換一部分複雜宣告,如此迴圈,把帶變數名的部分留到最後替換,得到的就是原宣告的最簡化版。舉例:

1. 原宣告:int *(*a[5])(int, char*);
變數名為a,直接用一個新別名pFun替換a就可以了:
typedef int *(*pFun)(int, char*); 
原宣告的最簡化版:
pFun a[5]; 

2. 原宣告:void (*b[10]) (void (*)());
變數名為b,先替換右邊部分括號裡的,pFunParam為別名一:
typedef void (*pFunParam)();
再替換左邊的變數b,pFunx為別名二:
typedef void (*pFunx)(pFunParam);
原宣告的最簡化版:
pFunx b[10];

3. 原宣告:doube(*)() (*e)[9]; 
變數名為e,先替換左邊部分,pFuny為別名一:
typedef double(*pFuny)();
再替換右邊的變數e,pFunParamy為別名二
typedef pFuny (*pFunParamy)[9];
原宣告的最簡化版:
pFunParamy e; 

理解複雜宣告可用的“右左法則”:從變數名看起,先往右,再往左,碰到一個圓括號就調轉閱讀的方向;括號內分析完就跳出括號,還是按先右後左的順序,如此迴圈,直到整個宣告分析完。舉例:
int (*func)(int *p);
首先找到變數名func,外面有一對圓括號,而且左邊是一個*號,這說明func是一個指標;然後跳出這個圓括號,先看右邊,又遇到圓括號,這說明(*func)是一個函式,所以func是一個指向這類函式的指標,即函式指標,這類函式具有int*型別的形參,返回值型別是int。
int (*func[5])(int *);
func右邊是一個[]運算子,說明func是具有5個元素的陣列;func的左邊有一個*,說明func的元素是指標(注意這裡的*不是修飾func,而是修飾func[5]的,原因是[]運算子優先順序比*高,func先跟[]結合)。跳出這個括號,看右邊,又遇到圓括號,說明func陣列的元素是函式型別的指標,它指向的函式具有int*型別的形參,返回值型別為int。

也可以記住2個模式:
type (*)(....)函式指標 
type (*)[]陣列指標 
---------------------------------

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

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

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

相關推薦

關於typedef指標陣列陣列指標指標函式的說明

用途一: 定義一種型別的別名,而不只是簡單的巨集替換。可以用作同時宣告指標型的多個物件。比如: char* pa, pb; // 這多數不符合我們的意圖,它只聲明瞭一個指向字元變數的指標,  // 和一個字元變數; 以下則可行: typedef char* PCHAR; // 一般用大寫 PCHAR pa,

陣列記憶體申請釋放指標陣列陣列指標

一 陣列指標的空間釋放 int (*p)[3] = new int [4][3]; // ... delete []p; //---1 delete[](*p); //---2在釋放這個二維陣列時,應該使用1和2哪種方式呢?哪種對呢? 其實兩種方法都是可以的,二維陣

演算法:給定一個整數陣列一個目標值找出陣列中和為目標值的兩個數、判斷一個整數是否是迴文數

<!-- 給定一個整數陣列和一個目標值,找出陣列中和為目標值的兩個數。 你可以假設每個輸入只對應一種答案,且同樣的元素不能被重複利用。 示例: 給定 nums = [2, 7, 11, 15], target = 9 因為 nums[0] + nums[1] = 2 + 7 = 9

go語言學習--指標陣列陣列指標

陣列指標(也稱行指標)定義 int (*p)[n];()優先順序高,首先說明p是一個指標,指向一個整型的一維陣列,這個一維陣列的長度是n,也可以說是p的步長。也就是說執行p+1時,p要跨過n個整型資料的長度。 如要將二維陣列賦給一指標,應這樣賦值:int a[3][4];int (*p)[4]; //

Java的陣列list升序降序逆序函式Collections.sortArrays.sort的使用

list升序,降序,逆序List<Integer>list =new ArrayList<Integer>();//如果list是 5 7 2 6 8 1 41、升序:Collections.sort(list) //list: 1 2 4 5 6 7 82、降序:Collection

18. C語言 -- 指標陣列陣列指標

本部落格主要內容為 “小甲魚” 視訊課程《帶你學C帶你飛》【第一季】 學習筆記,文章的主題內容均來自該課程,在這裡僅作學習交流。在文章中可能出現一些錯誤或者不準確的地方,如發現請積極指出,十分感謝。 也歡迎大家一起討論交流,如果你覺得這篇文章對你有所幫助,記得評論、點贊哦 ~(。

Java基礎個人筆記之ScannerRandom類陣列

Scaner類作用:接收鍵盤錄入 1.用前需先用import匯入Scanner包 import java.util.Scanner;匯入包 public class ScannerDemo{ public static void main(String[] args){ //建立Scanne

c語言的指標陣列陣列指標函式指標

#include <stdio.h> #include <stdlib.h> int func(int x){ return x; } int* func2(int x){ int *p=&x; return

對c語言中指標陣列陣列指標的認識

1.陣列:一批具有同名的同屬性的資料就組成了一個數組。(“[ ]”也是陣列型別的一部分)     由此可知:(1)陣列是一組有序資料的集合;(2)陣列中的每一個元素都屬於同一個資料型別。  定義一維陣列的一般形式:型別符  陣列名【常量表達式】(陣列也

淺析指標陣列陣列指標

/************************************************************************/ /* 淺析指標陣列和陣列指標 指標陣列:array of pointers 陣列指標:a pointer to an array 舉例說明

C語言陣列篇(二)指標陣列陣列指標

陣列指標 和 指標陣列           這兩個名詞可以說是經常搞混了         陣列指標--> 陣列的

PHP陣列XML相互轉換的函式微信中常見的Sign生成函式

//陣列轉Xml function ArrToXml($arr) { if(!is_array($arr) || count($arr) == 0) return ''; $xml = "<xml>"; foreach ($arr as $key=>$val) {

劍指off:在一個二維陣列中(每個一維陣列的長度相同)每一行都按照從左到右遞增的順序排序每一列都按照從上到下遞增的順序排序。請完成一個函式,輸入這樣一個二維陣列和一個整數判斷陣列中是否含有該整數

題目描述 在一個二維陣列中(每個一維陣列的長度相同),每一行都按照從左到右遞增的順序排序,每一列都按照從上到下遞增的順序排序。請完成一個函式,輸入這樣的一個二維陣列和一個整數,判斷陣列中是否含有該整數。 思路: 從左下角x開始尋找,x為此列最大數字,此行最小數字。如果目

給定一個整數陣列一個目標值找出陣列中和為目標值的兩個數。

給定一個整數陣列和一個目標值,找出陣列中和為目標值的兩個數。 你可以假設每 示例: 給定 nums = [2, 7, 11, 15], target = 9 因為 nums[0] + nums[1] = 2 + 7 = 9 所以返回 [0, 1] 個輸入只對應一種答

C++小知識——指標函式函式指標指標陣列陣列指標示例

一、指標函式和函式指標 1.1 指標函式 指標函式,其實就是返回值為指標的函式 例如: long *func(int) 它是一個名為func,返回值為long *,引數為int的指標函式 1.2 函式指標 就像陣列名是指向陣列第一個元素的常指標一樣,函式

前端演算法:給定一個整數陣列一個目標值找出陣列中和為目標值的兩個數、判斷一個整數是否是迴文數

<!-- 給定一個整數陣列和一個目標值,找出陣列中和為目標值的兩個數。 你可以假設每個輸入只對應一種答案,且同樣的元素不能被重複利用。 示例: 給定 nums = [2, 7, 11, 15], target = 9 因為 nums[0] + nums[

char a[](字串陣列)char *a(字串指標)區別

在C語言中,對字串的操作主要有兩種方式,一是使用字元陣列,char str[];二是使用字元指標。那麼二者有什麼區別呢?下面將分述二者的使用,最後進行比較。 一、字元陣列         使用char str[]定義一個字元陣列str,中括號內可以寫上數字表示陣列大