OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 8549|回复: 20

求助原子哥,MCU接收串口的是ASCII,怎么能快速转成十进制,比如31 30 30 2E 35 36,怎么通过程序快速转换成100.56

[复制链接]

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
发表于 2016-8-16 23:09:33 | 显示全部楼层 |阅读模式
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

19

主题

430

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1341
金钱
1341
注册时间
2016-4-22
在线时间
187 小时
发表于 2016-8-16 23:09:34 | 显示全部楼层
用atoi 不就好了吗,标准的C函数
伤情最是晚凉天,憔悴斯人不堪怜。
邀酒摧肠三杯醉,寻香惊梦五更寒。
钗头凤斜卿有泪,荼蘼花了我无缘。
小楼寂寞新雨月,也难如钩也难圆。
回复

使用道具 举报

58

主题

6294

帖子

1

精华

资深版主

Rank: 8Rank: 8

积分
11562
金钱
11562
注册时间
2014-4-1
在线时间
1318 小时
发表于 2016-8-17 08:25:40 | 显示全部楼层

31 = 1
32 = 2
。。。
2E = 。


回复

使用道具 举报

4

主题

346

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3657
金钱
3657
注册时间
2016-2-21
在线时间
542 小时
发表于 2016-8-17 09:02:08 | 显示全部楼层
有C库函数可以快速转换
回复

使用道具 举报

15

主题

61

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
274
金钱
274
注册时间
2016-7-15
在线时间
88 小时
发表于 2016-8-17 09:03:33 | 显示全部楼层
显示程序的时候加0x30
回复

使用道具 举报

22

主题

751

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1606
金钱
1606
注册时间
2015-6-10
在线时间
223 小时
发表于 2016-8-17 09:27:39 | 显示全部楼层
一般情况下  发送和接收都扩大100倍,毕竟单片机处理浮点数还是挺费劲的,处理整数简单
回复

使用道具 举报

58

主题

6294

帖子

1

精华

资深版主

Rank: 8Rank: 8

积分
11562
金钱
11562
注册时间
2014-4-1
在线时间
1318 小时
发表于 2016-8-17 09:32:23 | 显示全部楼层

所以,要看后面做什么,
是要显示,还是参与运算?处理方法都不同。

回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-17 09:45:26 | 显示全部楼层
shikihane 发表于 2016-8-17 09:02
有C库函数可以快速转换

怎么转换啊,还是不懂啊
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-17 09:46:12 | 显示全部楼层
止天 发表于 2016-8-17 09:27
一般情况下  发送和接收都扩大100倍,毕竟单片机处理浮点数还是挺费劲的,处理整数简单

对啊对啊,我上位机就是不想发送十六进制数,因为这样不直观啊,我的最终目的就是在上位机发送一个电压值,比如是3.3,然后在stm32中通过DA转换就输出3.3V,如果我用十六进制数,我要发送255,才能输出3.3V这样就很不直观了,所以我才想,如果可以直接按ascii码直接发送3.3就很好了,
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-17 09:46:25 | 显示全部楼层
xuande 发表于 2016-8-17 09:32
所以,要看后面做什么,
是要显示,还是参与运算?处理方法都不同。

对啊对啊,我上位机就是不想发送十六进制数,因为这样不直观啊,我的最终目的就是在上位机发送一个电压值,比如是3.3,然后在stm32中通过DA转换就输出3.3V,如果我用十六进制数,我要发送255,才能输出3.3V这样就很不直观了,所以我才想,如果可以直接按ascii码直接发送3.3就很好了,
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-17 10:31:43 | 显示全部楼层
bosir 发表于 2016-8-17 09:45
怎么转换啊,还是不懂啊

哪个函数可以用啊
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-17 10:33:07 | 显示全部楼层
ysq7120 发表于 2016-8-17 08:45
用atoi 不就好了吗,标准的C函数

这个我也想用,可是编译一直不行啊,因为我的buf是字符数组,但是atoi是const char *类型的,你能帮帮我吗
回复

使用道具 举报

19

主题

430

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1341
金钱
1341
注册时间
2016-4-22
在线时间
187 小时
发表于 2016-8-17 13:30:36 | 显示全部楼层
bosir 发表于 2016-8-17 10:33
这个我也想用,可是编译一直不行啊,因为我的buf是字符数组,但是atoi是const char *类型的,你能帮帮我 ...

