嵌入式 C 語言入門:函數封裝與參數傳遞學習筆記 —— 從定義到內存機制

前言

大家好,這里是 Hello_Embed。在前一篇筆記中,我們用循環實現了 LED 閃爍,其中重復使用了兩段幾乎一樣的延時代碼:

for(i = 0; i < 100000000; i++);  // 延時

這種重復不僅讓代碼冗余,還不利于后續修改(比如想調整延時時間,需要改兩處)。有沒有辦法把這段延時 “打包” 成一個可復用的模塊?這就需要用到 C 語言中的函數。本文將從函數的定義、聲明講起,結合實例說明參數傳遞的兩種方式(值傳遞與地址傳遞),以及函數在嵌入式開發中的實用價值。

📦 函數的基本概念:封裝重復代碼

函數的核心作用是 “封裝重復邏輯”,讓代碼更簡潔、易維護。以延時功能為例,我們可以把重復的 for 循環封裝成一個函數,需要時直接調用。

函數的定義與聲明

函數的定義格式為:

返回類型 函數名(參數列表) {// 函數體:實現具體功能
}
  • 返回類型:表示函數執行后返回的數據類型(若無需返回值,用void);
  • 參數列表:函數接收的輸入(若無需參數,用void);
  • 函數體:實現功能的代碼塊。
無參數、無返回值的函數

將延時功能封裝成無參數函數

void delay(void)  // 第一個void:無返回值;第二個void:無參數
{int i;for(i = 0; i < 100000000; i++);  // 延時邏輯
}

調用時直接寫delay();即可,替代原來的 for 循環。這樣修改延時時間只需改函數內部,無需到處找重復代碼。如果此函數涉及公司機密,還可以將函數封裝為庫,別人使用直接調用庫就可以了。

帶參數的函數

如果想靈活控制延時時長,可以給函數添加參數:

void delay(int cnt)  // 參數cnt:控制循環次數
{int i;for(i = 0; i < cnt; i++);  // 用參數cnt替代固定值
}

調用時通過delay(100000000);指定延時長度,甚至可以根據需求動態調整(比如delay(50000000);實現更短的延時)。

帶返回值的函數

有的時候我們可能會遇到有著兩個不同參數的函數,如下:

int add(int a, int b)
{int sum = a + b;return sum;
}

此時函數就有返回值sum,所以add的返回類型是int。這段代碼還可以將sum省去,直接返回a + b。下面我們來實戰演練一遍:

#include <stdio.h>
int add(int a, int b) { return a + b; }int main()
{int a1 = 10, b1 = 20;// 方式1:將返回值賦給變量,再使用int c1 = add(a1, b1);printf("The sum of %d and %d is %d\n", a1, b1, c1);// 方式2:直接在表達式中使用返回值printf("The sum of %d and %d is %d\n", a1, b1, add(a1, b1));return 0;
}

運行結果如圖:
請添加圖片描述
可以看到我們有兩種方法輸出兩數之和:一種是將函數值賦給定義好的c1,再打印c1;另一種是直接打印函數返回的兩數之和的值,這種使用方式也是合規的。

🔄 參數傳遞的兩種方式

函數參數的傳遞方式決定了函數內部能否修改外部變量,這在嵌入式開發中尤為重要(比如通過函數修改硬件寄存器的值)。

值傳遞:僅傳遞變量的 “副本”

先看一個例子:

#include <stdio.h>
// 嘗試修改參數a的值
void change_val(int a)
{printf("The value of a 1 is %d\n", a);  // 打印傳入的值a = 200;  // 修改參數aprintf("The value of a 2 is %d\n", a);  // 打印修改后的值
}int main()
{int a = 100;change_val(a);  // 傳入a的值printf("The value of a 3 is %d\n", a);  // 打印main中的areturn 0;
}

