在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Arduino預處理器指令教程

李歡 ? 來源:醉狼工作室 ? 作者:醉狼工作室 ? 2023-02-24 09:51 ? 次閱讀

這篇文章來源于DevicePlus.com英語網站的翻譯稿。

poYBAGPzFk-AMyEfAASJRZLy_C0605.jpg

適用于ROHM傳感器評估套件的輕量級Arduino中,我介紹了RohmMultiSensor——幫您輕松連接ROHM傳感器評估套件多個傳感器的Arduino庫。該庫的核心特征之一就是通過僅編譯與所需傳感器相關的庫部分,顯著減小程序的大小。這意味著當您使用較少的傳感器時,整體程序大小和內存使用量會減小。但是,這究竟是如何實現的呢?當您#include一個庫然后按下“Upload”(上傳)按鈕之后,幕后究竟會發生什么?

硬件

Arduino UNO

軟件

Arduino IDE

幾乎所有用過Arduino的人都使用過庫。這就是Arduino編程對初學者來說如此簡單的原因之一——您無需深入了解傳感器的工作原理;庫會替您完成大部分工作。將代碼分成單獨的文件也是一種很好的編程習慣。組織、調試和維護單個文件要比處理一大堆代碼容易得多。

想必Arduino初學者都已經熟悉了將庫添加到主程序中的#include命令。要了解這是如何實現的,我們首先應快速了解C/C++源代碼如何編譯成程序。別擔心,這聽起來比較復雜,其實很簡單。我們來看一下編譯的工作原理。

按“上傳”之后

我們先做一個快速實驗:啟動Arduino IDE,打開其中一個示例代碼(比如“Blink”),然后按“Verify”按鈕。假設程序中沒有語法錯誤,底部的控制臺應該會打印出有關程序大小和內存的一些信息。嗯,剛才我們成功地將C++源代碼編譯成了二進制文件。在編譯過程中發生了以下幾件事:

Arduino IDE執行了一種名為“語法檢查”的操作,以確保您編寫的程序是真正的C/C++源代碼。此時,如果發生函數拼寫錯誤或忘記分號,那么編譯就會停止。

語法檢查之后,Arduino IDE會啟動另一個名為preprocessor(預處理器)的程序。這是一個非常簡單的程序,如果文件是C/C++源代碼,它不會怎么樣。我們稍后會詳細討論這一步驟。那么現在我們假設結果是一個名為“擴展源代碼”的文件——一個文本文件。

然后,該擴展源代碼被移交給另一個名為compiler(編譯器)的程序。該編譯器(在Arduino IDE中是avr-gcc)接收文本源,并生成匯編文件。匯編一種人類可讀的低級編程語言,但是更接近機器代碼——適用于特定處理器的指令。這里就是您編寫程序之前必須選擇正確Arduino板的原因——不同的開發板具有不同的處理器,而處理器又具有不同的指令集。

處理您Arduino程序下一個的系統程序叫做assembler(匯編程序)。該程序會生成一個“目標文件”。該文件主要是機器代碼,但也可以包含針對其他目標文件對象的“引用”。這允許Arduino IDE“預編譯”一些編寫Arduino程序時會始終用到的庫,從而使整個過程更快。

最后一個階段稱為鏈接,由另一個名為linker(鏈接器,顯而易見)的程序完成。鏈接器獲取目標文件并添加缺少的內容——主要是來自其他目標文件的符號,以生產可執行文件。在此之后,程序完全轉換為機器代碼,并可以上傳到電路板。

poYBAGPzFlCAP8kHAADv1ACnGVg466.jpg

現在,我們對Arduino程序編譯有了一個基本的了解。但是在上述所有編譯階段中,我們將只關注第二個階段:預處理器。

預處理器基本知識

在上本中,我提到預處理器本質上非常簡單:接收文本輸入,搜索關鍵字,根據找到的內容進行一些操作,然后輸出不同的文本。它非常簡單,同時也非常強大,因為它允許你用普通C/C++語言完成一些本來會非常復雜的事情(如果可能)。