字符数组有啥问题,如果类型不匹配,类型转换一下就好了。
伤情最是晚凉天,憔悴斯人不堪怜。
邀酒摧肠三杯醉,寻香惊梦五更寒。
钗头凤斜卿有泪,荼蘼花了我无缘。
小楼寂寞新雨月,也难如钩也难圆。
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-18 10:00:25 | 显示全部楼层
ysq7120 发表于 2016-8-16 23:09
用atoi 不就好了吗,标准的C函数

大哥你也有做过类似的吗,能不能把代码发给我看看,因为我也用atoi这个函数,可是对于小数点就没办法了,而且对于两位数也不行,它是逐个转换成ascii码在进行计算,比如说,我输一个20,它先进行2的计算,然后再进行0的计算,并不是十进制的20
回复

使用道具 举报

19

主题

430

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1341
金钱
1341
注册时间
2016-4-22
在线时间
187 小时
发表于 2016-8-18 10:22:36 | 显示全部楼层
bosir 发表于 2016-8-18 10:00
大哥你也有做过类似的吗,能不能把代码发给我看看,因为我也用atoi这个函数,可是对于小数点就没办法了, ...

浮点数应该用atof或者strtod这两个函数,上面我没注意你是浮点额。
伤情最是晚凉天,憔悴斯人不堪怜。
邀酒摧肠三杯醉,寻香惊梦五更寒。
钗头凤斜卿有泪,荼蘼花了我无缘。
小楼寂寞新雨月,也难如钩也难圆。
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-18 16:27:14 | 显示全部楼层
问题已解决,谢谢大家啦,非常感谢
回复

使用道具 举报

8

主题

83

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
238
金钱
238
注册时间
2016-4-4
在线时间
31 小时
发表于 2016-8-19 16:13:22 | 显示全部楼层
bosir 发表于 2016-8-18 16:27
问题已解决,谢谢大家啦,非常感谢

求程序
回复

使用道具 举报

8

主题

83

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
238
金钱
238
注册时间
2016-4-4
在线时间
31 小时
发表于 2016-8-19 20:11:54 | 显示全部楼层

哥们,你有程序吗?能分享下吗?
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-23 21:41:54 | 显示全部楼层
wscs1993 发表于 2016-8-19 20:11
哥们,你有程序吗?能分享下吗?

可以可以啊怎么给你我复制代码吧#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"
#include "dac.h  "
#include "adc.h"
#include "stdlib.h"

/************************************************
ALIENTEK战舰STM32开发板实验4
串口实验
技术支持:www.openedv.com
淘宝店铺:http://eboard.taobao.com
关注微信公众平台微信号:"正点原子",免费获取STM32资料。
广州市星翼电子科技有限公司  
作者:正点原子 @ALIENTEK
************************************************/
void PWM_DAC_Set(u16 vol)
{
        float temp=vol;
        temp/=100;
        temp=temp*256/3.3;
        TIM_SetCompare1(TIM1,temp);
}