運行結果如圖,函數內部的a被改成了 200,但 main 中的a仍為 100:
請添加圖片描述
我們用反證法來理解這段代碼的結果:假設main中的achange_val中的a用同一塊內存,順序執行時,先是a = 100,接著調用函數打印一次,此時打印出a為 100;接著a = 200再打印,a應為 200;函數執行完畢后在主程序里再次打印,理論上a應為 200,但現實結果卻是 100。這說明:兩個變量a所占內存不同
為什么會這樣?這涉及到 “全局變量”“局部變量” 和 “棧” 的概念:

  • 在函數之外定義的變量稱為全局變量,在函數之內定義的變量稱為局部變量。
  • 程序運行時,內存中會劃出一塊區域稱為 “棧”,還有一個硬件寄存器叫SP(棧指針)。當運行到main函數時,會為其分配空間,SP會從原來的位置往下移動,SP原來位置與現在位置之間的區域就是main的棧,里面存放main的局部變量。
  • 同樣,change_val函數被調用時,也會有自己的棧來存放它的局部變量。
    所以,代碼中同名的a實際上是不同的變量:main中的amain的棧里,change_val中的a在它自己的棧里。這段代碼所側重的恰恰是參數之間的傳遞:我們將maina的值 100 傳入change_val中,所以第一次打印 100;函數內修改的是自己棧里的a,所以第二次打印 200;最后打印的是main函數中定義的a,所以還是 100。
地址傳遞:通過指針修改原變量

那如果我就是想通過change_val中的a改變main中的a的值呢?當然有辦法解決,這個時候就需要請出指針了:

#include <stdio.h>
// 用指針接收地址,修改原變量
void change_val(int *a)  // 參數為int型指針
{printf("The value of a 1 is %d\n", *a);  // 訪問指針指向的變量*a = 200;  // 修改指針指向的變量(即main中的a)printf("The value of a 2 is %d\n", *a);
}int main()
{int a = 100;change_val(&a);  // 傳入a的地址printf("The value of a 3 is %d\n", a);  // main中的a被修改return 0;
}

運行結果如圖,main 中的a成功被改為 200:
請添加圖片描述
原理:指針a存儲了 main 中a的地址,通過*a可以直接操作原變量的內存,因此修改會生效。這種方式在嵌入式中常用(比如通過函數修改寄存器的值,關于指針的更多知識可參考之前的筆記)。當change_val函數執行完畢,SP會重新指回main函數,代表一個函數的結束。

結尾

通過這篇筆記,我們系統學習了函數的基本概念:從無參數、無返回值的簡單封裝(如延時函數),到帶參數、帶返回值的靈活應用(如加法函數),再到參數傳遞的兩種方式 —— 值傳遞(僅傳副本,不影響原變量)和地址傳遞(用指針修改原變量)。我們還理解了局部變量在棧中的存儲特點,這是搞懂參數傳遞機制的關鍵。
函數是 C 語言模塊化編程的核心,在嵌入式開發中,我們可以將硬件初始化、數據處理等邏輯封裝成函數,讓代碼更清晰、易復用。下一篇筆記,我們會進一步學習函數的遞歸。Hello_Embed 會繼續陪伴大家,一步步掌握嵌入式 C 語言的精髓,敬請期待~

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/91412.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/91412.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/91412.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

第一個大語言模型的微調

模型推理 現在,我們的模型應該能夠針對輸入的任何短句生成類似尤達大師風格的句子作為回應。 該模型要求其輸入格式規范。我們需要構建一個 “消息” 列表 —— 在這個案例中,就是來自用戶的消息 —— 并通過提示表明輪到模型進行輸出,以促使其做出回答。 add_generation…

Linux內核驅動開發核心問題全解

&#x1f4d6; 推薦閱讀&#xff1a;《Yocto項目實戰教程:高效定制嵌入式Linux系統》 &#x1f3a5; 更多學習視頻請關注 B 站&#xff1a;嵌入式Jerry Linux內核驅動開發核心問題全解 本文系統梳理了 Linux 驅動開發、內核同步、中斷處理、內存管理、進程通信、系統啟動等典型…

