你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

stm32f103 串口通讯 printf 输出中文的问题

[复制链接]
hhhhhhhaa 提问时间:2018-8-24 11:54 /
使用stm32f103vct6的mcu  用printf输出到串口usart2上  笔记本用串口调试助手,怎么显示中文啊? 串口调试助手没法收到中文

收藏 评论25 发布时间:2018-8-24 11:54

举报

25个回答
tyhjrwx2011 回答时间:2018-8-24 13:08:26
江苏方言 放到 东北也不能通用,虽然都是汉语
hhhhhhhaa 回答时间:2018-9-4 09:06:50
stm1024 发表于 2018-9-3 14:03
把你发送的字符串和接收到的字节发出来,我看看

#include "stm32f10x.h"
#include "stdio.h"
#include "delay.h"
#include "sys.h"




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

};

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



//int fputc(int ch, FILE *f)
//{

//       
//        USART_SendData(USART2,ch);
//        while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET)
//        {
//               
//        }
//       
//  return ch;
//}


void My_USART2_Init(void)
{

       
        GPIO_InitTypeDef GPIO_InitStrue;
        USART_InitTypeDef USART_InitStrue;
        NVIC_InitTypeDef NVIC_InitStrue;
       
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);
       
        GPIO_InitStrue.GPIO_Mode=GPIO_Mode_AF_PP;
        GPIO_InitStrue.GPIO_Pin=GPIO_Pin_2;
        GPIO_InitStrue.GPIO_Speed=GPIO_Speed_10MHz;
  GPIO_Init(GPIOA,&GPIO_InitStrue);
       
        GPIO_InitStrue.GPIO_Mode=GPIO_Mode_IN_FLOATING;
        GPIO_InitStrue.GPIO_Pin=GPIO_Pin_3;
        GPIO_InitStrue.GPIO_Speed=GPIO_Speed_10MHz;
  GPIO_Init(GPIOA,&GPIO_InitStrue);
       
       
        USART_InitStrue.USART_BaudRate=9600;
        USART_InitStrue.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
        USART_InitStrue.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;
        USART_InitStrue.USART_Parity=USART_Parity_No;
        USART_InitStrue.USART_StopBits=USART_StopBits_1;
        USART_InitStrue.USART_WordLength=USART_WordLength_8b;
       
        USART_Init(USART2,&USART_InitStrue);
        USART_Cmd(USART2,ENABLE);
        USART_ITConfig(USART2,USART_IT_RXNE,ENABLE);
       
       
        NVIC_InitStrue.NVIC_IRQChannel=USART2_IRQn;
        NVIC_InitStrue.NVIC_IRQChannelCmd=ENABLE;
        NVIC_InitStrue.NVIC_IRQChannelPreemptionPriority=1;
        NVIC_InitStrue.NVIC_IRQChannelSubPriority=1;
        NVIC_Init(&NVIC_InitStrue);
       
       
}

void USART2_IRQHandler(void)
{
        u8 Res;
         if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)
                {
                Res =USART_ReceiveData(USART2);       
//                        Res++;
                USART_SendData(USART2, Res);
                       
     }
}
  
int main(void)
{

         
                        delay_init();                               
                        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
                        My_USART2_Init();
       
   
       
              while(1)
              {
                                                               
                                                printf("今天");
                                               
                                                        delay_ms(10);   
                                                                               
               
                }
               
       
}
hhhhhhhaa 回答时间:2018-9-4 08:58:54
stm1024 发表于 2018-9-3 16:52
UTF-8的“今”,“天”
分别是
“E4 BB 8A”

