|
呵呵,算是第一篇學(xué)術(shù)性的文章了吧,希望是一個(gè)好的開(kāi)始吧。先說(shuō)明一下,開(kāi)發(fā)平臺(tái)win7,工具RVMDK(keil),硬件stm32f103ve,打印到超級(jí)終端
前兩天開(kāi)始關(guān)注一下一直被擱在一邊的printf。。。其實(shí)應(yīng)該有一個(gè)月前就有看了一下,調(diào)用C語(yǔ)言官方庫(kù),實(shí)現(xiàn)可變參數(shù)printf向串口打印字符串,數(shù)字等,呵呵,提高級(jí)的~~
先是在百度上看到一篇有關(guān)C語(yǔ)言可變參數(shù)的文章,感覺(jué)不是很難,兩天前嘗試了一下(其實(shí)弄了很久)昨天終于成功了~~(J-Link一步一步跟蹤進(jìn)去的)沒(méi)有調(diào)用C語(yǔ)言官方庫(kù)的,完全按照自己的理解寫(xiě)的。呵呵,今天就順便發(fā)表一下博文。
可變參數(shù)函數(shù)聲明格式為:type func(type para1, ...);
省略號(hào)就表示后面可以沒(méi)有參數(shù),一個(gè)參數(shù),或者多個(gè)參數(shù),而且類(lèi)型不必事先確定。就像C語(yǔ)言官方庫(kù)的printf("hello%s",str);帶了一個(gè)可變參數(shù),呵呵
編寫(xiě)前的基礎(chǔ)知識(shí)。堆棧��!函數(shù)調(diào)用的時(shí)候,函數(shù)參數(shù)進(jìn)棧順序是自右向左,例如,如果調(diào)用的時(shí)候只有兩個(gè)參數(shù),則para2先入棧,然后para1再入棧。多個(gè)參數(shù)也一樣,而且因?yàn)橛肅語(yǔ)言不用關(guān)心硬件結(jié)構(gòu),所以其他平臺(tái)上也同樣適用(好像是這樣)。還有就是棧是連續(xù)的數(shù)據(jù)結(jié)構(gòu),兩個(gè)參數(shù)之間地址是相鄰的,所以只要知道第一個(gè)參數(shù)地址,第二個(gè)參數(shù)地址就同樣可以獲取了~~~還有需要了解的就是函數(shù)傳遞參數(shù)的知識(shí)了。同樣用void func(para1,para2, ...) 說(shuō)明一下。C語(yǔ)言為傳遞參數(shù)para1,para2,執(zhí)行函數(shù)func(para1,para2,...)時(shí)會(huì)為每個(gè)參數(shù)創(chuàng)建一個(gè)副本,_para1,_para2,這樣才不會(huì)影響para1,para2的內(nèi)容。那么和函數(shù)創(chuàng)建的這兩個(gè)副本就是我們需要用到的,進(jìn)棧的參數(shù)了�。�!想辦法獲取_para1地址,根據(jù)棧是連續(xù)的數(shù)據(jù)結(jié)構(gòu),就能獲取第二個(gè)參數(shù)para2的地址了~~
首先當(dāng)然要先獲取第一個(gè)參數(shù)的地址了,可以嘗試用一下方法
void func(uint8_t*para1)
{
uint8_t *p;
p =(uint8_t*)(¶1);//這樣就獲取到進(jìn)棧副本_para1的地址了
}
如果有兩個(gè)參數(shù)的
void func(uint8_t *para1,uint8_t *para2)
{
uint8_t *p,*test;
p =(uint8_t*)(¶1); //先獲取第一個(gè)參數(shù)進(jìn)棧的地址
test =(uint8_t*)(¶2); //獲取第二個(gè)參數(shù)進(jìn)棧的地址
p += 4; //同樣獲取到第二個(gè)參數(shù)進(jìn)棧地址了!!!不信可以自行測(cè)試
}
既然可以獲取到進(jìn)棧參數(shù)在棧上的地址,那么要如何利用呢?請(qǐng)看例子
uint8_t *str = "World";//定義一個(gè)指向字符串的指針
void func(uint8_t *para1,...); //聲明可變參數(shù)函數(shù)
void main()
{
func("Hello", p);//在主函數(shù)中測(cè)試,實(shí)際測(cè)試打印出的是"World"
}
void func()
{
uint8_t *p;//局部變量,用以獲取可變參數(shù)指針
uint8_t *next;
p =(uint8_t*)(¶1); //獲取第一個(gè)參數(shù)進(jìn)棧指針
p += 4;//獲取第二個(gè)進(jìn)棧參數(shù)指針
next =(uint8_t*)(*((uint32_t*)(next)));//此時(shí)next指向str地址,等同于str
USART_SendString(USART1,next); //我的stm32串口打印函數(shù),打印的是str的內(nèi)容,“World”
}
這樣成功地用到了可變參數(shù)了,呵呵,不過(guò)有一個(gè)缺點(diǎn)。就是必須知道參數(shù)的個(gè)數(shù),像C語(yǔ)言官方庫(kù)中也是需要用到%s,%d等在printf中說(shuō)明出可變參數(shù)類(lèi)型,并確定了可變參數(shù)的個(gè)數(shù)。從%+‘x’可以看出,‘x’表示可變參數(shù)數(shù)據(jù)類(lèi)型。我們?cè)谧约簩?xiě)帶可變參數(shù)函數(shù)的時(shí)候也要在可變參數(shù)里面想辦法得到參數(shù)個(gè)數(shù),給出參數(shù)類(lèi)型,例如%s就是一個(gè)很好了模板了
嘗試優(yōu)化一下,完成功能更加接近printf的函數(shù)。模仿C語(yǔ)言官方庫(kù),在第一個(gè)參數(shù)里面顯式或隱式給出參數(shù)個(gè)數(shù)和參數(shù)類(lèi)型,請(qǐng)看下面例子
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); //獲取第一個(gè)參數(shù)進(jìn)棧指針
if(*para1 == '%')
{
para++;
switch(*para1)
{
case 's':
p += 4;
next =(uint8_t*)(*((uint32_t*)(next)));//此時(shí)next指向str地址,等同于str
while(*next)
{
USART_Sned_8b(USART1,*next);//我的stm32串口打印函數(shù),打印一個(gè)字符*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ù)就實(shí)現(xiàn)了可變參數(shù)%s了,同樣的道理,可以繼續(xù)添加如%d,%c等,前提是理解了上面的例子,呵呵
還是有點(diǎn)模糊的同學(xué)請(qǐng)自行充電。。。
下面給出我自己在stm32上封裝的函數(shù)void USART_printf(USART_TypeDef* USARTx,uint8_t* data, ...);
i的作用是解決第一次串口發(fā)送數(shù)據(jù)重第二個(gè)字符發(fā)送的bug
void USART_printf(USART_TypeDef* USARTx, uint8_t* data,...)
{
uint8_t *position, *next,*next_addr;//position保存第一個(gè)參數(shù)入棧地址,next指向下一個(gè)
static uint8_t i = 1; //解決bug,第一次串口發(fā)送數(shù)據(jù)重第二個(gè)字符發(fā)送
position = (uint8_t*)(&data);//獲取第一個(gè)參數(shù)入棧地址
next = position;//保存在next
while(*data)//第二個(gè)參數(shù),字符串
{
if(*(data - i) == '%' ) //判斷是否為'%'
{
data++; //指針移至下一個(gè),為下一步判斷做準(zhǔn)備
switch(*(data - i))//如果是'%'需要判斷下一個(gè)字符是s/d
{
case 's'://字符串處理
next += 4;//獲取下一個(gè)參數(shù)入棧地址
next_addr =(uint8_t*)(*((uint32_t*)(next)));//恢復(fù)第二個(gè)參數(shù)入棧錢(qián)地址
while(*next_addr)//發(fā)送字符串函數(shù)循環(huán)
{
USART_Send_8b(USARTx, *next_addr);//發(fā)一個(gè)字符
next_addr++; //指針移至下一位
}
data++; //跳過(guò)發(fā)送字符's'
break; //結(jié)束本次switch
case 'd': //整變變量處理
next += 4; //獲取下一個(gè)參數(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ā)送一個(gè)字符
next_addr++; //指針移到下一位
}
data++; //跳過(guò)發(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'); //回到第一個(gè)字符位置
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');//收尾工作,換行
}
呵呵,我整個(gè)函數(shù)就是這樣封裝的,還有些地方可以改進(jìn),不過(guò)我還沒(méi)查,過(guò)一兩天在說(shuō)吧,這兩天先休息一下好了。
歡迎大家一起學(xué)習(xí)~~
|
|