【C++篇】C++11入門:踏入C++新世界的大門

文章目錄C11簡介列表初始化1. {}初始化2. initializer_list容器initializer_list的使用場景聲明1. auto2. decltype3. nullptrSTL中的變化1. 新容器array容器forward_list容器unordered_map和unordered_set容器2. 新接口C11簡介 C98/03&#xff1a;在2003年C標準委員會曾經提交…

Java 日期時間處理:分類、用途與性能分析

Java提供了多種日期時間處理API&#xff0c;隨著版本演進不斷改進。以下是主要日期時間類的分類、用途和性能分析&#xff1a;一、Java日期時間API分類1. 傳統日期時間API (Java 1.0/1.1)java.util.Date - 表示特定的瞬間&#xff0c;精確到毫秒java.util.Calendar - 抽象類&am…

[Linux]學習筆記系列 --GCC

文章目錄屬性__cleanup__attribute_malloc__ 用于標記函數返回一個新分配的內存塊__attribute_alloc_size__ 用于指定分配的內存大小__attribute__((const)) 標記為純函數(pure function)__attribute__((__externally_visible__)) 使其在編譯器優化過程中保持對外部模塊的可見性…

【龍澤科技】汽車維護與底盤拆裝檢修仿真教學軟件【風光580】

產品簡介汽車維護與底盤拆裝檢修仿真教學軟件是依托《全國職業院校技能大賽》“汽車維修”賽項中“汽車維護與底盤拆裝檢修模塊”競賽模塊&#xff0c;自主開發的一款仿真教學軟件。軟件采用仿真仿真技術模擬實際汽車維修工的崗位技能操作流程&#xff0c;操作內容主要包括&…

Spring之【循環引用】

