欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺析C語言編程中的數(shù)組越界問題

 更新時(shí)間:2015年11月17日 17:22:09   作者:wind19  
這篇文章主要介紹了淺析C語言編程中的數(shù)組越界問題,通過內(nèi)存空間來討論其導(dǎo)致的程序崩潰問題,需要的朋友可以參考下

因?yàn)镃語言不檢查數(shù)組越界,而數(shù)組又是我們經(jīng)常用的數(shù)據(jù)結(jié)構(gòu)之一,所以程序中經(jīng)常會遇到數(shù)組越界的情況,并且后果輕者讀寫數(shù)據(jù)不對,重者程序crash。下面我們來分析一下數(shù)組越界的情況:
1) 堆中的數(shù)組越界

因?yàn)槎咽俏覀冏约悍峙涞?,如果越界,那么會把堆中其他空間的數(shù)據(jù)給寫掉,或讀取了其他空間的數(shù)據(jù),這樣就會導(dǎo)致其他變量的數(shù)據(jù)變得不對,如果是一個(gè)指針的話,那么有可能會引起crash

2) 棧中的數(shù)組越界

因?yàn)闂J窍蛳略鲩L的,在進(jìn)入一個(gè)函數(shù)之前,會先把參數(shù)和下一步要執(zhí)行的指令地址(通過call實(shí)現(xiàn))壓棧,在函數(shù)的入口會把ebp壓棧,并把esp賦值給ebp,在函數(shù)返回的時(shí)候,將ebp值賦給esp,pop先前棧內(nèi)的上級函數(shù)棧的基地址給ebp,恢復(fù)原?;?,然后把調(diào)用函數(shù)之前的壓入棧的指令地址pop出來(通過ret實(shí)現(xiàn))。

棧是由高往低增長的,而數(shù)組的存儲是由低位往高位存的 ,如果越界的話,會把當(dāng)前函數(shù)的ebp和下一跳的指令地址覆蓋掉,如果覆蓋了當(dāng)前函數(shù)的ebp,那么在恢復(fù)的時(shí)候esp就不能指向正確的地方,從而導(dǎo)致未可知的情況,如果下一跳的地址也被覆蓋掉,那么肯定會導(dǎo)致crash。

-------------------------

 壓入的參數(shù)和函數(shù)指針

-------------------------

                        aa[4]

                        aa[3]

合法的數(shù)組空間   aa[2]

                        aa[1]

                        aa[0]

-------------------------

 

###sta.c###

#include <stdio.h>

void f(int ai)
{
int aa[5]={1,2,3};
int i = 1;
for (i=0;i<10;i++)
 aa[i]=i;
printf("f()/n");
}

void main()
{
f(3);
printf("ok/n");
}

 

 

###sta.s###

     .file  "sta.c"                 ;說明匯編的源程序
    .section    .rodata           ;說明以下是只讀數(shù)據(jù)區(qū)
.LC0:
    .string "f()"                  ;"f()" 的類型是string,地址為LC0
    .text                       ;代碼段開始
.globl f                         ;f為全局可訪問
    .type  f, @function            ; f是函數(shù)
f:
    pushl  %ebp
    movl  %esp, %ebp
    subl  $40, %esp
    movl  $0, -24(%ebp)
    movl  $0, -20(%ebp)
    movl  $0, -16(%ebp)
    movl  $0, -12(%ebp)
    movl  $0, -8(%ebp)
    movl  $1, -24(%ebp)
    movl  $2, -20(%ebp)
    movl  $3, -16(%ebp)
    movl  $1, -4(%ebp)
    movl  $0, -4(%ebp)
    jmp   .L2
.L3:
    movl  -4(%ebp), %edx
    movl  -4(%ebp), %eax
    movl  %eax, -24(%ebp,%edx,4)
    addl  $1, -4(%ebp)
.L2:
    cmpl  $9, -4(%ebp)
    jle   .L3
    movl  $.LC0, (%esp)
    call  puts
    leave
    ret
    .size  f, .-f                     ;用以計(jì)算函數(shù)f的大小
    .section    .rodata
.LC1:
    .string "ok"
    .text
.globl main
    .type  main, @function
main:
    leal  4(%esp), %ecx
    andl  $-16, %esp
    pushl  -4(%ecx)
    pushl  %ebp
    movl  %esp, %ebp
    pushl  %ecx
    subl  $4, %esp
    movl  $3, (%esp)
    call  f
    movl  $.LC1, (%esp)
    call  puts
    addl  $4, %esp
    popl  %ecx
    popl  %ebp
    leal  -4(%ecx), %esp
    ret
    .size  main, .-main
    .ident "GCC: (GNU) 4.1.2 20070115 (SUSE Linux)"        ;說明是用什么工具編譯的
    .section    .note.GNU-stack,"",@progbits

 

