1.指標是一個儲存對象地址的變數。這裡要注意的是,指標跟地址概念的不同,
指標是一種地址變數,通常也叫指標變數,統稱指標。而地址則是地址變數的值。
2.對一個一般類型的地址進行sizeof運算,結果是4個位元組,而a[0]由於代表了一個數組,
sizeof(a[0])的結果是整個數組的長度8x9xsizeof(int),並非4個位元組。
具有數群組類型的地址跟數組名一樣都是一個符號地址常量,因此它必定是一個右值。
3(*p)[9]就是這種直接聲明符加[]的形式。
p左邊的*在這裡不是取值運算子,而是一個聲明符,它指出p是一個指標。
而()括弧是不能去掉的,如果去掉了,由於[]運算子優先順序比*高,p就會先跟[]結合,
這樣p就變成了一個指標數組,而不是指向數組的指標。
4.為什麼除了第一維之外其他維數的都要寫明大小
int p[][6];
定義一個指標的時候,首先必須定出指標的類型,由於這(第一維的值)是一個指向數組的指標,
如果數組的元素的類型定下來了,那麼這個指標的類型也就定下來了。前面說過,
C語言的多維陣列實質上是數組的嵌套,那麼所指向數組的元素必定具有“”數群組類型“”,
也就是說,這個數組的元素是一個具有6個int元素的數組,
因此,p定義的時候,必須指定第二維的上界,這樣才能把p的類型定下來。
5.字串常量與一般數組的主要區別,是字串常量存放在靜態儲存區,而一般數組
(非static)則是在棧中靜態分配的。
6.聲明修飾符的先後問題
對於儲存類說明符、類型說明符和類型限定符的排列順序,C標準並沒有規定其順序,
誰嵌套誰都可以。換言之,上面的聲明可以寫成:
int static const i=10, j=20, k=30;或者const int static i=10, j=20, k=30;
這無所謂,跟原聲明是一樣的。再舉一個有趣的例子:
上面舉的例子是變數的聲明,函數的聲明也同樣道理,例如:
static const int func(void);
7.
typedef int a[10];
typedef void (*p)(void);
他們會以為a[10]是int的別名,(*p)(void)是void的別名,但這樣的別名看起來又似乎不是合法的名字,
於是陷入困惑之中。
實際上,上面的語句把a聲明為具有10個int元素的數組的類型別名,p是一種函數指標的類型別名。
雖然在功能上,typedef可以看作一個跟int PARA分離的動作,但文法上typedef屬於儲存類聲明說明符,
因此嚴格來說,typedef int PARA整個是一個完整的聲明。
例子:
typedef int a[2];
void main()
{
a b;
b[0] = 1;
b[1] = 2;
printf("%d/n", b[0]);
}
8.
const int *p;與int * const p的區別,這兩個聲明的中文名稱常常搞得混亂不堪。
第一個聲明的const是聲明說明符,它修飾p所指向的對象,但p仍然是可變的,
這意味著p是一個指向常量的指標,簡稱常量指標。第二個聲明的const是聲明符的一部分,
它修飾的對象是p,這意味著p是一個常量,而且是一個指標類型的常量,簡稱指標常量。
//例子:
int a = 2;
int b = 3;
const int *p = &a;
//p的值可以改變 改變後*p值改變 但是不能直接給*p賦值
//比如 *p = 3;是錯的 但是可以p = &b;則*p 的值變為3
int * const p2 = &a;
//p2是常量 既不能改變p2的值 p2 = &b是錯的 但是可以改變*p2的值
//比如*p2 = 10; 則此時 a = 10;
*p2 = 10;
p = &b;
printf("*p %d /n",*p);
printf("*p2 %d /n",*p2);
“野指標”產生的3種原因:
指標變數沒有被初始化。指標變數剛被建立時不會自動成為NULL指標,它的預設值是隨機的,它會亂指一氣;
指標p被free或者delete之後,沒有置為NULL,讓人誤以為p是個合法的指標;
指標操作超越了變數的作用範圍;
防止野指標的發生
用malloc或new申請記憶體之後,應該立即檢查指標值是否為NULL。
不要忘記為數組和動態記憶體賦初值。防止將未被初始化的記憶體作為右值使用。
避免數組或指標的下標越界,特別要當心發生“多1”或者“少1”操作。
動態記憶體的申請與釋放必須配對,防止記憶體流失。
用free或delete釋放了記憶體之後,立即將指標設定為NULL,防止產生“野指標”。
大儲存類型的指標向小儲存類型的指標轉換,會帶來錯誤風險
如 int a =0x12345678;
int *pa = &a;
char *pc = (char*)pa;
(*pa)的數值為0x12345678; (*pc)的數值為0x12;
小儲存類型的指標向大儲存類型的指標轉換,會帶來極大的錯誤風險
如 char b = 0x12;
char *pb = &b;
int *pd = (int*)pb;
*pb的數值為0x12,*pd 的數值未知;