吴忠躺衫网络科技有限公司

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

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

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

在MCU編程中局部變量賦初始值的重要性

jf_pJlTbmA9 ? 來(lái)源:瑞薩MCU小百科 ? 作者:瑞薩MCU小百科 ? 2023-10-16 18:29 ? 次閱讀

C語(yǔ)言編程過(guò)程中,由于計(jì)算需要,會(huì)使用各種各樣的變量,用于給需要訪問(wèn)的地址取個(gè)名稱,方便編程中使用,代碼維護(hù)者也容易理解。

這里先給大家分享一個(gè)案例,讓大家意識(shí)到變量賦初始值的重要性。

某用戶在基于瑞薩MCU:RA6T2做開(kāi)發(fā)時(shí),發(fā)現(xiàn)一個(gè)問(wèn)題,MCU發(fā)出的CAN數(shù)據(jù)幀總是莫名其妙的出錯(cuò),比如應(yīng)用中明明只使用了CAN的擴(kuò)展幀,但是使用捕捉工具總是能捕捉到遠(yuǎn)程幀,出現(xiàn)遠(yuǎn)程幀的情況毫無(wú)規(guī)律可言,有時(shí)添加一個(gè)定時(shí)器中斷,該現(xiàn)象就不會(huì)出現(xiàn)了,有時(shí)修改了代碼里某處跟CAN沒(méi)有任何關(guān)系的代碼,該問(wèn)題又會(huì)出現(xiàn),過(guò)了兩周時(shí)間調(diào)試無(wú)果。在介入Debug時(shí)發(fā)現(xiàn),他使用的是CAN擴(kuò)展幀,擴(kuò)展幀使用29位ID標(biāo)識(shí)符,而且對(duì)ID區(qū)數(shù)據(jù)定義了一個(gè)如下結(jié)構(gòu)體:

wKgaomUD5XKAKA7FAAAHACC9mOU044.png

他在需要發(fā)送CAN幀時(shí),申請(qǐng)一個(gè)如上結(jié)構(gòu)體的臨時(shí)變量can_id,在把can_id.id賦值后,再把該變量的地址傳遞給CAN的發(fā)送函數(shù),在發(fā)送函數(shù)里使用如下語(yǔ)句把id的數(shù)據(jù)寫(xiě)入CAN的發(fā)送消息緩沖寄存器

wKgaomUD5XOAApX2AAAHaPYEckQ954.png

如下圖,其中第30位的0表示數(shù)據(jù)幀,并不是遠(yuǎn)程幀,31位的1表示擴(kuò)展幀。

wKgZomUD5XWAca2cAADiCZqaiwY457.png

用戶是把can_id的所有數(shù)據(jù)賦值給了CFDTMID0寄存器,假如can_id.dummy中第二個(gè)位是1,會(huì)有什么后果呢?CFDTMID0.TMRTR=1,即CAN會(huì)發(fā)送遠(yuǎn)程幀。

用戶又問(wèn):我沒(méi)有給dummy賦值啊,為什么dummy的第二個(gè)位會(huì)變成1呢?這就是問(wèn)題所在了,就是因?yàn)樗麤](méi)有給can_id.dummy賦值,所以can_id.dummy有可能為任意的值。下面詳細(xì)分析一下,為什么這個(gè)局部變量的值會(huì)隨意變化。

大家知道,變量根據(jù)存儲(chǔ)類(lèi)型和用途,一般可以分成:全局變量和局部變量。全局變量,就是指分配了固定地址的變量,全局變量可以在整個(gè)代碼范圍內(nèi)使用。我們?cè)谏暾?qǐng)全局變量時(shí),有時(shí)對(duì)它賦一個(gè)初始值,也時(shí)也不會(huì)賦初始值,在代碼上可能看不出有什么區(qū)別,但是編譯器在編譯程序時(shí),是區(qū)別對(duì)待他們的。對(duì)于有初始化的變量,編譯器還需要在Code Flash里(代碼存儲(chǔ)區(qū))分配一段空間,把變量的初始值全部存儲(chǔ)在該區(qū)域里,并且在MCU的啟動(dòng)代碼里插入一段程序,把這些Code Flash區(qū)的初始值拷貝到變量對(duì)應(yīng)的RAM地址中。假如上面的can_id是全局變量,并且申明變量的同時(shí)并按下圖賦初始值:

wKgaomUD5XeAaBPmAAAqgT5kmag262.jpg

這時(shí)can_id.dummy=0,如果代碼中用戶沒(méi)有再賦值,它的值也不會(huì)變化,這樣就不會(huì)發(fā)生用戶的那個(gè)遠(yuǎn)程幀的問(wèn)題了。對(duì)于沒(méi)有賦初始值的全局變量,編譯器只是分配RAM的地址,并不會(huì)修改RAM地址里的數(shù)據(jù),那么這個(gè)變量的值就會(huì)依賴于MCU啟動(dòng)時(shí)RAM里的值了。為了避免未賦值的全局變量出現(xiàn)上述的問(wèn)題,我們一般會(huì)在MCU啟動(dòng)代碼里插入未賦初始值全局變量的清零操作,相當(dāng)于做了一個(gè)未賦初始值的全局變量的初始化賦值操作。

像上面的案例,can_id申請(qǐng)的是局部變量,這又是什么情況呢?

因?yàn)镸CU的RAM資源有限,為了最大限度的利用RAM,MCU會(huì)提前分配一塊RAM區(qū)域,叫堆棧區(qū),這塊區(qū)域大家共用,對(duì)于只需要在某個(gè)函數(shù)內(nèi)使用的變量,引入了局部變量概念。在開(kāi)始執(zhí)行該函數(shù)時(shí),才從堆棧里分配地址給局部變量使用,函數(shù)執(zhí)行結(jié)束后,該變量占用的RAM區(qū)域被堆棧回收,當(dāng)下次再調(diào)用該函數(shù),再重新分配RAM。因此對(duì)于局部變量,每次申請(qǐng)到的地址是不同的,該地址很可能是其它函數(shù)使用過(guò)并改寫(xiě)數(shù)據(jù)了的,因此每次函數(shù)調(diào)用時(shí)can_id.dummy的數(shù)據(jù)是不確定的。因?yàn)槎褩^(qū)里的數(shù)據(jù)是被反復(fù)利用的,即使MCU的初始化代碼對(duì)堆棧區(qū)域做清零處理,也是沒(méi)有意義的。

由此看來(lái),局部變量在申請(qǐng)的時(shí)候賦一個(gè)初始值,是非常有必要的。雖然有時(shí)候賦初始值沒(méi)有用,但是出現(xiàn)問(wèn)題時(shí)常常是致命的,而且也是非常難以定位的,你可能覺(jué)得我的代碼里后面肯定會(huì)賦值的,但是后面維護(hù)該項(xiàng)目的其他工程師并不一定意識(shí)到這一點(diǎn)。像類(lèi)似上面的案例,我在其他用戶當(dāng)中也是經(jīng)常見(jiàn)到的。因此軟件工程師在編程的時(shí)候,一定要養(yǎng)成局部變量賦初始值的習(xí)慣。

來(lái)源:瑞薩MCU小百科
免責(zé)聲明:本文為轉(zhuǎn)載文章,轉(zhuǎn)載此文目的在于傳遞更多信息,版權(quán)歸原作者所有。本文所用視頻、圖片、文字如涉及作品版權(quán)問(wèn)題,請(qǐng)聯(lián)系小編進(jìn)行處理