從main函數(shù)開始壓入f函數(shù)的參數(shù)開始,堆棧的調(diào)用情況如下

20151117171954436.gif (341×267)

圖1  壓入?yún)?shù)

20151117172018789.gif (315×241)

圖二  通過call 命令壓入下一跳地址 IP

20151117172103760.gif (319×250)

圖三  函數(shù)f 通過pushl   %ebp 把 ebp保存起來

20151117172121305.gif (414×210)

圖四  函數(shù) f 通過movl    %esp, %ebp讓ebp指向esp,這樣esp就可以進(jìn)行修改,在函數(shù)返回的時(shí)候用ebp的值對esp進(jìn)行恢復(fù)

20151117172136004.gif (407×540)

圖五  函數(shù) f 通過subl    $40, %esp 給函數(shù)的局部變量預(yù)留空間

20151117172152902.gif (418×531)

圖六  int數(shù)組 aa[5]占用了20個(gè)字節(jié)的空間,然后 int i占用了4個(gè)字節(jié)的空間(緊鄰著之前壓入棧的%ebp)

故,如果aa[5]進(jìn)行賦值,則會把 i 的值覆蓋掉,

如果對aa[6]進(jìn)行賦值,則會把 棧中的 %ebp 覆蓋掉,那么在函數(shù) f 返回的時(shí)候則不能對ebp進(jìn)行恢復(fù),即main函數(shù)的ebp變成了我們覆蓋掉的值,程序不知道會發(fā)生什么事情,但因?yàn)槲覀兊某绦蚪酉聛頉]有調(diào)用棧中的內(nèi)容,故還是可以運(yùn)行的。

如果對aa[7]進(jìn)行賦值,則會把棧中的 %IP 覆蓋掉,在函數(shù) f 返回的時(shí)候就不能正確地找到下一跳的地址,會crash;

相關(guān)文章

  • C語言解線性方程的四種方法

    C語言解線性方程的四種方法

    這篇文章主要介紹了C語言解線性方程的四種方法,大家參考使用,學(xué)習(xí)線性代數(shù)的同學(xué)一定能用到
    2013-11-11
  • vscode刷acm、leetcode的題目

    vscode刷acm、leetcode的題目

    vscode是一款越來越受碼農(nóng)們喜愛的軟件,大多數(shù)人學(xué)習(xí)編程繞不開的一部分就是算法,很多人都喜歡刷LeetCode的題目,本文就來介紹一下
    2021-06-06
  • 利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法示例

    利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法示例

    百馬百擔(dān)是道經(jīng)典的算法題,下面這篇文章主要給大家介紹了利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-12-12
  • C++ 中回文數(shù)判斷簡單實(shí)例

    C++ 中回文數(shù)判斷簡單實(shí)例

    這篇文章主要介紹了C++ 中回文數(shù)判斷簡單實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2017-05-05
  • C語言實(shí)現(xiàn)文件讀寫

    C語言實(shí)現(xiàn)文件讀寫

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)文件讀寫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • VS2019+Opencv4.0+Win10配置詳解

    VS2019+Opencv4.0+Win10配置詳解

    這篇文章主要介紹了VS2019+Opencv4.0+Win10配置詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • C++ push方法與push_back方法的使用與區(qū)別

    C++ push方法與push_back方法的使用與區(qū)別

    這篇文章主要介紹了C++ push方法與push_back方法的使用與區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-12-12
  • 弦圖ZOJ 1015 Fishing Net 判定方法

    弦圖ZOJ 1015 Fishing Net 判定方法

    弦圖,算法完全按照CDQ的PPT上給的最大勢算法(MCS)完美消除序列..需要的朋友可以參考下
    2012-11-11
  • C++中回調(diào)函數(shù)及函數(shù)指針的實(shí)例詳解

    C++中回調(diào)函數(shù)及函數(shù)指針的實(shí)例詳解

    這篇文章主要介紹了C++中回調(diào)函數(shù)及函數(shù)指針的實(shí)例詳解的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下
    2017-10-10
  • 詳談C++的內(nèi)存泄漏問題

    詳談C++的內(nèi)存泄漏問題

    下面小編就為大家?guī)硪黄斦凜++的內(nèi)存泄漏問題。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-05-05

最新評論