野指標+空指標+萬能指標
2.3 指標大小
l 使用sizeof()測量指標的大小,得到的總是:4或8
l sizeof()測的是指標變數指向儲存地址的大小
l 在32位平臺,所有的指標(地址)都是32位(4位元組)
l 在64位平臺,所有的指標(地址)都是64位(8位元組)
int *p1;
int **p2;
char *p3;
char **p4;
printf("sizeof(p1) = %d\n", sizeof(p1));
printf("sizeof(p2) = %d\n", sizeof(p2));
printf("sizeof(p3) = %d\n", sizeof(p3));
printf
printf("sizeof(double *) = %d\n", sizeof(double *));
7.2.4 野指標和空指標
指標變數也是變數,是變數就可以任意賦值,不要越界即可(32位為4位元組,64位為8位元組),但是,任意數值賦值給指標變數沒有意義,因為這樣的指標就成了野指標,此指標指向的區域是未知(作業系統不允許操作此指標指向的記憶體區域)。所以,野指標不會直接引發錯誤,操作野指標指向的記憶體區域才會出問題。
int a = 100;
int *p;
p = a; //把a的值賦值給指標變數p,p為野指標, ok,不會有問題,但沒有意義
p = 0x12345678; //給指標變數p賦值,p為野指標, ok,不會有問題,但沒有意義
*p = 1000; //操作野指標指向未知區域,記憶體出問題,err
但是,野指標和有效指標變數儲存的都是數值,為了標誌此指標變數沒有指向任何變數(空閒可用),C語言中,可以把NULL賦值給此指標,這樣就標誌此指標為空指標,沒有任何指標。
int *p = NULL;
NULL是一個值為0的巨集常量:
#define NULL ((void *)0)
7.2.5萬能指標void *
void *指標可以指向任意變數的記憶體空間:
void *p = NULL;
int a = 10;
p = (void *)&a; //指向變數時,最好轉換為void *
//使用指標變數指向的記憶體時,轉換為int *
*( (int *)p ) = 11;
printf("a = %d\n", a);
7.2.6 const修飾的指標變數
int a = 100;
int b = 200;
//指向常量的指標
//修飾*,指標指向記憶體區域不能修改,指標指向可以變
const int *p1 = &a; //等價於int const *p1 = &a;
//*p1 = 111; //err
p1 = &b; //ok
//指標常量
//修飾p1,指標指向不能變,指標指向的記憶體可以修改
int * const p2 = &a;
//p2 = &b; //err
*p2 = 222; //ok
指標 指標的定義和使用: 1、指標:是一種資料型別 指標變數也是一種變數 2、指標格式: 對應的資料型別 * p:指標型別變數 用來指向一個變數的地址 3、通過指標修改變數的值 *p = 200; 4、指標型別在記憶體中的大小 : 在32位作業系統中所有指標大小都是4個位元組大小 列印指標記憶體大小格式: sizeof(int *) || sizeof(p) 5、記憶體按照 unsignned int 為每個一個記憶體分配編號 6、講解: ①定義變數 int a =10; 數值10 存在記憶體 為a開闢的空間中 a的地址為0xff00 ②定義指標 int * p = &a 指標變數p 在記憶體中存貯的是 a的地址 0xff00,指標變數p 在記憶體中的大小在32位系統中都是4個位元組大小 ③*p = 200 通過操作指標變數p 所儲存的 a的地址 來改變a的值 ④指標p 有一個自己的記憶體地址 指標p地址 與 定義變數a的地址 0xff00不同野指標、空指標與 萬能指標 1、野指標:野指標是指向一個未知的記憶體空間,可能在讀寫的時候出現錯誤。 0-255都是系統保留的 不可以讀,不可以寫 2、空指標 沒有指向任何的地址(其指向0的地址) 空制指標就是指向記憶體編號為零的空間,操作該記憶體空間會報錯,一般情況空指標用於程式條件判斷 3、萬能指標:void * 指標可以指向任意變數的記憶體空間 const修飾指標 1.通過指標修改const修飾的常量 const int a = 10; int *p += &a; *p = 100; a =100; 可以通過1級指標修改一個常量的值 2.const修飾指標型別 int * 不能改變指標變數指向的記憶體地址的值 但是可以改變指標指向的地址 const int *p = &a p=&b *p = 100 可以改變指標指向的地址 3.const修飾指標變數 能改變指標指向地址的值,但不能改變指標指向的地址 int * const p = &a; *p = 100; p=&b 可以修改指標指向地址的值 4.const修飾指標型別修飾指標變數 不能改變指標指向的的值 也不能改變指標指向的地址 const int * const *p = &a; *p =100; p = &b;
指標和陣列、指標運算: 1、陣列名是陣列的首地址,這是一個常量 2、指向陣列的指標 格式int arr [10] = { 0 } ; int * p = arr; 當操作指標的時候 間接操作了陣列 arr[i] = p[i]; 3、指標的降級操作 ,取當前地址的值:p[5]、*p 4、對指向陣列的指標進行加減操作,可能會導致陣列下標越界 。 5、相同的型別的指標相減 結果是兩個指標相差的長度 區別:陣列名通過sizeof可以求出陣列大小,指標只包含陣列的首地址資訊 課後練習題:輸入格式:【 空格 你好 空格 】輸出格式為【你好】 指標陣列 1、儲存char *型別的地址陣列 char * arr[] ={"hello","world","niao","baobei"}; 2、指標陣列,它是陣列,陣列的每個元素都是指標型別。 多級指標: 1、如果二級指標前面加一個* 代表一級指標的值 2、二級指標前面加** 代表指標指向一級指標指向地址的值,加*降維度 3、如果n級指標在前面加n個*就是指標指向一級指標指向地址的值