int main(void)
{
       
        u16 t;  
        u16 len;       
        u16 times=0;
        delay_init();                     //延时函数初始化          
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
        uart_init(9600);         //串口初始化为115200
        LED_Init();                             //LED端口初始化
        KEY_Init();          //初始化与按键连接的硬件接口
         Dac1_Init();
        // Dac2_Init();
         Adc_Init();
         
                 TIM1_PWM_Init(255,0);        //TIM1 PWM初始化, Fpwm=72M/256=281.25Khz.
  TIM_SetCompare1(TIM1,0);//初始值为0       
       
        while(1)
        {  
        /*        if(USART_RX_STA&0x8000)
                {                                          
                        len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
                //        printf("\r\n您发送的消息为:\r\n\r\n");
                       
                       
                        for(t=0;t<len;t++)
                        {
                                USART_SendData(USART1, USART_RX_BUF[t]);//向串口1发送数据
                                        //        printf("消息为%d",USART_RX_BUF[t]);
                                DAC_SetChannel1Data(DAC_Align_8b_R,USART_RX_BUF[t] );
                                        DAC_SetChannel2Data(DAC_Align_8b_R,USART_RX_BUF[t] );
                                TIM_SetCompare1(TIM1,USART_RX_BUF[t]);
                               
                               
                                while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
                                //printf("print('\r\n')");//插入换行
                               
                        }
       
                       
                        //printf("print('\r\n')");//插入换行
                        USART_RX_STA=0;
                }
                */
                if(USART_RX_STA&0x8000)
                {                                          
                        len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
                                if(len==1)
                                {
                                        USART_SendData(USART1, USART_RX_BUF[0]*25.5);//向串口1发送数据
                                       
                                DAC_SetChannel1Data(DAC_Align_8b_R,USART_RX_BUF[0]*25.5 );
                                        DAC_SetChannel2Data(DAC_Align_8b_R,USART_RX_BUF[0]*25.5 );
                                TIM_SetCompare1(TIM1,USART_RX_BUF[0]*25.5);
                                }
                                if(len==2)
                                {
                                        USART_SendData(USART1,  (USART_RX_BUF[0]*10+USART_RX_BUF[1])*25.5);//向串口1发送数据                               
                                DAC_SetChannel1Data(DAC_Align_8b_R, (USART_RX_BUF[0]*10+USART_RX_BUF[1])*25.5);
                                        DAC_SetChannel2Data(DAC_Align_8b_R, (USART_RX_BUF[0]*10+USART_RX_BUF[1])*25.5);
                                TIM_SetCompare1(TIM1, (USART_RX_BUF[0]*10+USART_RX_BUF[1])*25.5);
                                       
                                }
                                if(len==3)
                                {
                                        USART_SendData(USART1,  (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1)*25.5+0.3);//向串口1发送数据                               
                                DAC_SetChannel1Data(DAC_Align_8b_R, (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1)*25.5+0.3);
                                        DAC_SetChannel2Data(DAC_Align_8b_R, (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1)*25.5 +0.3);
                                TIM_SetCompare1(TIM1, (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1)*25.5+0.3);
                                       
                                }
                                if(len==4)
                                {
                                        USART_SendData(USART1,  (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1+USART_RX_BUF[3]*0.01)*25.5+0.3);//向串口1发送数据                               
                                DAC_SetChannel1Data(DAC_Align_8b_R, (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1+USART_RX_BUF[3]*0.01)*25.5+0.3 );
                                        DAC_SetChannel2Data(DAC_Align_8b_R, (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1+USART_RX_BUF[3]*0.01)*25.5+0.3);
                                TIM_SetCompare1(TIM1, (USART_RX_BUF[0]+USART_RX_BUF[2]*0.1+USART_RX_BUF[3]*0.01)*25.5+0.3);
                                       
                                }
                                while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
                                USART_RX_STA=0;
                               
                       
                }
                else
                {
                        times++;
                        /*if(times%5000==0)
                        {
                                printf("\r\n战舰STM32开发板 串口实验\r\n");
                                printf("正点原子@ALIENTEK\r\n\r\n");
                        }*/
                        //if(times%200==0)printf("请输入数据,以回车键结束\n");  
                        if(times%30==0)LED0=!LED0;//闪烁LED,提示系统正在运行.
                        delay_ms(10);   
                }
        }         
}

回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-23 21:42:22 | 显示全部楼层
wscs1993 发表于 2016-8-19 20:11
哥们,你有程序吗?能分享下吗?

#include "sys.h"
#include "usart.h"          
#include "stdlib.h"
//////////////////////////////////////////////////////////////////////////////////          
//如果使用ucos,则包括下面的头文件即可.
#if SYSTEM_SUPPORT_OS
#include "includes.h"                                        //ucos 使用          
#endif
//////////////////////////////////////////////////////////////////////////////////         
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//ALIENTEK STM32开发板
//串口1初始化                  
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//修改日期:2012/8/18
//版本:V1.5
//版权所有,盗版必究。
//Copyright(C) 广州市星翼电子科技有限公司 2009-2019
//All rights reserved
//********************************************************************************
//V1.3修改说明
//支持适应不同频率下的串口波特率设置.
//加入了对printf的支持
//增加了串口接收命令功能.
//修正了printf第一个字符丢失的bug
//V1.4修改说明
//1,修改串口初始化IO的bug
//2,修改了USART_RX_STA,使得串口最大接收字节数为2的14次方
//3,增加了USART_REC_LEN,用于定义串口最大允许接收的字节数(不大于2的14次方)
//4,修改了EN_USART1_RX的使能方式
//V1.5修改说明
//1,增加了对UCOSII的支持
//////////////////////////////////////////////////////////////////////////////////           


