再談C指標

來源:互聯網
上載者:User

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 的數值未知;

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.