預處理器會搜索以井號(#)開頭且后面有文本的行。這種語句叫做預處理器指令,是預處理器的一種“命令”。預處理器指令的完整列表以及詳細文檔的地址如下所示:

https://gcc.gnu.org/onlinedocs/cpp/Index-of-Directives.html#Index-of-Directives。

接下來,我將主要關注#include、#define和條件指令,因為這是Arduino最有用的指令。如果您想了解一些更“奇異”的指令,比如#assert 或 #pragma, 請參閱上述地址,以獲取官方信息。

添加額外代碼:#include 指令

這可能是最著名的預處理器指令,不僅Arduino愛好者都知道,而且C/C++編程人員也都了解。原因很簡單:該指令的作用是包含庫。但是,這究竟是如何實現的呢?確切的語法如下所示:

#include 

#include "file"

兩者的區別比較小,主要在于預處理器搜索file(文件)的確切位置。如果是第一句,預處理器僅搜索IDE指定的目錄。如果是第二句,預處理器首先查看包含源文件的文件夾,且僅當沒有在該目錄下找到file(文件) 時, 它才會搜索第一句的搜索目錄。由于包含庫的文件夾是在Arduino IDE中指定的,因此在包含庫時兩者之間沒有重大區別。

當預處理器找到文件時,它只是將其內容復制粘貼到源代碼中,以替代程序中的#include指令。但是,如果在任何目錄中都找不到此文件,就會引發致命錯誤,編譯停止。

要記住,預處理器只處理文本——無法理解那些特殊字母和數字的含義。最重要的是,它對所包含的內容和包含次數絕對不會進行更高級別的檢查。讓我們來看一下使用編寫不正確的庫會發生什么。

#include 

void setup() {

}

#include 

void loop() {

}

這個Arduino程序中沒有多少內容。請注意我們包含了一個名為“ExampleLibrary.h”的文件,而且我們包含了兩次。

//This is an example library

int a = 0;

//End of example library

“ExampleLibrary.h”的內容如下所示。同樣,除了一個整數變量之外,沒有多少內容。那么當我們編譯這個Arduino程序時會發生什么呢?

pYYBAGPzFlKABQHUAAJzvAUPScQ446.jpg

錯誤信息顯示變量a聲明了兩次,這導致編譯失敗。這是預處理器完成后源代碼的樣子。

//This is an example  library

int a = 0;

//End of example library

void setup() {

}

//This is an example  library

int a = 0;

//End of example library

void loop() {

}

顯而易見,不應該多次包含庫,但是如何在不依賴用戶的情況下實現這一目標?標準解決方案是將整個庫包含在以下結構中:

#ifndef _EXAMPLE_LIBRARY_H
#define _EXAMPLE_LIBRARY_H

//This is an example  library

int a = 0;

//End of example library

#endif

現在,第一次包含庫時,預處理器會檢查是否存在用“_EXAMPLE_LIBRARY_H”定義的內容。由于沒有類似的東西存在,預處理器繼續下一行并定義一個名為“_EXAMPLE_LIBRARY_H”的常量。然后,庫代碼被復制到程序中。

當第二次包含庫時,預處理器會再次檢查是否存在名為“_EXAMPLE_LIBRARY_H”的常量。這次,由于上一個#include命令已經定義了該常量,所以預處理器不會向程序中添加任何內容。于是,編譯成功完成。#ifdef 和 #endif是條件指令,我們稍后將對此進行討論。

定義事物:#define 指令

在上一個例子中,我們用#define指令創建了一個常量,以決定是否包含一個庫。在官方文檔中,任何由#define指令定義的東西都被稱為macro(宏), 因此本文中我會一直沿用這個術語。該指令的語法如下:

#define macro_name macro_body

大多數Arduino初學者可能會對宏感到困惑。如果我定義一個宏:

#define X 10

那么這與以下變量聲明有什么區別呢?

int Y = 10;

同樣,這一切都歸結為預處理器僅處理文本。遇到#define指令時,預處理器會搜索其余的源代碼并將所有出現的“X”替換為“10”。這意味著與變量不同,宏的值永遠不會改變。此外,您必須牢記預處理器只搜索以#define開頭的源代碼。讓我們看一下使用尚未定義的宏會發生什么情況。

int Y = X;
#define X 10
int Z = X;

void setup() {
 
}

void loop() {
 
}

編譯上述代碼會發生以下錯誤:

pYYBAGPzFlWAbpBYAAGCLzCzabE068.jpg

預處理后的代碼如下所示:

int Y = X;
int Z = 10;

void setup() {

}

void loop() {
 
}

第一行包含X,它被看作一個變量。但是,該變量從未聲明,因此編譯停止。

盡管#define指令最常見的用途是創建帶名稱的常量,但是它可以做的遠不止這些。例如,假設您想知道兩個給定數字中哪一個較小。您可以編寫一個實現此功能的函數。

int min(int a, int b) {
 if(a < b) {
   return(a);
 }
 return(b);
}

或者使用更簡單的三元運算符:

int min(int a, int b) {
 return((a < b) ? a : b);
}

但是,這兩個函數都將被編譯并占用寶貴的程序存儲空間。我們可以使用以下類似函數的宏來實現相同效果,但是占用的程序空間卻會變少。

#ifndef MIN
 #define MIN(A, B)     (((A) < (B)) ? (A) : (B))
#endif

現在,每個“MIN(A, B)”都會被替換為“(((A) < (B)) ? (A) : (B))”,其“A”和“B”可以是數字,也可以是變量。請注意,#define包含在相同的保護性結構中,以防止用戶重復定義宏。

創建宏時,您必須記住,系統將宏作為文本進行處理。這就是為什么在上面的定義中,幾乎所有內容都包含在括號中。請猜測以下運算的結果。

#ifndef MULTIPLY
 #define MULTIPLY(A, B)     A * B
#endif

//some code...

int result = MULTIPLY(2 - 0, 3);

結果應該是6,因為2–0=2,然后2x3=6,對吧?如果我告訴你結果是2呢?實際編譯的內容如下:

int result = 2 - 0 * 3;

由于乘法優先于減法,因此很明顯結果肯定是2,因為3x0=0,然后2-0=2。正確的版本如下所示:

#ifndef MULTIPLY
 #define MULTIPLY(A, B)     ((A) * (B))
#endif

條件編譯:#if指令

在前面的例子中,我使用了#ifndef指令,于是我可以檢查是否已經包含了庫。該指令可用于實現僅用C/C++語言不可能實現的內容:條件語句。這些指令的語法如下所示:

#if expression

  //compile this code

#elif different_expression
 
 //compile this different code

#else
 
 //compile this entirely different code

#endif

條件語句的常用功能是檢查一個宏是否已定義。為此,您可以使用幾個專門的指令:

#ifndef macro_name

  //compile this code if macro_name does not exist

#endif

我們已經熟悉了上述內容,因為我們之前使用此指令來檢查是否已包含庫。您也可以使用這個條件:

#ifdef macro_name

  //compile this code if macro_name exists

#endif

以上語句只是#if defined的簡寫,可根據單個條件測試多個宏。請注意,每個條件都必須用#endif 指令結束,從而指定代碼的哪些部分受條件影響,哪些部分不受條件影響。

我們來看一個實際的例子。假設您編寫了一個庫,并且希望它在Arduino UNO和Arduino Mega上都能正常工作。這主意不錯,對吧?便攜代碼總比為另一塊電路板修改庫更容易。但是,如果您的庫使用了SPI總線呢?該總線在Arduino UNO上用的是11-13引腳,但是在Mega上卻是50-52引腳。

那么您如何告訴編譯器根據不同開發板使用相應的引腳呢?您猜對了——條件語法!根據您在Arduino IDE中選擇(“Tools” > “Board”菜單)的開發板,IDE將定義不同的宏,從而僅編譯與所選開發板相關的代碼部分!這非常強大,因為您可以實現以下功能:

#if defined(__AVR_ATmega168__) || defined(__AVR_ATmega328P__)
 
 //this will compile for Arduino UNO, Pro and older boards
 int _sck = 13;
 int _miso = 12;
 int _mosi = 11;

#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
 
 //this will compile for Arduino Mega
 int _sck = 52;
 int _miso = 50;
 int _mosi = 51;

#endif

怎么樣,漂亮吧?僅用三行代碼,我們就制作了一個多平臺便攜庫!另外,這正是RohmMultiSensor庫(適用于ROHM傳感器評估套件的輕量級Arduino庫)如何知道應該為所選傳感器編譯哪些代碼。如果您看一下頭文件RohmMultiSensor.h里面的內容,您只會看到幾個#ifdef和幾個#include指令。由于所有特定傳感器代碼都存儲在單獨的.cpp文件中,因此將新傳感器添加到庫中很容易——只需創建另一個文件,然后創建與其他傳感器相同的#ifdef – #include – #endif結構即可。完成!

提供反饋:#warning 和 #error 指令

我們最后要介紹的指令是#warning#error。兩者但是不言自明,語法如下:

#warning "message"

#error "message"

預處理器遇到這些指令時,它會將message打印到Arduino IDE控制臺中。兩者之間的區別在于,發生#warning之后,編譯正常進行,而#error則會完全停止編譯。

我們可以在前文的例子中使用這兩個語句:

#if defined(__AVR_ATmega168__) || defined(__AVR_ATmega328P__)

  //this will compile for Arduino UNO, Pro and older boards
 int _sck = 13;
 int _miso = 12;
 int _mosi = 11;
 
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)

 //this will compile for Arduino Mega
 int _sck = 52;
 int _miso = 50;
 int _mosi = 51;
 
