RM新时代网站-首页

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

C語言中的野指針是怎么來的?

Q4MP_gh_c472c21 ? 來源:技術(shù)讓夢想更偉大 ? 作者:李肖遙 ? 2021-06-01 16:43 ? 次閱讀

一、什么是野指針?

指針是C語言的靈魂,同時也是很容易讓人犯錯的重難點,用錯了指針將是一個災難。

指針變量的本質(zhì)是值,這個特殊的值是一個內(nèi)存地址值,而合法的內(nèi)存地址包括定義的變量的地址(棧)、malloc函數(shù)申請堆內(nèi)存返回的地址(但未使用free釋放,是在堆空間動態(tài)申請)

需要注意的是,野指針不是NULL指針,通常NULL指針可以使用if語句來判斷,但是C語言中沒有任何方法用來判斷一個指針是否為野指針!

二、野指針是怎么來的?

通常野指針是因為指針變量中保存的值不是一個合法的內(nèi)存地址或者指向不可用內(nèi)存的指針而造成的。

而且野指針往往會造成內(nèi)存越界、段錯誤等難以找到的問題,下面分幾種情況來說說野指針的由來。

局部指針變量沒有被初始化

//在win10_64位+vs2017

//來源:技術(shù)讓夢想更偉大

//作者:李肖遙

#include 《stdio.h》#include 《string.h》

struct Student

{

char* name;

int number;

};

int main()

{

struct Student s;

strcpy(s.name, “Lixiaoyao”); // OOPS!

s.number = 99;

return 0;

}

局部變量不像全局變量那樣,不賦值會自動初始化為0,指針name指向的內(nèi)存空間地址是隨機的,不能向隨機地址空間寫數(shù)據(jù)。我們在定義局部指針變量時應該初始化為NULL,局部變量則初始化為0

使用已經(jīng)釋放過后的指針

//在win10_64位+vs2017

//來源:技術(shù)讓夢想更偉大

//作者:李肖遙

#include 《stdio.h》#include 《malloc.h》#include 《string.h》

void func(char* p)

{

printf(“%s

”, p);

free(p);

}

int main()

{

char* s = (char*)malloc(5);

strcpy(s, “Lixiaoyao”);//數(shù)組越界

func(s);

printf(“%s

”, s); // OOPS!使用已經(jīng)釋放的指針s

return 0;

}

malloc申請的堆空間釋放后,意味著把這片內(nèi)存歸還到空閑鏈表,其它程序可以使用這片空間,如果其它程序使用了這個空間,可能導致其它程序莫名其妙的被關(guān)閉,所以一定要在釋放過后將指針變量的值賦值為NULL。

指針所指向的變量在指針之前被銷毀

//在win10_64位+vs2017

//來源:技術(shù)讓夢想更偉大

//作者:李肖遙

#include 《stdio.h》

char* func()

{

char p[] = “Lixiaoyao”;

return p;

}

int main()

{

char* s = func();

printf(“%s

”, s); // OOPS!

return 0;

}

func函數(shù)被調(diào)用的時候,棧區(qū)存放了局部數(shù)組p,func返回之后,棧頂指針退出,占用的內(nèi)存已經(jīng)被釋放掉,此時指針s指向一個被釋放掉了棧空間,如果棧空間值被修改了,就不會打印出預期結(jié)果,s就變成了一個野指針,所以我們絕對不要在函數(shù)中返回局部變量和局部數(shù)組的地址。

進行了錯誤指針運算

//在win10_64位+vs2017

//來源:技術(shù)讓夢想更偉大

//作者:李肖遙

#include 《stdio.h》

void main()

{

int a[10] = {1,2,3,4,5,6,7,8,9,10};

int *p;

for (int *p = &a[9];p 》= a;){

*--p = 0;

}

}

程序中在數(shù)組第1個元素a[0]被清除之后,指針p的值還繼續(xù)減下去,而接下去的一次比較運算是用于結(jié)束循環(huán)的。但表達式p》= a(p 》= &a[0])的值是未定義的。

為避免這種情況,一定要確保字符數(shù)組要以‘’結(jié)尾,為防止內(nèi)存越界,自己編寫的內(nèi)存相關(guān)函數(shù)需要指定正確的長度信息

進行了錯誤的強制類型轉(zhuǎn)換

//在win10_64位+vs2017

//來源:技術(shù)讓夢想更偉大

//作者:李肖遙

#include 《stdio.h》#include 《string.h》

int main()

{

int a = 1;

int p = &a;

printf(“%d

”,*((int*)p));

/*

在64位下輸出錯誤

32位下輸出a的值 1

*/

return 0;

}

上面的程序在64位下輸出錯誤,32位下輸出a的值1,在我們寫嵌入式程序的時候,會將int類型的一個數(shù)據(jù)強制轉(zhuǎn)換成一個指針類型用來表示寄存器的地址,這個時候就需要注意了。

怎么避免野指針?

知道了野指針產(chǎn)生的原因,避免方法就出來了,在指針的解引用之前,確保指針指向一個絕對可用的空間。

定義指針時,同時初始化為NULL

在指針解引用之前,先去判斷這個指針是不是Null

指針使用完之后,將其賦值為NULL

在指針使用之前,將其賦值綁定給一個可用地址空間

原文標題:詳解C語言那些“可怕”的野指針

文章出處:【微信公眾號:嵌入式ARM】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

責任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7604

    瀏覽量

    136685
  • 指針
    +關(guān)注

    關(guān)注

    1

    文章

    480

    瀏覽量

    70551