審核編輯 黃宇

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

    關(guān)注

    146

    文章

    17317

    瀏覽量

    352646
  • CAN
    CAN
    +關(guān)注

    關(guān)注

    57

    文章

    2769

    瀏覽量

    464385
  • 編程
    +關(guān)注

    關(guān)注

    88

    文章

    3637

    瀏覽量

    93983
  • 變量
    +關(guān)注

    關(guān)注

    0

    文章

    613

    瀏覽量

    28466
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    TLC5615芯片輸出的初始值是不是為0?

    TLC5615芯片輸出的初始值是不是為0,還有為什么我的5615芯片我給了數(shù)據(jù)進(jìn)入要么就是輸出的0V要么就是輸出的4.8或者5V
    發(fā)表于 01-20 09:25

    電橋電子測(cè)試中的重要性

    電橋電子測(cè)試中的重要性體現(xiàn)在多個(gè)方面,以下是詳細(xì)的分析: 一、精確測(cè)量電參數(shù) 電橋作為一種精密的測(cè)量工具,能夠精確測(cè)量電阻、電容、電感等電參數(shù)。
    的頭像 發(fā)表于 01-09 10:03 ?183次閱讀

    PROM器件物聯(lián)網(wǎng)設(shè)備中的重要性

    PROM器件(Programmable Read-Only Memory,可編程只讀存儲(chǔ)器)物聯(lián)網(wǎng)(IoT)設(shè)備中扮演著至關(guān)重要的角色。以下是關(guān)于PROM器件物聯(lián)網(wǎng)設(shè)備中
    的頭像 發(fā)表于 11-23 11:35 ?354次閱讀

    元器件電路設(shè)計(jì)中的重要性

    元器件電路設(shè)計(jì)中的重要性是不言而喻的,它們構(gòu)成了電路的基本單元,并決定了電路的功能、性能以及可靠。以下從幾個(gè)方面詳細(xì)闡述元器件電路設(shè)計(jì)中的重要
    的頭像 發(fā)表于 10-29 16:17 ?557次閱讀

    plc編程st語(yǔ)言怎么編

    控制。下面是一個(gè)基本的ST語(yǔ)言編程指南,包括如何開(kāi)始編寫(xiě)一個(gè)簡(jiǎn)單的程序。 1. 基本結(jié)構(gòu) ST程序通常包含變量聲明、函數(shù)(或過(guò)程)定義和主程序塊。 變量聲明 :程序開(kāi)始處聲明所有使用
    的頭像 發(fā)表于 08-25 10:05 ?1508次閱讀

    XC8編譯器調(diào)用函數(shù)后局域變量值被改變是哪里出了問(wèn)題?

    最近發(fā)現(xiàn)個(gè)問(wèn)題,有個(gè)函數(shù)里邊調(diào)用別的函數(shù)后老出錯(cuò),查找程序沒(méi)有問(wèn)題,經(jīng)過(guò)仿真后發(fā)現(xiàn)在調(diào)用一個(gè)子函數(shù)后有幾個(gè)局部變量值會(huì)發(fā)生改變,再分析原因,發(fā)現(xiàn)本函數(shù)的局部變量和子函數(shù)的局部變量有地址重合的,子函數(shù)
    發(fā)表于 07-26 07:36

    NLP技術(shù)人工智能領(lǐng)域的重要性

    智能的橋梁,其重要性日益凸顯。本文將從NLP的定義、發(fā)展歷程、核心技術(shù)、應(yīng)用領(lǐng)域以及對(duì)人工智能領(lǐng)域的深遠(yuǎn)影響等多個(gè)維度,深入探討NLP技術(shù)人工智能領(lǐng)域的重要性
    的頭像 發(fā)表于 07-04 16:03 ?684次閱讀

    copy table中存放的是否為初始值不為0的data段所在PFlash的地址信息?

    1. copy table中存放的是否為初始值不為0的data段所在PFlash的地址信息?(比如:int a = 10; 變量a被存放到了PFlash的0x8030001c處,copy table
    發(fā)表于 07-02 07:51

    求助,ADC接地的重要性

    ADC接地的重要性
    發(fā)表于 06-04 07:56

    論RISC-V的MCU中UART接口的重要性

    的適用重要性某些應(yīng)用場(chǎng)景中,只需要異步通信能力的UART接口就能滿足需求,從而簡(jiǎn)化了系統(tǒng)設(shè)計(jì)和實(shí)現(xiàn)。 綜上所述,RISC-V的MCU中UART接口的
    發(fā)表于 05-27 15:52

    XC8調(diào)用函數(shù)后局域變量值被改變是怎么回事呀!

    最近發(fā)現(xiàn)個(gè)問(wèn)題,有個(gè)函數(shù)里邊調(diào)用別的函數(shù)后老出錯(cuò),查找程序沒(méi)有問(wèn)題,經(jīng)過(guò)仿真后發(fā)現(xiàn)在調(diào)用一個(gè)子函數(shù)后有幾個(gè)局部變量值會(huì)發(fā)生改變,再分析原因,發(fā)現(xiàn)本函數(shù)的局部變量和子函數(shù)的局部變量有地址重合的,
    發(fā)表于 04-30 15:33

    局部變量超出堆棧范圍,想把占用內(nèi)存多的變量申請(qǐng)?jiān)谌秩绾巫觯?/a>

    目前局部變量超出堆棧范圍,現(xiàn)在想把占用內(nèi)存多的變量申請(qǐng)?jiān)谌郑绾巫?/div>
    發(fā)表于 03-20 08:03

    集成芯片的重要性和必要

    集成芯片在現(xiàn)代科技和工業(yè)中占據(jù)著至關(guān)重要的地位,其重要性和必要主要體現(xiàn)在以下幾個(gè)方面。
    的頭像 發(fā)表于 03-18 15:17 ?1335次閱讀

    C語(yǔ)言中的三種形式變量

    局部變量一個(gè)函數(shù)、代碼塊內(nèi)部聲明的變量,只能被該函數(shù)或者代碼塊內(nèi)部應(yīng)用。局部變量函數(shù)之外不可用。
    發(fā)表于 03-11 17:34 ?801次閱讀
    C語(yǔ)言中的三種形式<b class='flag-5'>變量</b>

    蜂鳴器的用途和重要性

    蜂鳴器的用途和重要性? 蜂鳴器是一種能夠發(fā)出高音頻或低音頻聲音的設(shè)備,常用于警報(bào)系統(tǒng)、電子設(shè)備、通信系統(tǒng)和家庭電器中。它作為一種重要的聲音輸出設(shè)備,具有廣泛的用途和重要性。 首先,蜂鳴器
    的頭像 發(fā)表于 02-19 10:27 ?3626次閱讀
    百家乐官网AG| 伯爵百家乐官网娱乐网| 漾濞| 大发888dafabet| 百家乐游戏合法吗| 百家乐官网任你博娱乐场开户注册| 大玩家百家乐官网现金网| 保时捷娱乐城可靠吗| 诺贝尔百家乐官网的玩法技巧和规则 | 千亿娱百家乐官网的玩法技巧和规则 | 六合彩今天开什么| 全讯网zq06| 筹码百家乐的玩法技巧和规则| 百家乐开户博彩论坛| 娱乐城百家乐官网的玩法技巧和规则| 百家乐官网庄闲排| 澳门百家乐官网走势图| 百家乐官网趋势方向| 百家乐代理合作| bet365手机版| 大发888游戏平台黄埔网| 威尼斯人娱乐场66 | 昌宁县| 大东方娱乐城| E世博开户| 云鼎娱乐场送彩金| 大发888 dafa888 gzsums| 威尼斯人娱乐城客户端| 南京百家乐的玩法技巧和规则| 赌场百家乐赌场| 线上百家乐赌法| 百家乐现金网平台排行榜| 凯旋门百家乐娱乐城| 属兔魔羯女在哪个方位做生意| KK百家乐官网的玩法技巧和规则| 宝马会百家乐官网的玩法技巧和规则| 网上百家乐官网怎么赌能赢钱| 百家乐官网大白菜| 豪华百家乐官网桌子| 赌场百家乐官网是如何玩| 网络百家乐官网漏洞|