當前位置:才華齋>計算機>C語言>

C語言中指標的用法介紹

C語言 閱讀(2.49W)

指標是一個特殊的變數,它裡面儲存的數值被解釋成為記憶體裡的一個地址。要搞清一個指標需要搞清指標的四方面的內容:指標的型別,指標所指向的型別,指

C語言中指標的用法介紹

針的值或者叫指標所指向的記憶體區,還有指標本身所佔據的記憶體區。讓我們分別說明。

先宣告幾個指標放著做例子:

例一:

(1)int *ptr;

(2)char *ptr;

(3)int **ptr;

(4)int (*ptr)[3];

(5)int *(*ptr)[4];

1。 指標的型別。

從語法的角度看,你只要把指標宣告語句裡的指標名字去掉,剩下的部分就是這個指標的型別。這是指標本身所具有的型別。讓我們看看例一中各個指標的型別:

(1)int *ptr; //指標的型別是 int *

(2)char *ptr; //指標的型別是char *

(3)int **ptr; //指標的型別是 int **

(4)int (*ptr)[3]; //指標的型別是 int(*)[3]

(5)int *(*ptr)[4]; //指標的型別是 int *(*)[4]

怎麼樣?找出指標的型別的方法是不是很簡單?

2。指標所指向的型別。

當你通過指標來訪問指標所指向的記憶體區時,指標所指向的型別決定了編譯器將把那片記憶體區裡的內容當做什麼來看待。從語法上看,你只須把指標宣告語句中的指標名字和名字左邊的指標宣告符*去掉,剩下的就是指標所指向的型別。例如:

(1)int *ptr; //指標所指向的型別是 int

(2)char *ptr; //指標所指向的的型別是char

(3)int **ptr; //指標所指向的的型別是 int *

(4)int (*ptr)[3]; //指標所指向的的型別是 int()[3]

(5)int *(*ptr)[4]; //指標所指向的的型別是 int *()[4]

在指標的算術運算中,指標所指向的型別有很大的作用。指標的型別(即指標本身的型別)和指標所指向的型別是兩個概念。當你對C越來越熟悉時,你會發現,把與指標攪和在一起的"型別"這個概念分成"指標的型別"和"指標所指向的型別"兩個概念,是精通指標的關鍵點之一。我看了不少書,發現有些寫得差的書中,就把指標的這兩個概念攪在一起了,所以看起書來前後矛盾,越看越糊塗。

3。 指標的值,或者叫指標所指向的記憶體區或地址。

指標的值是指標本身儲存的數值,這個值將被編譯器當作一個地址,而不是一個一般的數值。在32位程式裡,所有型別的指標的值都是一個32位整數,因為32位程式裡記憶體地址全都是32位長。指標所指向的記憶體區就是從指標的值所代表的那個記憶體地址開始,長度為si

Generated by Foxit PDF Creator ? Foxit Software

For evaluation (指標所指向的型別)的一片記憶體區。以後,我們說一個指標的值是XX,就相

當於說該指標指向了以XX為首地址的一片記憶體區域;我們說一個指標指向了某塊記憶體區域,就相當於說該指標的值是這塊記憶體區域的首地址。指標所指向的記憶體區和指標所指向的型別是兩個完全不同的概念。在例一中,指標所指向的型別已經有了,但由於指標還未初始化,所以它所指向的記憶體區是不存在的,或者說是無意義的。以後,每遇到一個指標,都應該問問:這個指標的型別是什麼?指標指向的型別是什麼?該指標指向了哪裡?

4。 指標本身所佔據的記憶體區。

指標本身佔了多大的記憶體?你只要用函式sizeof(指標的型別)測一下就知道了。在32位平臺裡,指標本身佔據了 4個位元組的長度。指標本身佔據的記憶體這個概念在判斷一個指標表示式是否是左值時很有用。

第二章。指標的算術運算

指標可以加上或減去一個整數。指標的這種運算的意義和通常的數值的加減運算的意義是不一樣的。例如:

例二:

1。 char a[20];

2。 int *ptr=a;

...

...

3。 ptr++;

在上例中,指標ptr的型別是 int*,它指向的型別是 int,它被初始化為指向整形變數a。接下來的第3句中,指標ptr被加了1,編譯器是這樣處理的:它把指標ptr的值加上了sizeof(int),在32位程式中,是被加上了4。由於地址是用位元組做單位的,故 ptr 所指向的地址由原來的變數 a的地址向高地址方向增加了 4 個位元組。

由於char型別的長度是一個位元組,所以,原來 ptr是指向陣列a的第0號單元開始的四個位元組,此時指向了陣列a中從第4號單元開始的四個位元組。

我們可以用一個指標和一個迴圈來遍歷一個數組,看例子:

例三:

例三:

int array[20];

int *ptr=array;

...

//此處略去為整型陣列賦值的程式碼。

...

for(i=0;i<20;i++)

{

(*ptr)++;

ptr++;

Generated by Foxit PDF Creator ? Foxit Software

For evaluation only.}

這個例子將整型陣列中各個單元的值加1。由於每次迴圈都將指標ptr加1,所以每次迴圈都能訪問陣列的下一個單元。

再看例子:

例四:

1。 char a[20];

2。 int *ptr=a;

...

...

3。 ptr+=5;

在這個例子中,ptr被加上了5,編譯器是這樣處理的:將指標ptr的值加上5乘 sizeof(int),在 32 位程式中就是加上了 5 乘 4=20。由於地址的單位是位元組,故現在的 ptr 所指向的地址比起加 5 後的 ptr 所指向的地址來說,向高地址方向移動了20個位元組。在這個例子中,沒加 5前的ptr指向陣列a的第0號單元開始的四個位元組,加 5 後,ptr 已經指向了陣列 a 的合法範圍之外了。雖然這種情況在應用上會出問題,但在語法上卻是可以的。這也體現出了指標的靈活性。