原文標題:詳解C語言那些“可怕”的野指針

文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    C語言中申請的堆內(nèi)存能不能自動釋放

    C語言中申請的堆內(nèi)存能不能自動釋放?每次都要手動 free 太麻煩,也容易忘記。 學過 C++ 的同學,應該首先能想到智能指針。 但是這是C
    的頭像 發(fā)表于 11-27 09:33 ?107次閱讀

    C語言程序設計教程第4版第8講:指針

    C語言指針講解
    發(fā)表于 11-20 14:10 ?0次下載

    C語言指針學習筆記

    本文從底層內(nèi)存分析,徹底讓讀者明白C語言指針的本質(zhì)。
    的頭像 發(fā)表于 11-05 17:40 ?228次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>指針</b>學習筆記

    C語言中的socket編程基礎

    Socket編程簡介 Socket是一種通信機制,允許程序之間進行通信。在C語言中,socket編程是網(wǎng)絡編程的基礎。通過使用socket,程序可以發(fā)送和接收數(shù)據(jù),實現(xiàn)不同計算機之間的通信
    的頭像 發(fā)表于 11-01 16:51 ?301次閱讀

    C語言指針運算符詳解

    C語言中,當你有一個指向數(shù)組中某個元素的指針時,你可以對該指針執(zhí)行某些算術(shù)運算,例如加法或減法。這些運算可以用來遍歷數(shù)組中的元素,如ptr[i]等價于*(ptr + i)。然而,如果
    的頭像 發(fā)表于 10-30 11:16 ?237次閱讀

    C語言指針詳細解析

    可以對數(shù)據(jù)本身,也可以對存儲數(shù)據(jù)的變量地址進行操作。 指針是一個占據(jù)存儲空間的實體在這一段空間起始位置的相對距離值。在C/C++語言中,指針
    發(fā)表于 09-14 10:03

    c語言中從左到右結(jié)合怎么看

    C語言中,操作符的結(jié)合性(Associativity)是指當操作符在表達式中連續(xù)出現(xiàn)時,它們?nèi)绾闻c操作數(shù)結(jié)合的順序。對于大多數(shù)二元操作符(即需要兩個操作數(shù)的操作符),C語言遵循兩種基
    的頭像 發(fā)表于 08-20 11:42 ?843次閱讀

    面試中的高頻問題:指針函數(shù)與函數(shù)指針,你能完美應對嗎?

    一直覺得C語言較其他語言最偉大的地方就是C語言中指針,有些人認為
    的頭像 發(fā)表于 06-22 08:11 ?1703次閱讀
    面試中的高頻問題:<b class='flag-5'>指針</b>函數(shù)與函數(shù)<b class='flag-5'>指針</b>,你能完美應對嗎?

    嵌入式系統(tǒng)中C語言結(jié)構(gòu)體的基礎實現(xiàn)與應用

    C語言中的數(shù)組只能允許程序員定義存儲相同類型數(shù)據(jù)。但是結(jié)構(gòu)是C語言編程中允許您存儲不同數(shù)據(jù)類型的數(shù)據(jù)。
    發(fā)表于 03-12 14:29 ?494次閱讀
    嵌入式系統(tǒng)中<b class='flag-5'>C</b><b class='flag-5'>語言</b>結(jié)構(gòu)體的基礎實現(xiàn)與應用

    C語言中的typedef的應用

    C 語言提供了 typedef 關(guān)鍵字,您可以使用它為類型取一個新的名字。下面的實例為單字節(jié)數(shù)字定義了一個術(shù)語 BYTE。
    發(fā)表于 03-06 11:34 ?380次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言中</b>的typedef的應用

    C語言指針用法

    C語言編程中善用指針可以簡化一些任務的處理,而對于一些任務(比如動態(tài)內(nèi)存分配),必須要有指針才行的。也就是說精通C
    發(fā)表于 03-05 14:22 ?346次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>的<b class='flag-5'>指針</b>用法

    介紹C語言中錯誤處理和異常處理的一些常用的方法和策略

    C語言是一種低級的、靜態(tài)的、結(jié)構(gòu)化的編程語言,它沒有提供像C++或Java等高級語言中的異常處理機制,例如try-catch-finally
    的頭像 發(fā)表于 02-28 14:25 ?617次閱讀

    枚舉有多大?c語言枚舉end的作用是什么?

    可以是整數(shù)或字符,它們被稱為枚舉常量。枚舉常量可以通過枚舉成員引用。 枚舉的大小是取決于它所表示的范圍的大小。在C語言中,枚舉沒有固定的大小,而是根據(jù)其取值范圍確定的。如果枚舉類型
    的頭像 發(fā)表于 01-19 14:19 ?593次閱讀

    如何解決C語言中的“訪問權(quán)限沖突”異常?C語言引發(fā)異常原因分析

    一些措施解決和防止其發(fā)生。本文將詳細介紹C語言中訪問權(quán)限沖突異常的原因以及解決方法。 一、訪問權(quán)限沖突異常的原因分析 訪問權(quán)限沖突異??煞譃閮深悾涸L問私有成員和訪問未定義成員。下面分別分析這兩種異常的原因。 1. 訪問
    的頭像 發(fā)表于 01-12 16:03 ?5627次閱讀

    怎么寫出效率高、思路清晰的C語言程序?

    要用C語言的思維方式進行程序的構(gòu)架構(gòu)建 要有良好的C語言算法基礎,以此實現(xiàn)程序的邏輯構(gòu)架
    的頭像 發(fā)表于 01-02 14:20 ?561次閱讀
    RM新时代网站-首页