www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當(dāng)前位置:首頁 > 單片機 > 單片機
[導(dǎo)讀]先說明一下,開發(fā)平臺win7,工具RVMDK(keil),硬件stm32f103ve,打印到超級終端前兩天開始關(guān)注一下一直被擱在一邊的printf。。。其實應(yīng)該有一個月前就有看了一下,調(diào)用C語言官方庫,實現(xiàn)可變參數(shù)printf向串口打印字符

先說明一下,開發(fā)平臺win7,工具RVMDK(keil),硬件stm32f103ve,打印到超級終端

前兩天開始關(guān)注一下一直被擱在一邊的printf。。。其實應(yīng)該有一個月前就有看了一下,調(diào)用C語言官方庫,實現(xiàn)可變參數(shù)printf向串口打印字符串,數(shù)字等,呵呵,提高級的~~

先是在百度上看到一篇有關(guān)C語言可變參數(shù)的文章,感覺不是很難,兩天前嘗試了一下(其實弄了很久)昨天終于成功了~~(J-Link一步一步跟蹤進(jìn)去的)沒有調(diào)用C語言官方庫的,完全按照自己的理解寫的。呵呵,今天就順便發(fā)表一下博文。

可變參數(shù)函數(shù)聲明格式為:type func(type para1, ...);

省略號就表示后面可以沒有參數(shù),一個參數(shù),或者多個參數(shù),而且類型不必事先確定。就像C語言官方庫的printf("hello%s",str);帶了一個可變參數(shù),呵呵

編寫前的基礎(chǔ)知識。堆棧??!函數(shù)調(diào)用的時候,函數(shù)參數(shù)進(jìn)棧順序是自右向左,例如,如果調(diào)用的時候只有兩個參數(shù),則para2先入棧,然后para1再入棧。多個參數(shù)也一樣,而且因為用C語言不用關(guān)心硬件結(jié)構(gòu),所以其他平臺上也同樣適用(好像是這樣)。還有就是棧是連續(xù)的數(shù)據(jù)結(jié)構(gòu),兩個參數(shù)之間地址是相鄰的,所以只要知道第一個參數(shù)地址,第二個參數(shù)地址就同樣可以獲取了~~~還有需要了解的就是函數(shù)傳遞參數(shù)的知識了。同樣用voidfunc(para1, para2, ...)說明一下。C語言為傳遞參數(shù)para1,para2,執(zhí)行函數(shù)func(para1, para2, ...)時會為每個參數(shù)創(chuàng)建一個副本,_para1,_para2,這樣才不會影響para1,para2的內(nèi)容。那么和函數(shù)創(chuàng)建的這兩個副本就是我們需要用到的,進(jìn)棧的參數(shù)了?。?!想辦法獲取_para1地址,根據(jù)棧是連續(xù)的數(shù)據(jù)結(jié)構(gòu),就能獲取第二個參數(shù)para2的地址了~~

首先當(dāng)然要先獲取第一個參數(shù)的地址了,可以嘗試用一下方法

void func(uint8_t* para1)

{

uint8_t *p;

p = (uint8_t*)(¶1); //這樣就獲取到進(jìn)棧副本_para1的地址了

}

如果有兩個參數(shù)的

void func(uint8_t *para1, uint8_t *para2)

{

uint8_t *p, *test;

p = (uint8_t*)(¶1); //先獲取第一個參數(shù)進(jìn)棧的地址

test = (uint8_t*)(¶2); //獲取第二個參數(shù)進(jìn)棧的地址

p += 4; //同樣獲取到第二個參數(shù)進(jìn)棧地址了!!! 不信可以自行測試

}


既然可以獲取到進(jìn)棧參數(shù)在棧上的地址,那么要如何利用呢?請看例子

uint8_t *str = "World"; //定義一個指向字符串的指針

void func(uint8_t *para1, ...); //聲明可變參數(shù)函數(shù)

void main()

{

func("Hello", p); //在主函數(shù)中測試,實際測試打印出的是"World"

}

void func()

{

uint8_t *p; //局部變量,用以獲取可變參數(shù)指針

uint8_t *next;

p = (uint8_t*)(¶1); //獲取第一個參數(shù)進(jìn)棧指針

p += 4; //獲取第二個進(jìn)棧參數(shù)指針

next = (uint8_t*)(*((uint32_t*)(next)));//此時next指向str地址,等同于str

USART_SendString(USART1, next); //我的stm32串口打印函數(shù),打印的是str的內(nèi)容,“World”

}

這樣成功地用到了可變參數(shù)了,呵呵,不過有一個缺點。就是必須知道參數(shù)的個數(shù),像C語言官方庫中也是需要用到%s,%d等在printf中說明出可變參數(shù)類型,并確定了可變參數(shù)的個數(shù)。從%+‘x’可以看出,‘x’表示可變參數(shù)數(shù)據(jù)類型。我們在自己寫帶可變參數(shù)函數(shù)的時候也要在可變參數(shù)里面想辦法得到參數(shù)個數(shù),給出參數(shù)類型,例如%s就是一個很好了模板了