看你也是小白 给你普及一下吧 先从字符编码讲起。 1、美国人首先对其英文字符进行了编码,也就是最早的ascii码,用一个字节的低7位来表示英文的128个字符,高1位统一为0; 2、后来欧洲人发现尼玛你这128位哪够用,比如我高贵的法国人字母上面的还有注音符,这个怎么区分,得,把高1位编进来吧,这样欧洲普遍使用一个全字节进行编码,最多可表示256位。欧美人就是喜欢直来直去,字符少,编码用得位数少; 3、但是即使位数少,不同国家地区用不同的字符编码,虽然0--127表示的符号是一样的,但是128--255这一段的解释完全乱套了,即使2进制完全一样,表示的字符完全不一样,比如135在法语,希伯来语,俄语编码中完全是不同的符号; 4、更麻烦的是,尼玛这电脑高科技传到中国后,中国人发现我们有10万多个汉字,你们欧美这256字塞牙缝都不够。于是就发明了GB2312这些汉字编码,典型的用2个字节来表示绝大部分的常用汉字,最多可以表示65536个汉字字符,这样就不难理解有些汉字你在新华字典里查得到,但是电脑上如果不处理一下你是显示不出来的了吧。 5、这下各用各的字符集编码,这世界咋统一?俄国人发封email给中国人,两边字符集编码不同,尼玛显示都是乱码啊。为了统一,于是就发明了unicode,将世界上所有的符号都纳入其中,每一个符号都给予一个独一无二的编码,现在unicode可以容纳100多万个符号,每个符号的编码都不一样,这下可统一了,所有语言都可以互通,一个网页页面里可以同时显示各国文字。 6、然而,unicode虽然统一了全世界字符的二进制编码,但没有规定如何存储啊,亲。x86和amd体系结构的电脑小端序和大端序都分不清,别提计算机如何识别到底是unicode还是acsii了。如果Unicode统一规定,每个符号用三个或四个字节表示,那么每个英文字母前都必然有二到三个字节是0,文本文件的大小会因此大出二三倍,这对于存储来说是极大的浪费。这样导致一个后果:出现了Unicode的多种存储方式。 7、互联网的兴起,网页上要显示各种字符,必须统一啊,亲。utf-8就是Unicode最重要的实现方式之一。另外还有utf-16、utf-32等。UTF-8不是固定字长编码的,而是一种变长的编码方式。它可以使用1~4个字节表示一个符号,根据不同的符号而变化字节长度。这是种比较巧妙的设计,如果一个字节的第一位是0,则这个字节单独就是一个字符;如果第一位是1,则连续有多少个1,就表示当前字符占用多少个字节。 8、注意unicode的字符编码和utf-8的存储编码表示是不同的,例如"严"字的Unicode码是4E25,UTF-8编码是E4B8A5,这个7里面解释了的,UTF-8编码不仅考虑了编码,还考虑了存储,E4B8A5是在存储识别编码的基础上塞进了4E25。 9、UTF-8 使用一至四个字节为每个字符编码。128 个 ASCII 字符(Unicode 范围由 U+0000 至 U+007F)只需一个字节,带有变音符号的拉丁文、希腊文、西里尔字母、亚美尼亚语、希伯来文、阿拉伯文、叙利亚文及马尔代夫语(Unicode 范围由 U+0080 至 U+07FF)需要二个字节,其他基本多文种平面(BMP)中的字符(CJK属于此类-Qieqie注)使用三个字节,其他 Unicode 辅助平面的字符使用四字节编码。 10、最后,要回答你的问题,常规来看,中文汉字在utf-8中到底占几个字节,一般是3个字节,最常见的编码方式是1110xxxx 10xxxxxx 10xxxxxx。
hhhhhhhaa 回答时间:2018-8-24 11:58:55
怎么能把中文输出了啊
skyi 回答时间:2018-8-24 12:02:06
这个和中文编码相关的,单片机发送对应汉子的编码,一般应该是UTF-8,串口调试助手的显示区要支持UTF-8编码就可以中文显示了,Securt是支持显示中文的

评分

参与人数 1蝴蝶豆 +3 收起 理由
zero99 + 3

查看全部评分

hhhhhhhaa 回答时间:2018-8-24 12:24:19
skyi 发表于 2018-8-24 12:02
这个和中文编码相关的,单片机发送对应汉子的编码,一般应该是UTF-8,串口调试助手的显示区要支持UTF-8编码 ...

有没有具体的示例啊
hhhhhhhaa 回答时间:2018-8-24 12:49:03
hhhhhhhaa 发表于 2018-8-24 12:24
有没有具体的示例啊

看到没有啊

串口调试助手显示内容

串口调试助手显示内容
重定义fputc函数 .png

发送的内容

