C語言指針原來也可以這樣了解
什么是指針
指針是應(yīng)該對(duì)象,可以直接指向電腦存儲(chǔ)器中的某個(gè)地方,這個(gè)地方就是內(nèi)存單元,指針指向的是一個(gè)對(duì)象的地址。地址的指向就是內(nèi)存單元,一個(gè)內(nèi)存單元是一個(gè)字節(jié),在32位平臺(tái)上面,一個(gè)指針是4個(gè)字節(jié)。因?yàn)?2位的平臺(tái)有32根地址線,每根地址線是 1 bit,所以32位平臺(tái)的指針大小是 4 個(gè)字節(jié)。同理,64位平臺(tái)的指針大小就是 8 個(gè)字節(jié)。
指針的應(yīng)用
指針用于指向某個(gè)對(duì)象的地址,也可以通過指針解引用來修改對(duì)象。
int main()
{
int a = 10;
int* p = &a;
printf("%d\n", *p); //10
return 0;
}
&a 就是把 a 的地址取出來。int* 說明 p 是指針變量,指向的對(duì)象是整型。所以 *p 指向的就是 a 的地址,打印出來也是10。因?yàn)?*p和 a 指向的是同一塊內(nèi)存,所以對(duì) *p修改也就是對(duì) a 的值進(jìn)行修改。
int main()
{
int a = 10;
int* p = &a;
*p = 20;
printf("%d\n", a); //20
return 0;
}
這里就是通過 *p 對(duì) a 的值進(jìn)行修改。
指針 ± 整數(shù)
指針 + 指針
指針指向的是一個(gè)對(duì)象。比如指針 + 1,就是指向后一個(gè)元素,拿數(shù)組舉例。
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("%d\n", *arr);
printf("%d\n", *(arr + 1));
printf("%d\n", *(arr + 2));
return 0;
}

從圖片可知,指針每次 + 1,都會(huì)指向數(shù)組的下一個(gè)元素。如果是對(duì)數(shù)組取地址再 + 1的話,就會(huì)跳過整個(gè)數(shù)組,指向整個(gè)數(shù)組后面的地址。所以使用指針加減和取地址加減的時(shí)候,一定要注意。
指針 - 指針
指針 - 指針?biāo)愕氖莾蓚€(gè)指針之間的元素個(gè)數(shù)。
int my_strlen(char* s)
{
char* p = s;
while (*p != '\0')
p++;
return p - s;
}
int main()
{
char arr[10] = "abcdef";
int ret = my_strlen(arr);
printf("%d\n", ret); //6
return 0;
}
算的結(jié)果是 6,因?yàn)橹羔樝鄿p,得到了差的元素個(gè)數(shù)。
野指針
野指針就是指針執(zhí)行的位置使隨機(jī)的,是沒有被分配的內(nèi)存空間。
野指針成因
野指針是由于指針未初始化,指針越界訪問造成的。野指針也是經(jīng)常導(dǎo)致程序崩潰的原因。
指針未初始化
int main()
{
int* p;
*p = 20;
return 0;
}
這里的指針就沒有初始化,沒有初始化指向的內(nèi)容,就是野指針,而且這里連編譯也完成不了。
指針越界訪問
指針越界訪問是一件非??膳碌氖虑椋瑫?huì)導(dǎo)致程序崩潰,程序死循環(huán)。
int main()
{
int i = 0;
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
for (i = 0; i <= 12; i++)
{
arr[i] = 0;
printf("haha\n");
}
return 0;
}
上面這個(gè)代碼就死循環(huán)了,因?yàn)橹羔樤浇缭L問,造成了死循環(huán).這里是一個(gè)壓棧的情況造成的。用下面這張圖來解釋:

上面這張圖就描述了這個(gè)死循環(huán)的過程。
指針和數(shù)組
數(shù)組名其實(shí)表示的就是首元素的地址,傳參的時(shí)候傳數(shù)組名就等于是傳了數(shù)組的首元素地址。因?yàn)閿?shù)組在內(nèi)存當(dāng)中是連續(xù)存儲(chǔ)的,所以只要把首元素地址傳過去就可以了。
int main()
{
int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
printf("%p\n", &arr[0]);
printf("%p\n", arr);
return 0;
}

通過圖片就可以看出。
指針 ± 和數(shù)組的關(guān)系
當(dāng)指針 ± 的時(shí)候,解引用指向數(shù)組的元素也在發(fā)生改變,+1,表示向后移動(dòng)一個(gè)元素。
int main()
{
int arr[] = { 1,2,3,4,5,6,7,8,9,0 };
int* p = arr; //用指針來存放數(shù)組首元素的地址
int sz = sizeof(arr) / sizeof(arr[0]);
int i = 0;
for (i = 0; i < sz; i++)
{
printf("&arr[%d] = %p <====> p+%d = %p\n", i, &arr[i], i, p + i);
}
return 0;
}
使用數(shù)組的時(shí)候,也可以換做指針來使用,看起來會(huì)整潔一些。
二級(jí)指針
因?yàn)橹羔樧兞恳彩亲兞浚亲兞烤蜁?huì)有地址,那么指針變量的地址就可以用二級(jí)指針來描述。就是把一個(gè)變量的地址放在指針里面,然后再把指針的地址放在二級(jí)指針里面。所以就可以通過對(duì)二級(jí)指針解引用拿到一級(jí)指針的地址,對(duì)二級(jí)指針兩次解引用,就拿到變量的地址了。
int main()
{
int a = 10;
int* p = &a;
int** pp = &p;
printf("%p\n", a);
printf("%p\n", *p);
printf("%p\n", **pp);
return 0;
}

既然能拿到地址,那么對(duì)pp解引用之后修改,也就修改了a的值。
int main()
{
int a = 10;
int* p = &a;
int** pp = &p;
**pp = 20;
printf("%d\n", a);
return 0;
}

所以在使用二級(jí)指針的時(shí)候,也可以直接對(duì)變量進(jìn)行修改。
總結(jié)
本篇文章就到這里了,希望能給你帶來幫助,也希望能夠您能夠關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
關(guān)于C++中的static關(guān)鍵字的總結(jié)
C++的static有兩種用法:面向過程程序設(shè)計(jì)中的static和面向?qū)ο蟪绦蛟O(shè)計(jì)中的static。前者應(yīng)用于普通變量和函數(shù),不涉及類;后者主要說明static在類中的作用2013-09-09
Linux下g++編譯與使用靜態(tài)庫和動(dòng)態(tài)庫的方法
下面小編就為大家?guī)硪黄狶inux下g++編譯與使用靜態(tài)庫和動(dòng)態(tài)庫的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05
VC中LINK 2001 和 LINK 2009 的錯(cuò)誤的解決方法
最近將兩個(gè)開源C++項(xiàng)目編譯成windows版本的時(shí)候遇到很多問題,編譯的時(shí)候總是報(bào)錯(cuò),報(bào)的最多的是無法解析的外部符號(hào)”,經(jīng)過近3天的折騰總算都通過了,這里是一些總結(jié)2020-10-10
C語言詳解strcmp函數(shù)的分析及實(shí)現(xiàn)
strcmp函數(shù)語法為“int strcmp(char *str1,char *str2)”,其作用是比較字符串str1和str2是否相同,如果相同則返回0,如果不同,前者大于后者則返回1,否則返回-12022-05-05
Opencv基于CamShift算法實(shí)現(xiàn)目標(biāo)跟蹤
這篇文章主要為大家詳細(xì)介紹了Opencv基于CamShift算法實(shí)現(xiàn)目標(biāo)跟蹤,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
C語言 字符串首字母轉(zhuǎn)換成大寫簡(jiǎn)單實(shí)例
這篇文章主要介紹了C語言 字符串首字母轉(zhuǎn)換成大寫簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-05-05