//////////////////////////////////////////////////////////////////
//加入以下代码,支持printf函数,而不需要选择use MicroLIB          
#if 1
#pragma import(__use_no_semihosting)            
//标准库需要的支持函数                 
struct __FILE
{
        int handle;

};

FILE __stdout;      
//定义_sys_exit()以避免使用半主机模式   
_sys_exit(int x)
{
        x = x;
}
//重定义fputc函数
int fputc(int ch, FILE *f)
{      
        while((USART1->SR&0X40)==0);//循环发送,直到发送完毕   
    USART1->DR = (u8) ch;      
        return ch;
}
#endif

/*使用microLib的方法*/
/*
int fputc(int ch, FILE *f)
{
        USART_SendData(USART1, (uint8_t) ch);

        while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET) {}       
   
    return ch;
}
int GetKey (void)  {

    while (!(USART1->SR & USART_FLAG_RXNE));

    return ((int)(USART1->DR & 0x1FF));
}
*/

#if EN_USART1_RX   //如果使能了接收
//串口1中断服务程序
//注意,读取USARTx->SR能避免莫名其妙的错误          
u8 USART_RX_BUF[USART_REC_LEN];     //接收缓冲,最大USART_REC_LEN个字节.
//接收状态
//bit15,        接收完成标志
//bit14,        接收到0x0d
//bit13~0,        接收到的有效字节数目
u16 USART_RX_STA=0;       //接收状态标记          
unsigned char msgbuf[32];
  
void uart_init(u32 bound){
  //GPIO端口设置
  GPIO_InitTypeDef GPIO_InitStructure;
        USART_InitTypeDef USART_InitStructure;
        NVIC_InitTypeDef NVIC_InitStructure;
         
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);        //使能USART1,GPIOA时钟
  
        //USART1_TX   GPIOA.9
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;        //复用推挽输出
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9
   
  //USART1_RX          GPIOA.10初始化
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10  

  //Usart1 NVIC 配置
  NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;                //子优先级3
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQ通道使能
        NVIC_Init(&NVIC_InitStructure);        //根据指定的参数初始化VIC寄存器
  
   //USART 初始化设置

        USART_InitStructure.USART_BaudRate = bound;//串口波特率
        USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
        USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
        USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
        USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
        USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;        //收发模式

  USART_Init(USART1, &USART_InitStructure); //初始化串口1
  USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启串口接受中断
  USART_Cmd(USART1, ENABLE);                    //使能串口1

}

void USART1_IRQHandler(void)                        //串口1中断服务程序
        {
        char  Res;
#if SYSTEM_SUPPORT_OS                 //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
        OSIntEnter();   
#endif
        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)
                {
                Res =USART_ReceiveData(USART1);        //读取接收到的数据
               
                if((USART_RX_STA&0x8000)==0)//接收未完成
                        {
                        if(USART_RX_STA&0x4000)//接收到了0x0d
                                {
                                if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
                                else USART_RX_STA|=0x8000;        //接收完成了
                                }
                        else //还没收到0X0D
                                {       
                                if(Res==0x0d)USART_RX_STA|=0x4000;
                                else
                                        {
                                //        USART_RX_BUF[USART_RX_STA&0X3FFF]=atof(&Res );
                                               
                                                 msgbuf[USART_RX_STA&0X3FFF]=atof(&Res);
                                                USART_RX_BUF[USART_RX_STA&0X3FFF]=msgbuf[USART_RX_STA&0X3FFF];
                                        USART_RX_STA++;
                                        if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收          
                                        }                 
                                }
                        }                    
     }
#if SYSTEM_SUPPORT_OS         //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
        OSIntExit();                                                                                           
#endif
}
#endif       
回复

使用道具 举报

10

主题

65

帖子

0

精华

初级会员

Rank: 2

积分
106
金钱
106
注册时间
2016-4-16
在线时间
13 小时
 楼主| 发表于 2016-8-23 21:43:03 | 显示全部楼层
wscs1993 发表于 2016-8-19 20:11
哥们,你有程序吗?能分享下吗?

一个是主函数,一个是串口函数,我的主函数主要是对应于0-10v的算法吧,其他的还没搞
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-30 23:09

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表