发送的内容
ls164013 回答时间:2018-8-24 15:14:28
自己写一个printf输出函数,串口就可以打印中文了

评分

参与人数 1蝴蝶豆 +2 收起 理由
zero99 + 2

查看全部评分

ls164013 回答时间:2018-8-24 15:17:36
本帖最后由 ls164013 于 2018-8-24 15:24 编辑

int fputc(int c, FILE *fp)
{
        while((USART1->SR&1<<6)==0);
        USART1->DR=c;
        return 0;
}
下面是打印函数
sprintf((char *)a,"温度:%.1f C    湿度%.1f ",t,h);
            Show_HZORZF(0,0,a);
         
          //while((ADC1->SR&1<<1)==0);
    sprintf((char *)str,"温度:%d 湿度: %d¿光照:%d ",ADC[0],ADC[1],ADC[2]);
我这这样是可以串口输出汉字的

评分

参与人数 1蝴蝶豆 +3 收起 理由
zero99 + 3

查看全部评分

hhhhhhhaa 回答时间:2018-8-24 15:24:18
ls164013 发表于 2018-8-24 15:17
int fputc(int c, FILE *fp)
{
        while((USART1->SR&1SR&1

你这的乱码是什么啊?我知道为什么不能输出汉字了,在keil里的中文是UTF-8的编码格式,一个汉字用16进制表示的话是三个  举个例子吧
比如汉字“大”按照unicode转utf-8的编码规则,汉字使用3字节序列
所以套用三字节转换公式
0800 - FFFF
1110xxxx 10xxxxxx 10xxxxxx
其中用x代表的16位使用unicode相应的位来填充
0x5927转换为2进制0101 1001 0010 0111
填充到上面公式中的x中变成
11100101 10100100 10100111
用16进制表示为E5 A4 A7
而大的AIISC码用16进制表示的话就是  B4  F3
串口调试助手只能把AIISC码格式的16进制转换成汉字,不能把UTF-8类型的16进制汉字转换出来,所以才出现了乱码的现象
ls164013 回答时间:2018-8-24 15:27:43
hhhhhhhaa 发表于 2018-8-24 15:24
你这的乱码是什么啊?我知道为什么不能输出汉字了,在keil里的中文是UTF-8的编码格式,一个汉字用16进制 ...

第一次用这个 复制粘贴 出了点意外
hhhhhhhaa 回答时间:2018-8-24 15:30:04
我知道为什么不能输出汉字了,在keil里的中文是UTF-8的编码格式,一个汉字用16进制表示的话是三个  举个例子吧
比如汉字“大”按照unicode转utf-8的编码规则,汉字使用3字节序列
所以套用三字节转换公式
0800 - FFFF
1110xxxx 10xxxxxx 10xxxxxx
其中用x代表的16位使用unicode相应的位来填充
0x5927转换为2进制0101 1001 0010 0111
填充到上面公式中的x中变成
11100101 10100100 10100111
用16进制表示为E5 A4 A7
而大的AIISC码用16进制表示的话就是  B4  F3
串口调试助手只能把AIISC码格式的16进制转换成汉字,不能把UTF-8类型的16进制汉字转换出来,所以才出现了乱码的现象

用printf发送大

用printf发送大

这是串口调试助手显示出来的 接收到的大的 16进制 和发送过去的大 16进制的比较 ...

这是串口调试助手显示出来的 接收到的大的 16进制  和发送过去的大 16进制的比较 ...
hhhhhhhaa 回答时间:2018-8-24 15:31:01
ls164013 发表于 2018-8-24 15:27
第一次用这个 复制粘贴 出了点意外

加个QQ吧 1561648283
聊聊你怎么发中文的
hhhhhhhaa 回答时间:2018-8-24 16:33:13
ls164013 发表于 2018-8-24 15:14
自己写一个printf输出函数,串口就可以打印中文了

你试试看啊
feixiang20 回答时间:2018-8-24 23:19:51
确认你的编码格式
zealerlu 回答时间:2018-8-25 07:05:19
可以用友善串口调试助手,没测试过Unicode能否显示,但是界面比这些都优美些。

评分

参与人数 1蝴蝶豆 +2 收起 理由
zero99 + 2

查看全部评分

12下一页
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版