目錄前置知識SingletonBeanRegistryDefaultSingletonBeanRegistrySpring中處理循環引用的流程分析定義兩個具有循環引用特點的Bean執行A的實例化執行A的屬性填充(執行過程中發現A依賴B&#xff0c;就去執行B的實例化邏輯)執行B的實例化執行B的屬性填充執行B的初始化執行A的屬性…

LRU緩存淘汰算法的詳細介紹與具體實現

LRU&#xff08;Least Recently Used&#xff0c;最近最少使用&#xff09;是一種基于時間局部性原理的緩存淘汰策略。其核心思想是&#xff1a;最近被訪問的數據在未來更可能被再次使用&#xff0c;而最久未被訪問的數據應優先被淘汰&#xff0c;從而在有限的緩存空間內保留高…

JS-第十九天-事件(一)

一、事件基礎概念1.1 事件三要素事件源&#xff1a;觸發事件的元素事件類型&#xff1a;事件的種類&#xff08;如click、mouseover等&#xff09;事件處理程序&#xff1a;響應事件的函數1.2 事件流機制事件傳播分為三個階段&#xff1a;捕獲階段&#xff1a;事件從頂層開始&a…

Matplotlib(三)- 圖表輔助元素

文章目錄一、圖表輔助元素簡介二、坐標軸的標簽、刻度范圍和刻度標簽1. 坐標軸標簽1.1 x軸標簽1.2 y軸標簽1.3 示例&#xff1a;繪制天氣氣溫折線圖2. 刻度范圍和刻度標簽2.1 刻度范圍2.1.1 x軸刻度范圍2.1.2 y軸刻度范圍2.2 刻度標簽2.2.1 x軸刻度標簽2.2.2 y軸刻度標簽2.3 示…

【Linux基礎知識系列】第七十八篇 - 初識Nmap:網絡掃描工具

在網絡管理和安全領域&#xff0c;網絡掃描是一個不可或缺的工具。它可以幫助網絡管理員了解網絡中的設備、服務以及潛在的安全漏洞。Nmap&#xff08;Network Mapper&#xff09;是一個功能強大的開源網絡掃描工具&#xff0c;它能夠快速發現網絡中的主機、端口和服務&#xf…

EasyGBS的兩種錄像回看

EasyGBS 支持兩種錄像回看&#xff0c;即“平臺端”的錄像回看和“設備端”的錄像回看。本期我們來介紹兩者的區別和使用方法。一、平臺端錄像1、什么是平臺端錄像平臺端錄像是指由 EasyGBS 平臺直接錄制并存儲。2、配置平臺端錄像進入平臺&#xff0c;依次點擊【錄像回放】→【…

大模型學習思路推薦!

為進一步貫徹落實中共中央印發《關于深化人才發展體制機制改革的意見》和國務院印發《關于“十四五”數字經濟發展規劃》等有關工作的部署要求&#xff0c;深入實施人才強國戰略和創新驅動發展戰略&#xff0c;加強全國數字化人才隊伍建設&#xff0c;持續推進人工智能從業人員…

數據庫連接池性能優化實戰

背景我們公司正在處于某個項目的維護階段&#xff0c;領導對資源告警比較重視&#xff0c;服務器資源告警的就不說了&#xff0c;運維同學每隔一小時都會檢測線上環境的應用服務信息&#xff0c;例如&#xff1a;網關日志響應時間告警/nginx日志接口響應時間告警/日志關鍵字異常…

Excel常用函數大全,非常實用

一、數學與統計函數1. SUM作用&#xff1a;求和SUM(number1, [number2], ...)SUM(A1:A10) ? 計算A1到A10單元格的總和注意&#xff1a;自動忽略文本和空單元格2. AVERAGE作用&#xff1a;計算平均值AVERAGE(number1, [number2], ...)AVERAGE(B2:B20) ? 計算B列20個數據的平均…

性能優化(一):時間分片(Time Slicing):讓你的應用在高負載下“永不卡頓”的秘密

性能優化(一)&#xff1a;時間分片&#xff08;Time Slicing&#xff09;&#xff1a;讓你的應用在高負載下“永不卡頓”的秘密 引子&#xff1a;那張讓你瀏覽器崩潰的“無限列表” 想象一個場景&#xff1a;你需要渲染一個包含一萬個項目的列表。在我們的“看不見”的應用中&a…

《C++》STL--list容器詳解

在 C 標準模板庫(STL)中&#xff0c;list 是一個非常重要的序列容器&#xff0c;它實現了雙向鏈表的數據結構。與 vector 和 deque 不同&#xff0c;list 提供了高效的插入和刪除操作&#xff0c;特別是在任意位置。本文將深入探討 list 容器的特性、使用方法以及常見操作。 文…

Day 28:類的定義和方法

DAY 28 類的定義和方法 知識點學習 1. 類的定義 在Python中&#xff0c;類是創建對象的模板。使用class關鍵字來定義一個類。類名通常采用首字母大寫的命名方式&#xff08;PascalCase&#xff09;。 # 最簡單的類定義 class MyClass:pass # 使用pass占位符類的定義就像是…

OSPF綜合實驗報告冊

一、實驗拓撲二、實驗要求1、R4為ISP&#xff0c;其上只配置IP地址&#xff1b;R4與其他所直連設備間均使用公有IP&#xff1b; 2、R3-R5、R6、R7為MGRE環境&#xff0c;R3為中心站點&#xff1b; 3、整個OSPF環境IP基于172.16.0.0/16劃分&#xff1b;除了R12有兩個環回&#x…

網絡層6——內部網關協議RIP、OSPF(重點)

目錄 一、基本概念 1、理想的路由算法應具備的特點 2、分層次的路由選擇協議 二、內部網關協議RIP 1、特點 2、路由交換信息 3、距離向量算法 4、壞消息傳送慢問題 5、RIP報文格式 三、內部網關協議OSPF 1、特點 2、其他特點 3、自治系統區域劃分 4、OSPF的5中分…