關於C語言函數調用壓棧和返回值問題的疑惑

来源:http://www.cnblogs.com/joey-hua/archive/2016/05/28/5537201.html
-Advertisement-
Play Games

按照C編譯器的約定調用函數時壓棧的順序是從右向左,並且返回值是保存在eax寄存器當中。這個命題本該是成立的,下麵用一個小程式來反彙編觀察執行過程: 代碼解釋一下,asm的代碼中movl %%eax, %0的意思是把寄存器eax的值賦值給咱們程式的eax變數當中。但為什麼執行結果卻是: z is 11 ...


按照C編譯器的約定調用函數時壓棧的順序是從右向左,並且返回值是保存在eax寄存器當中。這個命題本該是成立的,下麵用一個小程式來反彙編觀察執行過程:

#include<stdio.h>

int add(int x, int y){
	return x+y;
}

int main(){
	int eax=0;
	int z =0;
	int x =6;
	int y =5;

	z=add(x,y);

	__asm__(
					"movl %%eax, %0"
				:"+b"(eax)
				:"m"(x)
					);

	printf("z is %d\n", z);
	printf("eax is %d\n", eax);
	return 0;
}

代碼解釋一下,asm的代碼中movl %%eax, %0的意思是把寄存器eax的值賦值給咱們程式的eax變數當中。但為什麼執行結果卻是:

z is 11
eax is 0

理論上應該是x和y相加返回的結果才對啊。反彙編一下此exe程式:

上面是main函數

[esp+1ch]對應的是eax,[esp+18h]對應的是z,[esp+10h]對應的是x,[esp+14h]對應的是y。再看下圖

esp自減了20h,說明開闢了20h也就是32位元組的棧空間,再看下圖:

先把[esp+10h]的值也就是x的值賦給eax,再把[esp+14h]的值也就是y的值賦給edx,再分別把它們賦給[esp+4]和[esp]處,註意這裡沒用push指令壓棧,但原理卻是一樣,因為用的是棧指針esp,還需要註意的是因為不是使用push指令,所以不是說誰先執行誰就先壓棧,而是觀察esp指向的位置來確定壓棧的先後順序,因為[esp]指向的是棧低。所以這裡就解釋了先把y壓棧,再把x壓棧,確實是從右向左壓棧

接下來再看add調用:

先取了y的值再取x的值,相加後結果保存在eax里。然後再回到main函數

調用完add後把eax的值賦值給了z,這就說明函數的返回值確實是保存在eax中。但為什麼列印出來的eax卻是0呢。

接著往下看,

首先把程式中eax變數的值賦給了eax寄存器,那當然就是0了。所以現在深入理解了C語言嵌入彙編的執行過程,就算指定了"+b"賦給ebx寄存器,但編譯器還是會先把變數的值賦給eax寄存器,再賦值給ebx,返回也是一樣的原理,如下圖:


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.位元組碼是對電腦可讀的程式。 2.運行工具:java.exe,編譯器:javac.exe。 3.java平臺是一個運行在其它基於硬體的平臺之上的純軟體平臺。 4.java平臺有兩個組件:Java虛擬機(簡稱JVM),Java應用程式編程介面(簡稱Java API)。 5.java程式源代碼文件的 ...
  • #include <stdio.h>int main(){ long a,b,c,d,e; scanf("%ld",&a); d=a; b=0; while(d>2) { d=d/2; b=b+1; } b=b-1; d=2; for(c=1;c<=b;c++) { d=d*2; } b=d*2; ...
  • 運行: ...
  • 本文代碼為原創一個簡陋的管理系統,只做功能的測試。並沒有去完善所有應有的功能,只做了輸入輸出查找。僅供參考! 菜單部分: 重點的管理部分: 主函數部分: ...
  • #include <stdio.h>int main(){ int a,b; scanf("%d",&a); b=0; while(a) { b+=a%10; a/=10; } printf("%d\n",b); return 0;} ...
  • RAD Server是一個應用服務框架平臺,可快速構建和部署應用服務。RAD Server提供自動化的Delphi和C++ REST/ JSON API的 發佈與管理、企業資料庫集成中間件、智能物聯網設備軟體(IoT Edgeware)和一系列通用應用服務如用戶目錄和認證服務,消息推 送服務,室內/ ...
  • 建立Bank類,類中有變數double balance表示存款,Bank類的構造方法能初始化餘額,Bank類中有存款的方法cunKuan(double balance),取款的發方法withDrawal(double dAmount),當取款的數額大於存款時,拋出InsufficientFundsE ...
  • 第一步:獲取apache-tomcat-8.0.35-windows-x64.zip並解壓 第二步:從命令行進入bin目錄 第三步:執行命令:service.bat install,這樣就會將tomcat8.exe文件註冊成系統的服務。 第四步:啟動服務 第五步:在瀏覽器地址欄輸入http://lo ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...