# C指針
學習 C 語言的指針既簡單又有趣。通過指針,可以簡化一些 C 編程任務的執行,還有一些任務,如動態內存分配,沒有指針是無法執行的。所以,想要成為一名優秀的 C 程序員,學習指針是很有必要的。
正如您所知道的,每一個變量都有一個內存位置,每一個內存位置都定義了可使用&運算符訪問的地址,它表示了在內存中的一個地址。
請看下面的實例,它將輸出定義的變量地址:
## 實例
```
#include <stdio.h>
int main ()
{
int var_runoob = 10;
int *p; // 定義指針變量
p = &var_runoob;
printf("var_runoob 變量的地址: %p\n", p);
return 0;
}
```
當上面的代碼被編譯和執行時,它會產生下列結果:
~~~
var_runoob 變量的地址: 0x7ffeeaae08d8
~~~

通過上面的實例,我們了解了什么是內存地址以及如何訪問它。接下來讓我們看看什么是指針。
## 什么是指針?
**指針**是一個變量,其值為另一個變量的地址,即,內存位置的直接地址。就像其他變量或常量一樣,您必須在使用指針存儲其他變量地址之前,對其進行聲明。指針變量聲明的一般形式為:
~~~
type *var-name;
~~~
在這里,**type**是指針的基類型,它必須是一個有效的 C 數據類型,**var-name**是指針變量的名稱。用來聲明指針的星號\*與乘法中使用的星號是相同的。但是,在這個語句中,星號是用來指定一個變量是指針。以下是有效的指針聲明:
```
int *ip; /* 一個整型的指針 */
double *dp; /* 一個 double 型的指針 */
float *fp; /* 一個浮點型的指針 */
char *ch; /* 一個字符型的指針 */
```
所有實際數據類型,不管是整型、浮點型、字符型,還是其他的數據類型,對應指針的值的類型都是一樣的,都是一個代表內存地址的長的十六進制數。
不同數據類型的指針之間唯一的不同是,指針所指向的變量或常量的數據類型不同。
## 如何使用指針?
使用指針時會頻繁進行以下幾個操作:定義一個指針變量、把變量地址賦值給指針、訪問指針變量中可用地址的值。這些是通過使用一元運算符\*來返回位于操作數所指定地址的變量的值。下面的實例涉及到了這些操作:
## 實例
```
#include <stdio.h>
int main ()
{
int var = 20; /* 實際變量的聲明 */
int *ip; /* 指針變量的聲明 */
ip = &var; /* 在指針變量中存儲 var 的地址 */
printf("var 變量的地址: %p\n", &var );
/* 在指針變量中存儲的地址 */
printf("ip 變量存儲的地址: %p\n", ip );
/* 使用指針訪問值 */
printf("*ip 變量的值: %d\n", *ip );
return 0;
}
```
當上面的代碼被編譯和執行時,它會產生下列結果:
~~~
var 變量的地址: 0x7ffeeef168d8
ip 變量存儲的地址: 0x7ffeeef168d8
*ip 變量的值: 20
~~~
## C 中的 NULL 指針
在變量聲明的時候,如果沒有確切的地址可以賦值,為指針變量賦一個 NULL 值是一個良好的編程習慣。賦為 NULL 值的指針被稱為**空**指針。
NULL 指針是一個定義在標準庫中的值為零的常量。請看下面的程序:
## 實例
```
#include <stdio.h>
int main ()
{
int *ptr = NULL;
printf("ptr 的地址是 %p\n", ptr );
return 0;
}
```
當上面的代碼被編譯和執行時,它會產生下列結果:
~~~
ptr 的地址是 0x0
~~~
在大多數的操作系統上,程序不允許訪問地址為 0 的內存,因為該內存是操作系統保留的。然而,內存地址 0 有特別重要的意義,它表明該指針不指向一個可訪問的內存位置。但按照慣例,如果指針包含空值(零值),則假定它不指向任何東西。
如需檢查一個空指針,您可以使用 if 語句,如下所示:
```
if(ptr) /* 如果 p 非空,則完成 */
if(!ptr) /* 如果 p 為空,則完成 */
```
## C 指針詳解
在 C 中,有很多指針相關的概念,這些概念都很簡單,但是都很重要。下面列出了 C 程序員必須清楚的一些與指針相關的重要概念:
| 概念 | 描述 |
| --- | --- |
| 指針的算術運算 | 可以對指針進行四種算術運算:++、--、+、- |
| 指針數組 | 可以定義用來存儲指針的數組。 |
| 指向指針的指針| C 允許指向指針的指針。 |
| 傳遞指針給函數 | 通過引用或地址傳遞參數,使傳遞的參數在調用函數中被改變。 |
| 從函數返回指針 | C 允許函數返回指針到局部變量、靜態變量和動態內存分配。 |
- 空白目錄
- 第一章 c語言簡介
- 1 通俗地理解什么是編程語言
- 2 C語言究竟是一門怎樣的語言
- 第二章 算法簡介&基本語法
- 1 算法簡介
- 2 C 程序結構
- 3 C 基本語法
- 第三章 數據類型
- 1 數據類型
- 2 變量
- 3 常量
- 第四章 運算符
- 1 算術運算符
- 2 關系運算符
- 3 邏輯運算符
- 4 位運算符
- 5 賦值運算符
- 6 雜項運算符(其他運算符)
- 7 c語言中的運算符優先級
- 第五章 控制流
- 1 判斷語句
- 2 switch語句
- 3 循環語句
- 4 流程控制相關案例
- 第六章 指針
- 1 c語言指針概述
- 2 指針的算術運算
- 3 指針數組
- 4 指向指針的指針
- 5 傳遞指針給函數
- 6 從函數返回指針
- 第七章 函數
- 1 函數的語法
- 2 作用域規則
- 3 形參與實參
- 第八章 數組
- 1 C語言中的數組
- 2 多維數組
- 3 傳遞數組給函數
- 4 從函數返回數組
- 第九章 指針
- 1 分鐘徹底理解C語言指針的概念
- 2 C語言指針變量的定義和使用
- 3 C語言指針變量的運算(加法、減法和比較運算)
- 4 C語言數組指針(指向數組的指針)
- 5 C語言字符串指針(指向字符串的指針)
- 第十章 結構體
- 第十一章 練習
- 第十二章 文件操作
- 第十三章 文件操作2