C語(yǔ)言指針類型與野指針引起的原因
一、指針是什么
指針也就是 內(nèi)存地址 ,在計(jì)算機(jī)上我們?cè)L問(wèn)數(shù)據(jù)需要通過(guò)內(nèi)存地址來(lái)訪問(wèn),在C語(yǔ)言中,指針變量是用來(lái)存放內(nèi)存地址的變量,在不同系統(tǒng)下,指針(內(nèi)存)地址的長(zhǎng)度不同,32位CPU下,由于有32根地址線,所以指針(內(nèi)存)地址是由32個(gè)bit位組成的,也就是4Byte,在64位CPU下,有64根地址線,所以地址由64個(gè)bit位組成,也就是8Byte,指針指向的地址都是一個(gè)內(nèi)存單元,一個(gè)內(nèi)存單元里面有1byte的數(shù)據(jù)。
創(chuàng)建第一個(gè)指針變量:
#include <stdio.h>
int a = 8; //先創(chuàng)建一個(gè)變量,此變量在內(nèi)存中有自己的地址。
int *pa = &a; //&符號(hào)取出a的地址,交給pa指針,*符號(hào)表示pa是指針類型
*pa = 10; //我們可以通過(guò)*pa修改a的值(沒(méi)有*的話pa是a的地址,*是解引用的意思,加上之后pa代表的就不是a的地址而是a)
printf("%d",*pa); //10

二、指針和指針類型
指針是用來(lái)存放地址的,那么為什么還分為char* int* short* long*…….?
一般情況下,char**類型的指針是為了存放char類型變量的地址,int類型的指針是為了存放int類型的地址,short類型的指針是為了存放short類型的地址…….但實(shí)際上,指針的類型有一下兩大作用:
- 指針類型決定了指針進(jìn)行±整數(shù)的時(shí)候±的步長(zhǎng)(字節(jié))
- 指針類型決定了對(duì)指針進(jìn)行解引用的時(shí)候能訪問(wèn)幾個(gè)字節(jié)
1.指針±整數(shù)
代碼示例:
#include <stdio.h>
int main()
{
int a = 10;
char* char_pa = (char*)&a;
short* short_pa = (short*)&a;
int* int_pa = (int*)&a;
printf("%p\n",&a); //00000071DCEFFC24 %p輸出a的的地址
printf("%p\n",char_pa); //00000071DCEFFC24
printf("%p\n",char_pa+1); //00000071DCEFFC25 加了一個(gè)字節(jié)
printf("%p\n",short_pa); //00000071DCEFFC24
printf("%p\n",short_pa+1); //00000071DCEFFC26 加了兩個(gè)字節(jié)
printf("%p\n",int_pa); //00000071DCEFFC24
printf("%p\n",int_pa+1); //00000071DCEFFC28 加了四個(gè)字節(jié)
}

? 可以看到在地址+1之前他們的地址都是與*a一樣,進(jìn)行+1之后,char加了一個(gè)字節(jié),short加了兩個(gè)字節(jié),int加了三個(gè)字節(jié),也就是說(shuō),指針類型能決定指針±的時(shí)候可以±多少個(gè)字節(jié),±的字節(jié)由類型的長(zhǎng)度決定。
2.指針解引用
代碼示例
#include <stdio.h>
int main()
{
int a = 0x11223344;
char *char_pa = (char*)&a;
short *short_pa = (short*)&a;
int *int_pa = &a;
printf("%d\n",*char_pa);
printf("%d\n",*short_pa);
printf("%d\n",*int_pa);
}
輸出結(jié)果

明明我們賦值的都是&a,那為什么會(huì)造成三個(gè)類型解引用都不一樣呢?,我們可以用下面代碼測(cè)試一下,看看內(nèi)存發(fā)生了什么。
int main()
{
int a = 0x11223344;
char *char_pa = (char*)&a;
*char_pa = 0;
}
這是在a沒(méi)有改動(dòng)之前

執(zhí)行*char_pa = 0;

a的一個(gè)字節(jié)被置為了0
int main()
{
int a = 0x11223344;
short *short_pa = (short*)&a;
*short_pa = 0;
}
執(zhí)行*short_pa = 0;

a的兩個(gè)字節(jié)被置為了0
int main()
{
int *int_pa = &a;
*int_pa = 0;
return 0;
}
執(zhí)行*int_pa = 0;

