历史上的今天
返回首页

历史上的今天

今天是:2024年08月25日(星期日)

正在发生

2021年08月25日 | STM32 串口通讯 发送 接收

2021-08-25 来源:eefocus

STM32的使用有利有弊,种类多---但是种类有太多,资料也是比较乱的,还有就是库的调用,经常忘记一些函数的使用------比如最常用的串口------


------------------------------------------------------------------------------USART ----设置-------------------------------


void USART1_Config(void)

{

        GPIO_InitTypeDef GPIO_InitStructure;

        USART_InitTypeDef USART_InitStructure;


        RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);  //USART1--时钟--与对应--GPIO--时钟开启

 


 //USART1的Tx---GPIO----PA.09----复用推挽输出

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

        GPIO_Init(GPIOA, &GPIO_InitStructure);


 //USART1的Rx---GPIO----PA.10----浮空输入

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

        GPIO_Init(GPIOA, &GPIO_InitStructure);


 //USART1的模式配置

        USART_InitStructure.USART_BaudRate = 115200;  //波特率

        USART_InitStructure.USART_WordLength = USART_WordLength_8b;  //串口传输的字长

        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); 


        USART_ClearFlag(USART1,USART_FLAG_TC); //清除串口1发送中断,否则第一个数不会发生


       USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); //接收中断使能----一般是在中断中需要用数组将接收到的数据保存起来时使用


        USART_Cmd(USART1, ENABLE);//使能USART1

}


-----------------------------------------/重定向c库函数----printf----到USART1-----------------------------------



int fputc(int ch, FILE *f)

{

        USART_SendData(USART1, (uint8_t) ch);  //发送一个字节数据到USART1 

        while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET); // 等待发送完毕


        return (ch);

}


-----------------------------------------/重定向c库函数----scanf------到USART1-----------------------

int fgetc(FILE *f)

{

        while (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET); //等待串口1输入数据 

        return (int)USART_ReceiveData(USART1);

}


------上面两个重定向----是将原来C库的函数与现在硬件的相对应---------


------我们也可以自己编写相似函数--------运用基本的两个函数----发送---USART_SendData()-----接收------USART_ReceiveData()---


-----------------************************************************-------比如--------%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%-----


void MyPrintfByte(unsigned char byte)   //串口发送一个字节

{

        USART_SendData(USART1, byte);        //通过库函数  发送数据

        while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);  //等待发送完成,检测 USART_FLAG_TC 是否置1       

}


---------------------------------------------

void MyPrintfStr(unsigned char *s)   //发送字符串 函数--指针--

{

        uint8_t i=0;  //定义一个局部变量  用来 发送字符串 ++运算


        while(s[i]!='')  // 每个字符串结尾 都是以  结尾的

        {

                USART_SendData(USART1,s[i]);       //通过库函数  发送数据

                while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET); //等待发送完成,检测 USART_FLAG_TC 是否置1

                 i++;                 //i++一次

        }

}


--------------------------------------------------

void MyPrintfArray(uint8_t send_array[],uint8_t num) //两个参数 一是数组内容, 二是数组长度1-255 

{

        uint8_t i=0;  //定义一个局部变量  用来 发送字符串 ++运算


         while(i        {

          USART_SendData(USART1,send_array[i]);        //通过库函数  发送数据

         while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);  //等待发送完成,检测 USART_FLAG_TC 是否置1

          i++;  //加一         

        }        

}






-----------------------------------------主函数中需要定义一个数组-----用来保存接收中断时接收到的数据--------------------


uint8_t RS232_RX_BUF[24];//数组

uint8_t RS232_RX_CNT=0;//数组实时下标


--------------------------到时候直接判断或者使用这个数组中的相对应位数据----------------------------------------




-----------------------------------------中断--------接收中断--------------------


----先申明在主函数中定义接收数组-----位外部变量----------------特别注意申明外部变量时不能赋值-------------


extern uint8_t RS232_RX_BUF[24];

extern uint8_t RS232_RX_CNT;


-------------------------------------------中断函数---------------------------


void USART1_IRQHandler(void) 

{

        uint8_t res;         

        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收到数据  

        {        

                res =USART_ReceiveData(USART1);  //读取接收到的数据   

               RS232_RX_BUF[RS232_RX_CNT]=res;  //记录接收到的值 

   


   //USART_SendData(USART1, res);//发送数据----回显


                RS232_RX_CNT++;      //接收数据增加1  

                if( RS232_RX_CNT>23)

                        RS232_RX_CNT=0;    

        }

        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) 

        { 

                 USART_ClearITPendingBit(USART1, USART_IT_RXNE);//清除接收标志位------每次接收完成都需要清除一下

        }


}


----------------------------------------------------------配置中断优先级------------------------


static void NVIC_Configuration(void)

{

        NVIC_InitTypeDef NVIC_InitStructure;

  

        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);//中断分组

  

        NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//中断源

        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;//抢占优先级

        NVIC_InitStructure.NVIC_IRQChannelSubPriority = ENABLE;

        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能

        NVIC_Init(&NVIC_InitStructure);

}


---------------------------------------------------------------在主函数中的配置调用--------------------------


