常規(guī)打印方法
在STM32的應(yīng)用中,我們常常對(duì)printf進(jìn)行重定向的方式來(lái)把打印信息printf到我們的串口助手。
在MDK環(huán)境中,我們常常使用MicroLIB+fputc的方式實(shí)現(xiàn)串口打印功能,即:


要實(shí)現(xiàn)fputc函數(shù)的原因是:printf函數(shù)依賴于fputc函數(shù),重新實(shí)現(xiàn)fputc內(nèi)部從串口發(fā)送數(shù)據(jù)即可間接地實(shí)現(xiàn)printf打印輸出數(shù)據(jù)到串口。
不知道大家有沒(méi)有看過(guò)正點(diǎn)原子裸機(jī)串口相關(guān)的例程,他們的串口例程里不使用MicroLIB,而是使用標(biāo)準(zhǔn)庫(kù)+fputc的方式。相關(guān)代碼如:
#if1 #pragmaimport(__use_no_semihosting) //標(biāo)準(zhǔn)庫(kù)需要的支持函數(shù) struct__FILE { inthandle; }; FILE__stdout; /** *@brief定義_sys_exit()以避免使用半主機(jī)模式 *@paramvoid *@returnvoid */ void_sys_exit(intx) { x=x; } intfputc(intch,FILE*f) { while((USART1->ISR&0X40)==0);//循環(huán)發(fā)送,直到發(fā)送完畢 USART1->TDR=(u8)ch; returnch; } #endif
關(guān)于這兩種方法的一些說(shuō)明可以查看Mculover666兄的《重定向printf函數(shù)到串口輸出的多種方法》這篇文章。這篇文章中不僅包含上面的兩種方法,而且也包含著在GCC中使用標(biāo)準(zhǔn)庫(kù)重定向printf的方法。
自己實(shí)現(xiàn)一個(gè)打印函數(shù)
以上的幾種方法基本上是改造C庫(kù)的printf函數(shù)來(lái)實(shí)現(xiàn)串口打印的功能。其實(shí)我們也可以自己實(shí)現(xiàn)一個(gè)串口打印的功能。
printf本身就是一個(gè)變參函數(shù),其原型為:
intprintf(constchar*__format,...);
所以,我們要重新封裝的一個(gè)串口打印函數(shù)自然也應(yīng)該是一個(gè)變參函數(shù)。具體實(shí)現(xiàn)如下:
1、基于STM32的HAL庫(kù)
左右滑動(dòng)查看全部代碼>>>
#defineTX_BUF_LEN256/*發(fā)送緩沖區(qū)容量,根據(jù)需要進(jìn)行調(diào)整*/ uint8_tTxBuf[TX_BUF_LEN];/*發(fā)送緩沖區(qū)*/ voidMyPrintf(constchar*__format,...) { va_listap; va_start(ap,__format); /*清空發(fā)送緩沖區(qū)*/ memset(TxBuf,0x0,TX_BUF_LEN); /*填充發(fā)送緩沖區(qū)*/ vsnprintf((char*)TxBuf,TX_BUF_LEN,(constchar*)__format,ap); va_end(ap); intlen=strlen((constchar*)TxBuf); /*往串口發(fā)送數(shù)據(jù)*/ HAL_UART_Transmit(&huart1,(uint8_t*)&TxBuf,len,0xFFFF); }
因?yàn)槲覀兪褂胮rintf函數(shù)基本不使用其返回值,所以這里直接用void類型了。
自定義變參函數(shù)需要用到va_start、va_end等宏,需要包含頭文件stdarg.h。關(guān)于變參函數(shù)的一些學(xué)習(xí)可以查看網(wǎng)上的一些博文,如:
https://www.cnblogs.com/wulei0630/p/9444062.html
這里我們使用的是STM32的HAL庫(kù),其給我們提供HAL_UART_Transmit接口可以直接把整個(gè)發(fā)送緩沖區(qū)的內(nèi)容給一次性發(fā)出去。
2、基于STM32標(biāo)準(zhǔn)庫(kù)
若是基于STM32的標(biāo)準(zhǔn)庫(kù),就需要一字節(jié)一字節(jié)的循環(huán)發(fā)送出去,具體代碼如:
左右滑動(dòng)查看全部代碼>>>
#defineTX_BUF_LEN256/*發(fā)送緩沖區(qū)容量,根據(jù)需要進(jìn)行調(diào)整*/ uint8_tTxBuf[TX_BUF_LEN];/*發(fā)送緩沖區(qū)*/ voidMyPrintf(constchar*__format,...) { va_listap; va_start(ap,__format); /*清空發(fā)送緩沖區(qū)*/ memset(TxBuf,0x0,TX_BUF_LEN); /*填充發(fā)送緩沖區(qū)*/ vsnprintf((char*)TxBuf,TX_BUF_LEN,(constchar*)__format,ap); va_end(ap); intlen=strlen((constchar*)TxBuf); /*往串口發(fā)送數(shù)據(jù)*/ for(inti=0;i
測(cè)試結(jié)果:


我們也可以使用我們的MyPrintf函數(shù)按照上一篇文章:《C語(yǔ)言、嵌入式中幾個(gè)非常實(shí)用的宏技巧》的方式封裝一個(gè)宏打印函數(shù):



以上就是我們自定義方式實(shí)現(xiàn)的一種串口打印函數(shù)。
但是,我想說(shuō):對(duì)于串口打印的使用,我們沒(méi)必要自己創(chuàng)建一個(gè)打印函數(shù)。
看到這,是不是有人想要打我了。。。??戳税胩?,你卻跟我說(shuō)沒(méi)必要用。。。
哈哈,別急,我們不應(yīng)用在串口打印調(diào)試方面,那可以用在其它方面呀。
(1)應(yīng)用一:
比如最近我在實(shí)際應(yīng)用中:我們的MCU跑的是我們老大自己寫(xiě)的一個(gè)小的操作系統(tǒng)+我們公司自己開(kāi)發(fā)的上位機(jī)。
我們MCU端與上位機(jī)使用的是串口通訊,MCU往上位機(jī)發(fā)送的數(shù)據(jù)有兩種類型,一種是HEX格式數(shù)據(jù),一種是字符串?dāng)?shù)據(jù)。
但是我們下位機(jī)的這兩種數(shù)據(jù),在通過(guò)串口發(fā)送之前都得統(tǒng)一把數(shù)據(jù)封包交給那個(gè)系統(tǒng)通信任務(wù),然后再由通信任務(wù)發(fā)出去。
在這里,就不能用printf了。老大也針對(duì)他的這個(gè)系統(tǒng)實(shí)現(xiàn)了一個(gè)deb_printf函數(shù)用于打印調(diào)試。
但是,那個(gè)函數(shù)既復(fù)雜又很雞肋,稍微復(fù)雜一點(diǎn)的數(shù)據(jù)就打印不出來(lái)了。
因此我利用上面的思路給它新封裝了一個(gè)打印調(diào)試函數(shù),很好用,完美地兼容了老大的那個(gè)系統(tǒng)。具體代碼就不分享了,大體代碼、思路如上。
(2)應(yīng)用二:
我們?cè)谑褂么谂cESP8266模塊通訊時(shí),可利用類似這樣的方式封裝一個(gè)發(fā)送數(shù)據(jù)的函數(shù),這個(gè)函數(shù)的使用可以像printf一樣簡(jiǎn)單。
可以以很簡(jiǎn)單的方式把數(shù)據(jù)透?jìng)髦练?wù)端,比如我以前的畢設(shè)中就有這么應(yīng)用:


-
STM32
+關(guān)注
關(guān)注
2312文章
11189瀏覽量
374557 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4421瀏覽量
67822 -
串口打印
+關(guān)注
關(guān)注
0文章
11瀏覽量
3353
原文標(biāo)題:串口打印知多少?
文章出處:【微信號(hào):mcu168,微信公眾號(hào):硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
【RA-Eco-RA2E1-V1.0開(kāi)發(fā)板試用】串口打印例程。
keil定義fputc函數(shù)
將C庫(kù)的printf函數(shù)重定向到UART的過(guò)程
【瑞薩RA6E2地奇星開(kāi)發(fā)板試用】OLED 屏幕驅(qū)動(dòng),串口打印
什么是KEIL的微庫(kù)(microlib)
使用Keil MicroLIB時(shí)自動(dòng)設(shè)置堆大小
NucleiStudio_IDE_201909串口打印浮點(diǎn)型數(shù)據(jù)時(shí)無(wú)數(shù)據(jù)輸出,軟件該如何配置串口才能打印浮點(diǎn)型數(shù)據(jù)?
【CPKCOR-RA8D1】基礎(chǔ)串口打印測(cè)試
【CPKCOR-RA8D1】+ 基礎(chǔ)串口打印測(cè)試
串口通信有哪些方式
【HarmonyOS 5】桌面快捷方式功能實(shí)現(xiàn)詳解
使用MicroLIB+fputc的方式實(shí)現(xiàn)串口打印功能
評(píng)論