嘗試優(yōu)化一下,完成功能更加接近printf的函數(shù)。模仿C語言官方庫,在第一個參數(shù)里面顯式或隱式給出參數(shù)個數(shù)和參數(shù)類型,請看下面例子

uint8_t *str = "World";

void func(uint8_t *para1, ...)

void main()

{

func("Hello %s", str);

}

void func(uint8_t *para1, ...)

{

uint8_t *p; //局部變量,用以獲取可變參數(shù)指針

uint8_t *next;//用以保存指向可變參數(shù)的源地址

p = (uint8_t*)(¶1); //獲取第一個參數(shù)進(jìn)棧指針

if(*para1 == '%')

{

para++;

switch(*para1)

{

case 's':

p += 4;

next = (uint8_t*)(*((uint32_t*)(next)));//此時next指向str地址,等同于str

while(*next)

{

USART_Sned_8b(USART1, *next);//我的stm32串口打印函數(shù),打印一個字符*P

next++;指針移到下一位

}

break;


default:

USART_Send_8b(USART1. '%');//不是%s,先發(fā)送字符'%'

USART_Send_8b(USART1, *p);//不是's',發(fā)送該字符

para1++; 指針移到下一位

break;

}

}//if(*para1 == '%')

else

{

USART_Send_8b(USART1, *p);//不是'%',發(fā)送該字符

para1++; 指針移到下一位

}


上面函數(shù)就實現(xiàn)了可變參數(shù)%s了,同樣的道理,可以繼續(xù)添加如%d,%c等,前提是理解了上面的例子,呵呵

還是有點模糊的同學(xué)請自行充電。。。

下面給出我自己在stm32上封裝的函數(shù)void USART_printf(USART_TypeDef* USARTx, uint8_t* data, ...);

i的作用是解決第一次串口發(fā)送數(shù)據(jù)重第二個字符發(fā)送的bug

void USART_printf(USART_TypeDef* USARTx, uint8_t* data, ...)

{

uint8_t *position, *next, *next_addr;//position保存第一個參數(shù)入棧地址,next指向下一個

static uint8_t i = 1; //解決bug,第一次串口發(fā)送數(shù)據(jù)重第二個字符發(fā)送

position = (uint8_t*)(&data);//獲取第一個參數(shù)入棧地址

next = position;//保存在next

while(*data)//第二個參數(shù),字符串

{

if(*(data - i) == '%' ) //判斷是否為'%'

{

data++; //指針移至下一個,為下一步判斷做準(zhǔn)備

switch(*(data - i))//如果是'%'需要判斷下一個字符是s/d

{

case 's'://字符串處理

next += 4;//獲取下一個參數(shù)入棧地址

next_addr = (uint8_t*)(*((uint32_t*)(next)));//恢復(fù)第二個參數(shù)入棧錢地址

while(*next_addr)//發(fā)送字符串函數(shù)循環(huán)

{

USART_Send_8b(USARTx, *next_addr);//發(fā)一個字符

next_addr++; //指針移至下一位

}

data++; //跳過發(fā)送字符's'

break; //結(jié)束本次switch

case 'd': //整變變量處理

next += 4; //獲取下一個參數(shù)入棧地址

next_addr = NumberToString((uint16_t)(*((uint32_t*)(next))));//獲取數(shù)字轉(zhuǎn)換成字符后的地址

while(*next_addr) //發(fā)送轉(zhuǎn)換好的字符串循環(huán)

{

USART_Send_8b(USARTx, *next_addr);//發(fā)送一個字符

next_addr++; //指針移到下一位

}

data++; //跳過發(fā)送字符'd'

break; //結(jié)束本次switch

default : //'%'后面不是s/d

USART_Send_8b(USARTx, '%'); //發(fā)送'%'

USART_Send_8b(USARTx, *(data - i)); //發(fā)送'%'后面字符

break; //結(jié)束本次switch

} //switch(*(data - i))

}

else if(*(data - i) == 'n' ) //如果是換行字符

{

USART_Send_8b(USARTx, 'r'); //回到第一個字符位置

USART_Send_8b(USARTx, 'n'); //下一行

data++;

}

else //if(*(data - i) == '%' )

{

USART_Send_8b(USARTx, *(data - i)); //不是'%',發(fā)送該字符

data++; //指針移至下一位

}

} //while(*data)

if(i == 1)

{

i = 0;

}

USART_Send_8b(USARTx, 'r');

USART_Send_8b(USARTx, 'n');//收尾工作,換行

}


本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

在嵌入式開發(fā)中,STM32的時鐘系統(tǒng)因其靈活性和復(fù)雜性成為開發(fā)者關(guān)注的焦點。然而,看似簡單的時鐘配置背后,隱藏著諸多易被忽視的陷阱,輕則導(dǎo)致系統(tǒng)不穩(wěn)定,重則引發(fā)硬件損壞。本文從時鐘源選擇、PLL配置、總線時鐘分配等關(guān)鍵環(huán)...

關(guān)鍵字: STM32 時鐘系統(tǒng)

在嵌入式系統(tǒng)開發(fā)中,STM32系列微控制器的內(nèi)部溫度傳感器因其低成本、高集成度特性,廣泛應(yīng)用于設(shè)備自檢、環(huán)境監(jiān)測等場景。然而,受芯片工藝差異和電源噪聲影響,其原始數(shù)據(jù)存在±1.5℃的固有誤差。本文從硬件配置、校準(zhǔn)算法、軟...

關(guān)鍵字: STM32 溫度傳感器

在能源效率與智能化需求雙重驅(qū)動下,AC-DC轉(zhuǎn)換器的數(shù)字控制技術(shù)正經(jīng)歷從傳統(tǒng)模擬方案向全數(shù)字架構(gòu)的深刻變革?;赟TM32微控制器的PFM(脈沖頻率調(diào)制)+PWM(脈沖寬度調(diào)制)混合調(diào)制策略,結(jié)合動態(tài)電壓調(diào)整(Dynam...

關(guān)鍵字: AC-DC STM32

當(dāng)前智能家居產(chǎn)品需求不斷增長 ,在這一背景下 ,對現(xiàn)有澆花裝置缺陷進(jìn)行了改進(jìn) ,設(shè)計出基于STM32單片機的全 自動家用澆花機器人。該設(shè)計主要由機械結(jié)構(gòu)和控制系統(tǒng)構(gòu)成 ,機械結(jié)構(gòu)通過麥克納姆輪底盤與噴灑裝置的結(jié)合實現(xiàn)機器...

關(guān)鍵字: STM32 麥克納姆輪 安全可靠 通過性強

用c++編程似乎是讓你的Arduino項目起步的障礙嗎?您想要一種更直觀的微控制器編程方式嗎?那你需要了解一下Visuino!這個圖形化編程平臺將復(fù)雜電子項目的創(chuàng)建變成了拖動和連接塊的簡單任務(wù)。在本文中,我們將帶您完成使...

關(guān)鍵字: Visuino Arduino ESP32 STM32

鏈表作為一種基礎(chǔ)的數(shù)據(jù)結(jié)構(gòu),在程序設(shè)計中扮演著重要角色。掌握鏈表的高效操作技巧,特別是逆序、合并和循環(huán)檢測,對于提升算法性能和解決復(fù)雜問題至關(guān)重要。本文將詳細(xì)介紹這些操作的C語言實現(xiàn),并分析其時間復(fù)雜度。

關(guān)鍵字: 鏈表 C語言

在C/C++多文件編程中,靜態(tài)變量(static)與全局變量的作用域規(guī)則看似簡單,實則暗藏諸多陷阱。開發(fā)者若未能準(zhǔn)確理解其鏈接屬性與生命周期,極易引發(fā)難以調(diào)試的內(nèi)存錯誤、競態(tài)條件以及維護災(zāi)難。本文將深入剖析這兩類變量的作...

關(guān)鍵字: 靜態(tài)變量 全局變量 C語言

在嵌入式系統(tǒng)和服務(wù)器開發(fā)中,日志系統(tǒng)是故障排查和運行監(jiān)控的核心組件。本文基于Linux環(huán)境實現(xiàn)一個輕量級C語言日志庫,支持DEBUG/INFO/WARN/ERROR四級日志分級,并實現(xiàn)按大小滾動的文件輪轉(zhuǎn)機制。該設(shè)計在某...

關(guān)鍵字: C語言 嵌入式系統(tǒng)

在嵌入式系統(tǒng)和底層驅(qū)動開發(fā)中,C語言因其高效性和可控性成為主流選擇,但缺乏原生單元測試支持成為開發(fā)痛點。本文提出一種基于宏定義和測試用例管理的輕量級單元測試框架方案,通過自定義斷言宏和測試注冊機制,實現(xiàn)無需外部依賴的嵌入...

關(guān)鍵字: C語言 嵌入式系統(tǒng) 驅(qū)動開發(fā)

在嵌入式系統(tǒng)開發(fā)中,實時操作系統(tǒng)(RTOS)的任務(wù)調(diào)度算法直接影響系統(tǒng)的響應(yīng)速度和資源利用率。時間片輪轉(zhuǎn)(Round-Robin, RR)作為一種經(jīng)典的公平調(diào)度算法,通過為每個任務(wù)分配固定時間片實現(xiàn)多任務(wù)并發(fā)執(zhí)行。本文將...

關(guān)鍵字: 實時操作系統(tǒng) RTOS C語言
關(guān)閉