int main(void)

 uint8_t i =0;

 uint8_t temp = 0;


 USART1_Config();

 NVIC_Configuration();

 

 while(1)

 {


              //如何从收到数据的数组中提取有用信息


 if(USART1_RX_BUF[0] == '^' && USART1_RX_CNT != 0)//判断数据头和数据个数

 {

 temp = 0;


                        //搜索数据尾


 for(i=0;i<20;i++)

 {

 if(USART1_RX_BUF[i] == '#')

 {

 temp = 1;

 }

 }


                        //数据头尾正确代表数据正确


 if(temp)

 {

 MyUsart1SendArray(USART1_RX_BUF,USART1_RX_CNT);

 //使用完数据后实时清空接收数据的数组 方便接收新数据

 for(i=0;i<23;i++)

 {

 USART1_RX_BUF[i] = 0;

 }

 USART1_RX_CNT = 0;//个数同样清零

 }

 }

 }

}



【--------------------------------------------最后  总结------------------------------------------------】


1.GPIO与时钟设置


2.模式配置选择-------注意打开中断允许-----USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);


3.中断函数的编写


4.中断优先级的设置


5.主函数调用


工程代码:https://download.csdn.net/download/xuxuechen/10426173


------------------------------------------------------------------485通讯---同理------------------

推荐阅读

史海拾趣

Holy Stone公司的发展小趣事

在追求业务发展的同时,Holy Stone始终注重品质提升和环境管理。公司陆续通过了ISO9001、ISO14001、QS9000、ISO/TS16949等国际认证,确保了产品的高品质和环保性。这些认证不仅提升了公司的品牌形象和市场竞争力,还为客户提供了更加可靠的产品和服务。此外,公司还积极履行社会责任,在环境治理、社会参与等方面做出了具体贡献。

e2v technologies公司的发展小趣事

在医疗领域,e2v也积极发挥其作用。公司针对放射治疗系统提供了价格合理、易于运输和安装的国内制造设备,支持中国各地医院和诊所的放射治疗工作。这些设备不仅提高了放射治疗的效果和效率,也为患者带来了更好的治疗体验。

Davicom公司的发展小趣事

面对不断变化的市场环境和客户需求,Davicom始终保持创新精神,不断寻求新的发展机遇。公司紧跟行业趋势,积极投入研发资源,推出了一系列具有创新性的产品。这些新产品不仅提升了公司的市场竞争力,也为公司的持续发展注入了新的动力。

B&K Precision公司的发展小趣事

随着业务的不断发展,科恩和班意识到电子测试和测量领域的巨大潜力。1951年,他们将公司更名为B&K Precision,并开始扩展业务,涉足其他电子测试和测量领域。工程师们凭借在电视测试设备领域的丰富经验,不断研发新产品,获得了多项专利,并推动公司迅速成为全球电子测量领域的领导者。

福斯特半导体(Foster)公司的发展小趣事

随着业务的不断发展,科恩和班意识到电子测试和测量领域的巨大潜力。1951年,他们将公司更名为B&K Precision,并开始扩展业务,涉足其他电子测试和测量领域。工程师们凭借在电视测试设备领域的丰富经验,不断研发新产品,获得了多项专利,并推动公司迅速成为全球电子测量领域的领导者。

CyOptics Inc公司的发展小趣事

除了数据通讯和电信市场外,CyOptics的光学产品还逐渐拓展至其他领域。例如,公司的光学组件已集成至光学收发器、转发器及线卡等设备中,为数据中心、高性能计算和企业存储网络等应用提供了强有力的支持。此外,CyOptics还积极开拓新兴市场,如数据库、地铁及长途干线市场等,为客户提供定制化的光学解决方案。

问答坊 | AI 解惑

想了解海尔单片机的请进

海尔单片机海尔单片机在华南区代理商,专门为客户开发设计各种单片机应用程序和电路板,目前拥有多种家电、工控、仪表成熟方案,欢迎广大用户合作。    开发过的大小家电类产品:电热水器、蒸汽清洁机、空调机控制板、洗衣机控制板、汽车 ...…

查看全部问答>

剖析切換式電源供應器的原理及常用元件規格

由於大部分電子器材(如電視機、音響、電腦…等)其內部元件所使用的電源均為直流電,故必須有電源供應器(或整流器)來把交流市電轉換成各種不同的直流電壓以使電器發揮功能。依其電路結構的不同,電源供應器可分為線性式和切換式電源供應器兩種, ...…

查看全部问答>

求助:一道Intel笔试题

题目:给出至少两种方法实现存储器管理,比如虚存。 题目中提到了虚存,那么答案应该不能再列举它。…

查看全部问答>

有谁用过r8c/38a? (瑞萨单片机)

有谁用过r8c/38a?(瑞萨单片机)…

查看全部问答>

希望能够给自己一个不要让自己堕落的机会

希望能够给我一个实用的机会………

查看全部问答>

瑞萨DIY活动获奖感言

万万没想到我获奖了。哈哈,来而不往非礼也,既然获奖了就发表下本人的获奖感言:首先十分谢谢EEWorld论坛,让我们众多的发烧友有机会近距离的学习瑞萨控制器。可能对多数的人来说瑞萨控制器只是听过但是从没有接触过,我们接触较多的还是51、ARM、 ...…

查看全部问答>