单片机交流
直播中

shawon

12年用户 622经验值
私信 关注
[问答]

如何用STM32来实现温度控制系统仿真?

如何用STM32来实现温度控制系统仿真

回帖(1)

郭舒静

2021-10-22 15:18:48
本文介绍如何用STM32来实现温度控制系统仿真,如果看完你还不会各种测试,那你真的没救了
准备

仿真软件:Proteus 8.9
自行去 https://www.zdfans.com/ 搜索,Proteus
下载,并安装,汉化,注意要安装在C盘根目录
Proteus配置

本文以软件自带的Oven来实现温度反馈控制。
添加STM32F103C6模块,LM016L液晶屏模块,一个黄色LED,一个绿色LED,两个100欧姆电阻,一个继电器开关,一对123k和5k的分压电阻,以及Oven模块。
如果嫌麻烦可以在这里下载我添加完成的:下载
添加完成后,连线如图所示,注意ad网标的设置。





到这里先告一段落
STM32流程

通过STM32的自带的ADC获取温度,与设定值进行比较后,通过IO口控制Oven的电源驱动,从而实现负反馈。与此同时,STM32还需要控制液晶屏的信息显示。
请先下载程序代码:下载
首先使用STM32CubeMX(下载) 打开ATest.ioc。
前面都已配置完成,跳转到Project Manager,选择你喜欢的IDE进行STM32程序开发。






打开工程,先进行显示器控制的开发,代码在程序包中有,因此这里只做节选说明


void printFloat(float value)
{
        int tmp,tmp1;
    tmp = (int)value;
    tmp1=(int)((value-tmp)*10)%10;
        sprintf(&buff[0],"%d.%drn",tmp,tmp1);
}


因为程序由于一些原因不能打印浮点数,这里做一个浮点数的打印


void Delay_us(uint16_t us)
{
        uint16_t differ=0xffff-us-5;                                        //设定定时器计数器起始值
        __HAL_TIM_SET_COUNTER(&htim3,differ);
        HAL_TIM_Base_Start(&htim3);                                        //启动定时器
        while(differ<0xffff-6)                                                        //补偿,判断
                differ=__HAL_TIM_GET_COUNTER(&htim3);                        //查询计数器的计数值
        HAL_TIM_Base_Stop(&htim3);
}


这里是延迟函数


void LcdWriteCom(uint8_t com)
{
        Delay_us(20);
        GPIOA->BSRR = 0x00ff0000;
        GPIOA->BSRR = (com);
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_15,GPIO_PIN_RESET);        //LCDRS
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_14,GPIO_PIN_RESET);        //LCDRW
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_13,GPIO_PIN_RESET);        //LCDEN
        Delay_us(10);
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_13,GPIO_PIN_SET);        //LCDEN
        Delay_us(10);
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_13,GPIO_PIN_RESET);        //LCDEN
        Delay_us(10);
}


这里是液晶屏控制命令的函数,按照液晶屏的协议将控制命令写入液晶屏


void LcdWriteDate(uint8_t date)
{
        Delay_us(20);
        GPIOA->BSRR = 0x00ff0000;
        GPIOA->BSRR = (date);
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_15,GPIO_PIN_SET);        //LCDRS
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_14,GPIO_PIN_RESET);        //LCDRW
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_13,GPIO_PIN_RESET);        //LCDEN
        Delay_us(10);
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_13,GPIO_PIN_SET);        //LCDEN
        Delay_us(10);
        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_13,GPIO_PIN_RESET);        //LCDEN
        Delay_us(10);
}


这里是液晶屏数据命令的函数,当写入数据时,将会使液晶屏在指针位置显示字符


void LCD1602Init(void)
{
        uint8_t index=0;
        HAL_Delay(100);
        LcdWriteCom(0x38);  //设置16*2显示,8位数据接口
        LcdWriteCom(0x0c); //开显示,显示光标且闪烁
        LcdWriteCom(0x06);//写一个指针自动加一
        LcdWriteCom(0x01);//清屏  
        HAL_Delay(100);//延时一段时间时间,等待LCD1602稳定         
        LcdWriteCom(0x80);//设置第一行 数据地址指针
        for(index=0;index<13;index++)
                LcdWriteDate(table1[index]);  //写入数据
}


这里是液晶屏初始化函数,将第一行写入"Temperature:"


void LCD1602WriteCommand(uint8_t comm)
{
        LcdWriteCom(0xc0 + 14);
        LcdWriteDate(comm);  //写入数据   
}
————————————————



下面介绍主函数,凡是自己的代码都要在对应USER CODE中添加





初始化变量





初始化ADC,液晶屏





这里首先采集了ADC读数,随后调用printFloat将浮点数转为字符串,接着将字符串打印到第二行进行显示。随后根据现在的ADC读数来判断是否需要继续加热。这里设定温度为30度。
程序完成了,点击编译,生成hex文件。
运行

在Proteus中双击STM32部件,打开属性,添加hex文件,点击OK
对于MDK,hex路径在工程的MDK-ARMATestATest.hex





点击左下角运行,开始仿真





可见程序开始正常运行了。
但是会发现温度有时超过限制,而且比较慢,会上下抖动比较大,这里推荐使用PWM进行控制,效果最好,将在以后进行介绍。
举报

更多回帖

发帖
×
20
完善资料,
赚取积分