這篇文章來源于DevicePlus.com英語網站的翻譯稿。
在適用于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(鏈接器,顯而易見)的程序完成。鏈接器獲取目標文件并添加缺少的內容——主要是來自其他目標文件的符號,以生產可執行文件。在此之后,程序完全轉換為機器代碼,并可以上傳到電路板。
現在,我們對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程序時會發生什么呢?
錯誤信息顯示變量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() { }
編譯上述代碼會發生以下錯誤:
預處理后的代碼如下所示:
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文章
611瀏覽量
35791 -
Arduino
+關注
關注
188文章
6477瀏覽量
187554 -
預處理器
+關注
關注
0文章
13瀏覽量
2243
發布評論請先 登錄
相關推薦
評論