#else

 #error “Unsupported board selected!”

#endif

這樣,當用戶嘗試為其他Arduino開發板(比如Yún、LilyPad等)編譯該庫時,編譯會失敗,與沒有定義SPI引腳沒有任何關系。

結論

在本文中,我們介紹了C/C++預處理器的相關知識。希望您看過本文之后,就不會再害怕編譯預處理器、或指令等術語了。我總結一下本文描述的最重要的幾點內容:

編寫庫時,請務必將其放在 #ifndef – #define – #endif結構中。這個結構我們已經見過多次了。這可能會為您省去一些麻煩。定義類似函數的宏時同樣應該這樣做。

編寫代碼時,應確保程序易于移植到其他Arduino板上。相信我,未雨綢繆要比出現不兼容問題之后再想法解決要容易得多。

分而治之!幾個較小的文件總比一個1000多行的大文件要好得多。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 指令
    +關注

    關注

    1

    文章

    608

    瀏覽量

    35763
  • Arduino
    +關注

    關注

    188

    文章

    6472

    瀏覽量

    187357
  • 預處理器
    +關注

    關注

    0

    文章

    13

    瀏覽量

    2241
收藏 人收藏

    評論

    相關推薦

    請問如何使用預處理指令#pragma禁止優化某段代碼?哪里有c2000編譯預處理指令的說明資料?

    本帖最后由 一只耳朵怪 于 2018-6-13 16:51 編輯 如何使用預處理指令#pragma禁止優化某段代碼?哪里有c2000編譯預處理
    發表于 06-13 04:57

    Intel 8051兼容預處理器接口

    Intel 8051兼容預處理器接口
    發表于 02-12 12:12

    預處理器在Build Settings中定義錯誤

    這個問題用PSoC Creator 3.3(3.3.0.410)進行。你好社區我問你關于一個問題的幫助(Bug?)在PSoC Creator。我想要的是:在編譯環境中定義一個帶有處理器值的預處理器
    發表于 02-22 06:25

    怎么使用預處理程序指令

    你好, 我想使用預處理器指令進行條件編譯。我有一段代碼,我想在定義預處理器指令時包含這些代碼。在SPC5Studio中定義它的位置?這需要哪些設置? 在此先感謝您的回復。 麥克風。以
    發表于 06-21 07:21

    怎么使用#assert預處理程序指令

    大家好。在我的源代碼中,我使用“assert”預處理器指令來檢查常量值的一致性。如果我以簡單的方式使用它,通過直接賦值,一切如預期:define ABC 0x7Fassert ABC
    發表于 04-15 09:41

    STM32CubeIDE暗模式預處理器突出顯示錯誤怎么解決?

    我在 Ubuntu 18.04 系統上運行 STM32CubeIDE。我已將其切換為暗模式進行編程,但代碼中的任何“ #if ”預處理器指令都有淺色背景。這使得無法閱讀。我已經查看了語法著色的所有
    發表于 12-01 07:39

    處理器指令集設計

    處理器指令集設計垂直指令格式指令類型及其使用頻度CISC指令集特點 RISC指令集特點
    發表于 10-29 17:13 ?64次下載
    微<b class='flag-5'>處理器</b><b class='flag-5'>指令</b>集設計

    預處理器的工作原理作用

    預處理器的工作原理作用,希望對學者們有幫助。
    發表于 10-29 11:40 ?0次下載

    基于FPGA的傳像光纖束圖像預處理器

    基于FPGA的傳像光纖束圖像預處理器,下來看看
    發表于 08-30 15:10 ?12次下載

    C語言預處理命令的分類和工作原理詳細說明

    C 語言編程過程中,經常會用到如 #include、#define 等指令,這些標識開頭的指令被稱為預處理指令預處理
    發表于 11-25 10:34 ?18次下載
    C語言<b class='flag-5'>預處理</b>命令的分類和工作原理詳細說明

    C語言預處理指令及分類

    C/C++ 程序中的源代碼中包含以 # 開頭的各種編譯指令,這些指令稱為預處理指令預處理指令
    的頭像 發表于 11-29 10:14 ?2283次閱讀

    嵌入式C預處理器的基本概念和常用指令

    在嵌入式系統開發中,C預處理器是非常重要的一部分,可以在編譯之前對源代碼進行宏替換、條件編譯和包含等處理。在本文中,我們將介紹嵌入式C預處理器的基本概念和常用指令
    的頭像 發表于 04-13 16:11 ?943次閱讀

    C語言有哪些預處理操作?

    C語言的預處理是在編譯之前對源代碼進行處理的階段,它主要由預處理器完成。預處理器是一個獨立的程序,它負責對源代碼進行一些文本替換和處理,生成
    的頭像 發表于 12-08 15:40 ?635次閱讀
    C語言有哪些<b class='flag-5'>預處理</b>操作?

    C語言中的預處理器

    所有的預處理器命令都是以井號(#)開頭。它必須是第一個非空字符,為了增強可讀性,預處理器指令應從第一列開始。
    發表于 03-01 12:16 ?953次閱讀
    C語言中的<b class='flag-5'>預處理器</b>

    處理器指令的獲取過程

    處理器指令的獲取是計算機執行程序過程中的關鍵環節,它決定了微處理器如何對數據和指令進行處理。以下將詳細闡述微
    的頭像 發表于 10-05 15:16 ?343次閱讀
    主站蜘蛛池模板: 国产成人啪午夜精品网站| 中文字幕在线观看你懂的| 亚洲一区二区三区四| 欧美性色生活片天天看99| 国产免费一区二区三区| 日本xxxx色视频在线观看免| 国产精品香蕉成人网在线观看| 777奇米影音| 天天插日日干| sihu在线| 国产中日韩一区二区三区| 欧美色婷婷| 色佬网| 成人综合在线视频| 国产精品女人在线观看| 久久婷婷国产综合精品| 天天操天天舔天天干| 国产一级特黄aa大片爽爽| 亚洲色图22p| 国产一级片免费看| 国产黄色大片又色又爽| 亚洲午夜精品久久久久久成年| 国产精品久久永久免费| 在线免费黄色网址| 欧美乱xxxxxxxxx| 国产福利乳摇在线播放| tv电影天堂| 天天操天天插| 日本xxxx69hd| www.久操| 亚洲68283精品人体| 亚洲精品老司机综合影院| 91中文在线观看| 午夜黄色网| 高清欧美性xxxx成熟| 欧美a网站| 扒开双腿猛进入jk校视频| 精品国产免费久久久久久婷婷| 视频一区中文字幕| 亚洲人xx视频| 香港三级在线视频|