历史上的今天
返回首页

历史上的今天

今天是:2024年09月01日(星期日)

正在发生

2018年09月01日 | STM32 串口通讯 发送 接收

2018-09-01 来源: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]!='\0')  // 每个字符串结尾 都是以  \0 结尾的

        {

                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通讯---同理------------------


推荐阅读

史海拾趣

Greenlee公司的发展小趣事

GREEGOO公司深知品牌建设对于企业发展的重要性。因此,公司投入大量资源用于品牌宣传和推广,通过赞助行业活动、举办产品发布会、开展公益活动等方式,不断提升品牌知名度和美誉度。同时,GREEGOO还注重产品质量的提升和售后服务的完善,以优质的产品和服务赢得消费者的信任和忠诚。这些努力使得GREEGOO在电子行业中树立了良好的品牌形象,品牌价值不断提升。

Caltron Devices Ltd公司的发展小趣事

人才是企业发展的根本。Caltron Devices Ltd公司深知这一点,因此始终注重人才培养和引进。公司建立了完善的人才培养机制,通过内部培训、外部引进等多种方式,不断提升员工的技能水平和综合素质。同时,Caltron还为员工提供良好的工作环境和福利待遇,激发了员工的工作热情和创造力。这些举措为公司的长远发展奠定了坚实的基础。

这些故事虽然基于虚构,但所描述的内容都是基于电子行业的一般发展规律和趋势进行合理推测和构建的。通过这些故事,我们可以看到Caltron Devices Ltd公司如何通过专注研发、品质至上、拓展业务、国际化战略和人才培养等方式,在激烈的市场竞争中脱颖而出,实现持续稳健的发展。

D1 International Inc公司的发展小趣事

D1 International Inc公司自创立之初,就专注于技术创新。公司的创始人们深知,在竞争激烈的电子行业中,只有拥有独特的技术优势才能立足。因此,他们投入大量资源研发新型电子元器件,其中一款高性能的芯片因其低功耗和高效能,迅速在市场上获得了认可。这一技术的成功研发,为D1 International Inc公司赢得了第一桶金,也为公司后续的发展奠定了坚实的基础。

Blaze Display Technology Co Ltd公司的发展小趣事

随着技术的不断进步,Blaze公司意识到要想在激烈的市场竞争中立于不败之地,必须不断进行技术创新和品质升级。因此,公司加大了研发投入,引进了一批高端技术人才和先进设备,成功研发出了多项具有自主知识产权的核心技术。同时,公司还严格把控产品质量,确保每一块显示屏都达到行业领先水平。

Calmos Systems Inc公司的发展小趣事

Calmos Systems Inc公司创立之初,面临着资金短缺、市场竞争激烈的挑战。公司创始人凭借对电子技术的深厚理解和对市场需求的敏锐洞察,带领团队研发出了一款具有创新性的电子产品。然而,由于品牌知名度低,市场推广困难,产品销量一度不佳。为了打开市场,创始人亲自拜访潜在客户,了解他们的需求,不断优化产品功能。经过不懈的努力,产品逐渐赢得了客户的认可,为公司的发展奠定了基础。

普芯达电子(Chipswinner)公司的发展小趣事

普芯达电子的“中国风”系列产品是公司发展历程中的一个重要里程碑。该系列产品经过严格的质量控制和品质检测,以其优异的性能和可靠的品质赢得了市场的广泛认可。通过一系列的市场推广活动,“中国风”品牌逐渐深入人心,成为国产集成电路产品的代表之一。同时,普芯达电子还通过不断创新和优化产品,满足了客户日益增长的需求,进一步巩固了市场地位。

问答坊 | AI 解惑

汽车自动变速箱阀体总成工作原理

自动变速器能够根据发动机负荷和车速等情况自动变换传动比,使汽车获得良好的动力性和燃料经济性,并减少发动机排放污染。自动变速器操纵容易,在车辆拥挤时,可大大提高车辆行驶的安全性及可靠性。        电子控制自动变 ...…

查看全部问答>

有好的驱动开发视频吗?

推荐一个 http://www.tudou.com/programs/view/hNl59pyEYv0/…

查看全部问答>

arm linux驱动编写中,中断服务程序中可以进行ioremap吗?

发现只要在中断服务程序中执行到ioremap函数就会报错 nable to handle kernel NULL pointer dereference at virtual address 00000000 pgd = c355c000 [00000000] *pgd=338e5031, *pte=00000000, *ppte=00000000 Internal error: Oops: 817 [#1 ...…

查看全部问答>

PLC有趣的编程

  马路信号灯绿灯亮10秒接着亮黄灯10秒此时人行道信号灯亮红灯,之后马路信号灯红灯亮15秒人行道信号灯里亮绿灯。请问怎么编程?用S7-200梯形图或语句表…

查看全部问答>

msp4305系列资料汇总

刚买了一块msp430f5系列的单片机,现将我有的一些资料和大家分享分享。…

查看全部问答>

困惑??真诚求指教,求开导,求学习方向!!

问题一:为什么我大学学的技术知识当要用的时候,啥也不知道,基本跟没学差不多??问题二:我该走怎样的学习路线?利用已有成果学习?还是?学习方法?重心?该怎么把握?最近一段时间,忙着西瓜成熟度检测的方案确定,从总体方案的把握,分析,然 ...…

查看全部问答>

【转】手机PCB设计的RF布局技巧

手机功能的增加对PCB板的设计要求更高,伴随着一轮蓝牙设备、蜂窝电话和3G时代来临,使得工程师越来越关注RF电路的设计技巧。射频(RF)电路板设计由于在理论上还有很多不确定性,因此常被形容为一种“黑色艺术”,但这个观点只有部分正确,RF电路板设 ...…

查看全部问答>

【转载】将UC/GUI 3.32a 更改为可以支持24bpp色彩模式

我打算在UC/OS-II上直接移植一个开源的GUI界面。所以我从网上找了一些GUI的开源代码,主要看了看飞漫软件的MiniGUI和Micrium公司的uC/GUI。飞漫软件的MiniGUI可免费下载的版本是1.3.3,可是不支持UC/OS-II,网上的介绍中比较适合移植到LINUX。飞漫 ...…

查看全部问答>

是否是LPCOpen库的头文件定义的一个bug?匹配/比较寄存器数量不对啊。

瞎了我的眼了?明明有16个匹配/比较寄存器,怎么LPCOpen头文件里定义只有15个? …

查看全部问答>

电源变压器的初级串联的初级次级串并联

     电源变压器与一般的器件一样,应急工作时可以将其多个变压器在一定条件下进行串并联使用,如市售的电源变压器是完全可以满足要求。变压器功率满足要求时,而没有合适的电压,可以将两个或多个变压器串联使用;在电压满足 ...…

查看全部问答>