国产精品免费无遮挡无码永久视频-国产高潮视频在线观看-精品久久国产字幕高潮-国产精品99精品无码视亚

C語(yǔ)言的那些小秘密之堆棧

發(fā)布時(shí)間:2016-2-19 09:06    發(fā)布者:designapp
關(guān)鍵詞: C語(yǔ)言 , 堆棧

在講解堆棧之前,我們先要來(lái)說(shuō)說(shuō)其實(shí)我們常說(shuō)的堆棧是兩種數(shù)據(jù)結(jié)構(gòu)。那么什么是堆什么又是棧呢?
棧,是硬件。主要作用表現(xiàn)為一種數(shù)據(jù)結(jié)構(gòu),是只能在某一端插入和刪除的特殊線性表。它按照后進(jìn)先出的原則存儲(chǔ)數(shù)據(jù),先進(jìn)入的數(shù)據(jù)被壓入棧底,最后的數(shù)據(jù)在棧頂,需要讀數(shù)據(jù)的時(shí)候從棧頂開(kāi)始彈出數(shù)據(jù)(最后一個(gè)數(shù)據(jù)被第一個(gè)讀出來(lái))。棧是允許在同一端進(jìn)行插入和刪除操作的特殊線性表。允許進(jìn)行插入和刪除操作的一端稱為棧頂(top),另一端為棧底(bottom);棧底固定,而棧頂浮動(dòng);棧中元素個(gè)數(shù)為零時(shí)稱為空棧。插入一般稱為進(jìn)棧(PUSH),刪除則稱為退棧(POP)。 棧也稱為先進(jìn)后出表。棧可以用來(lái)在函數(shù)調(diào)用的時(shí)候存儲(chǔ)斷點(diǎn),做遞歸時(shí)要用到棧!
以上定義是在經(jīng)典計(jì)算機(jī)科學(xué)中的解釋。
在計(jì)算機(jī)系統(tǒng)中,棧則是一個(gè)具有以上屬性的動(dòng)態(tài)內(nèi)存區(qū)域。程序可以將數(shù)據(jù)壓入棧中,也可以將數(shù)據(jù)從棧頂彈出。在i386機(jī)器中,棧頂由稱為esp的寄存器進(jìn)行定位。壓棧的操作使得棧頂?shù)牡刂窚p小,彈出的操作使得棧頂?shù)牡刂吩龃蟆?br /> 棧在程序的運(yùn)行中有著舉足輕重的作用。最重要的是棧保存了一個(gè)函數(shù)調(diào)用時(shí)所需要的維護(hù)信息,這常常稱之為堆棧幀或者活動(dòng)記錄。堆棧幀一般包含如下幾方面的信息:
1. 函數(shù)的返回地址和參數(shù)
2. 臨時(shí)變量:包括函數(shù)的非靜態(tài)局部變量以及編譯器自動(dòng)生成的其他臨時(shí)變量。
堆,是一種動(dòng)態(tài)存儲(chǔ)結(jié)構(gòu),實(shí)際上就是數(shù)據(jù)段中的自由存儲(chǔ)區(qū),它是C語(yǔ)言中使用的一種名稱,常常用于動(dòng)態(tài)數(shù)據(jù)的存儲(chǔ)分配。堆中存入一數(shù)據(jù),總是以2字節(jié)的整數(shù)倍進(jìn)行分配,地址向增加方向變動(dòng)。堆可以不斷進(jìn)行分配直到?jīng)]有堆空間為止,也可以隨時(shí)進(jìn)行釋放、再分配,不存在次序問(wèn)題。
堆和棧在使用時(shí)相向生長(zhǎng),棧向上生長(zhǎng),即向小地址方向生長(zhǎng),而堆向下增長(zhǎng),即向大地址方向,其間剩余部分是自由空間。使用過(guò)程中要防止增長(zhǎng)過(guò)度而導(dǎo)致覆蓋。
一般的程序我們都是使用小內(nèi)存模式。
明白了堆和棧的概念之后我們來(lái)看一個(gè)面試的c語(yǔ)言題目。代碼要相關(guān)要求如下所示:
#include
using namespace std;
void print()
{
//這里進(jìn)行打印arr數(shù)組,print不準(zhǔn)傳參數(shù)
}
int main()
{
int s=0;
int ss=0;
char *str="fdsafdsafdsafdsafdsafdsafdsa";
char fdsa='f';
char srt[8];
int arr[]={32,43,3,567,987,21,56};//數(shù)值隨即
print();
return 0;
}
剛剛一開(kāi)始看到這個(gè)題目時(shí),你可能有點(diǎn)發(fā)懵,心想可能在不傳遞參數(shù)的情況下打印arr數(shù)組的內(nèi)容,但是看看我們的標(biāo)題就應(yīng)該知道該題跟棧有關(guān)系,在做之前我們先來(lái)回顧幾個(gè)知識(shí)點(diǎn)。
push操作先修改指針,后將信息入棧。
ESP為堆棧指針,棧頂有ESP寄存器來(lái)定位。壓棧的操作使得棧頂?shù)牡刂窚p小,彈出的操作使得棧頂?shù)牡刂吩龃蟆?br /> EBP是32位的BP,EBP是基址指針,EBP與BP的關(guān)系就像AX與AL、AH的關(guān)系一樣。BP為基指針寄存器,用它課直接存取堆棧中的數(shù)據(jù),他的作用在調(diào)用函數(shù)時(shí)保存ESP,使函數(shù)結(jié)束時(shí)可以正確返回。
c的默認(rèn)函數(shù)壓棧操作為:
參數(shù)是從右向左壓棧的,默認(rèn)四字節(jié)對(duì)齊,函數(shù)里面定義的變量是默認(rèn)對(duì)齊方式----變量首地址是自身結(jié)構(gòu)體里邊最大標(biāo)準(zhǔn)數(shù)據(jù)類型字節(jié)的整數(shù)倍。
我們先來(lái)看看上面這段代碼的匯編語(yǔ)句:
//*******************************************start*********************************************//
.file "push.c"
.text
.globl print
.type print, @function
print:
pushl %ebp
movl %esp, %ebp
popl %ebp
ret
.size print, .-print
.section .rodata
.LC0:
.string "fdsafdsafdsafdsafdsafdsafdsa"
.text
.globl main
.type main, @function
main:
pushl %ebp
movl %esp, %ebp
andl $-16, %esp
subl $64, %esp
movl %gs:20, %eax
movl %eax, 60(%esp)
xorl %eax, %eax
movl $0, 44(%esp)
movl $0, 40(%esp)
movl $.LC0, 36(%esp)
movb $102, 51(%esp)
movl $32, 8(%esp)
movl $43, 12(%esp)
movl $3, 16(%esp)
movl $567, 20(%esp)
movl $987, 24(%esp)
movl $21, 28(%esp)
movl $56, 32(%esp)
call print
movl $0, %eax
movl 60(%esp), %edx
xorl %gs:20, %edx
je .L3
call __stack_chk_fail
.L3:
leave
ret
.size main, .-main
.ident "GCC: (Ubuntu/Linaro 4.5.2-8ubuntu4) 4.5.2"
.section .note.GNU-stack,"",@progbits
//*******************************************end*********************************************//
看看上面的匯編代碼,我們的重點(diǎn)放在紅色字體部分,在函數(shù)的開(kāi)頭部分都有這么兩行代碼: pushl %ebp --------------------------->保存上一個(gè)函數(shù)的棧底
movl %esp, %ebp --------------------------->用來(lái)保存當(dāng)前堆棧指針的值
ebp存放當(dāng)前函數(shù)棧低的地址,就是說(shuō)ebp可以看做一個(gè)指針,指向棧頂,而其實(shí)棧頂存放的數(shù)據(jù)就是上一個(gè)函數(shù)的ebp的值,即就是main函數(shù)的棧底。
                                
               
明白了上面的內(nèi)容,那么我們就可以實(shí)現(xiàn)題目的要求了。代碼如下所示:
#include
using namespace std;
void print()
{
//這里進(jìn)行排序,print不準(zhǔn)傳參數(shù) unsigned int _ebp; __asm{
mov _ebp,ebp
}
int *p=(int *)(*(int *)_ebp-4-4-4-4-8-7*4);
for(int i=0;i


為了強(qiáng)調(diào)默認(rèn)的字節(jié)對(duì)齊概念,我們?cè)賮?lái)修改下代碼得到的運(yùn)行結(jié)果可以上面得做一個(gè)比較。
代碼如下,紅色部分為修改代碼。
#include
using namespace std;
void print()
{
//這里進(jìn)行排序,print不準(zhǔn)傳參數(shù)
unsigned int _ebp;
__asm{
mov _ebp,ebp
}
int *p=(int *)(*(int *)_ebp-4-4-4-4-8-7*4);
for(int i=0;i


如果我們修改了 char srt[6];之后去把int *p=(int *)(*(int *)_ebp-4-4-4-4-8-7*4);修改為int *p=(int *)(*(int *)_ebp-4-4-4-4-6-7*4);,注意紅色部分的對(duì)比,運(yùn)行結(jié)果就變?yōu)榱耍?br />

顯然對(duì)比可知運(yùn)行結(jié)果出錯(cuò)了。在此多次一舉的給出對(duì)比無(wú)非是為了大家能夠?qū)ψ止?jié)的對(duì)齊方式加以重視。當(dāng)然以上內(nèi)容難免有錯(cuò),畢竟c語(yǔ)言博大精深,如果有不正確的地方,請(qǐng)糾正。
本文地址:http://www.4huy16.com/thread-160931-1-1.html     【打印本頁(yè)】