a的四個(gè)字節(jié)被置為了0
由此,我們可以得出結(jié)論,指針類型的第二大作用就是,指針類型決定了對(duì)指針解引用時(shí),能訪問(wèn)或修改幾個(gè)字節(jié),這由數(shù)據(jù)類型的大小決定
三.野指針
野指針會(huì)指向一段實(shí)際的內(nèi)存,但是野指針是指指針指向的位置是不可知的(隨機(jī)的,不正確的,沒(méi)有初始化的,沒(méi)有明確限制的),它指向哪里我們不知道,或者說(shuō)它指向的空間已經(jīng)被我們釋放,那么他就是一個(gè)野指針,在程序中,我們必須要避免野指針的出現(xiàn)。
下面是容易出現(xiàn)野指針的場(chǎng)景
1.引起野指針的原因
(1)指針未初始化。
#include <stdio.h>
int main()
{
int *p; //指針沒(méi)有指向明確的地址,那么將是隨即地址,也就是野指針
*p = 1;
}
(2)指針越界訪問(wèn)
#inlcude <stdio.h>
int main()
{
int arr[10] = {0};
int *p = arr;
for(int i=0;i<13;i++)
{
*(p++) = 8; //當(dāng)指針超出arr數(shù)組的范圍時(shí),p就是野指針
}
}
(3)指針指向的空間被釋放
int main()
{
int *p = malloc(10 * sizeof(int));
free(p);
*p = 10; //指針p指向的內(nèi)存空間已經(jīng)被釋放,此時(shí)的p就是野指針
return 0;
}
2.如果避免野指針
(1)指針初始化
(2)小心指針越界
(3)指針指向空間釋放即使置NULL
(4)避免返回局部變量的地址
(5)指針使用之前檢查有效性
int main()
{
int *p = NULL; //使用之前初始化為NULL
if(p!=NULL)
{
*p = 10; //使用之前檢查有效性
}
int *pa = malloc(10 * sizeof(int));
//此處省略n行代碼
free(p);
pa ==NULL; //指針用完之后,及時(shí)置空。
}
到此這篇關(guān)于C語(yǔ)言指針類型與野指針引起的原因的文章就介紹到這了,更多相關(guān)C語(yǔ)言指針類型與野指針內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
數(shù)據(jù)結(jié)構(gòu) 數(shù)組順序存儲(chǔ)詳細(xì)介紹
這篇文章主要介紹了數(shù)據(jù)結(jié)構(gòu) 數(shù)組順序存儲(chǔ)詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下2017-05-05
C語(yǔ)言中的結(jié)構(gòu)體內(nèi)嵌函數(shù)用法
這篇文章主要介紹了C語(yǔ)言中的結(jié)構(gòu)體內(nèi)嵌函數(shù)用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02
C++ 實(shí)現(xiàn)LRU 與 LFU 的緩存算法
設(shè)計(jì)和實(shí)現(xiàn)一個(gè)LRU 緩存機(jī)制。其支持獲取數(shù)據(jù) get 和 寫(xiě)入數(shù)據(jù) put,設(shè)計(jì)并實(shí)現(xiàn)最少訪問(wèn)頻率(LFU)緩存的數(shù)據(jù)結(jié)構(gòu)。LFU的每個(gè)數(shù)據(jù)塊都有一個(gè)引用計(jì)數(shù),所有數(shù)據(jù)塊按照引用計(jì)數(shù)排序,具有相同引用計(jì)數(shù)的數(shù)據(jù)塊則按照時(shí)間進(jìn)行排序。其支持get 和 put,具體了解請(qǐng)看下文2021-09-09
詳解C語(yǔ)言中fseek函數(shù)和ftell函數(shù)的使用方法
這篇文章主要介紹了C語(yǔ)言中fseek函數(shù)和ftell函數(shù)的使用方法,兩個(gè)函數(shù)分別用于設(shè)置和返回文件指針stream的位置,需要的朋友可以參考下2016-03-03
MFC擴(kuò)展DLL中導(dǎo)出類和對(duì)話框的實(shí)現(xiàn)方法
這篇文章主要介紹了MFC擴(kuò)展DLL中導(dǎo)出類和對(duì)話框的實(shí)現(xiàn)方法,詳細(xì)講述了實(shí)現(xiàn)擴(kuò)展DLL中導(dǎo)出類和對(duì)話框的具體步驟與方法,具有不錯(cuò)的實(shí)用價(jià)值,需要的朋友可以參考下2014-10-10
C語(yǔ)言數(shù)組棧實(shí)現(xiàn)模板
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言數(shù)組棧實(shí)現(xiàn)模板,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12
C語(yǔ)言約瑟夫環(huán)的實(shí)現(xiàn)
這篇文章主要介紹了C語(yǔ)言約瑟夫環(huán)的實(shí)現(xiàn)的相關(guān)資料,這里主要是利用數(shù)據(jù)數(shù)據(jù)結(jié)果中循環(huán)鏈表來(lái)實(shí)現(xiàn),需要的朋友可以參考下2017-08-08