本站部分文章為轉(zhuǎn)載或網(wǎng)友發(fā)布,目的在于傳遞和分享信息,并不代表本網(wǎng)贊同其觀點(diǎn)和對(duì)其真實(shí)性負(fù)責(zé);文章版權(quán)歸原作者及原出處所有,如涉及作品內(nèi)容、版權(quán)和其它問(wèn)題,我們將根據(jù)著作權(quán)人的要求,第一時(shí)間更正或刪除。
您需要登錄后才可以發(fā)表評(píng)論 登錄 | 立即注冊(cè)

廠商推薦

  • Microchip視頻專區(qū)
  • Microchip第22屆中國(guó)技術(shù)精英年會(huì)——采訪篇
  • 常見(jiàn)深度學(xué)習(xí)模型介紹及應(yīng)用培訓(xùn)教程
  • 技術(shù)熱潮席卷三城,2025 Microchip中國(guó)技術(shù)精英年會(huì)圓滿收官!
  • Microchip第22屆中國(guó)技術(shù)精英年會(huì)上海首站開(kāi)幕
  • 貿(mào)澤電子(Mouser)專區(qū)

相關(guān)視頻

關(guān)于我們  -  服務(wù)條款  -  使用指南  -  站點(diǎn)地圖  -  友情鏈接  -  聯(lián)系我們
電子工程網(wǎng) © 版權(quán)所有   京ICP備16069177號(hào) | 京公網(wǎng)安備11010502021702
快速回復(fù) 返回頂部 返回列表