标题:
这是我提出功能要求,由豆包AI生成的程序代码,请各位看看,能用吗?
[打印本页]
作者:
ccczzzyy
时间:
2025-6-27 16:30
标题:
这是我提出功能要求,由豆包AI生成的程序代码,请各位看看,能用吗?
这个代码实现了 STC8H8K64U 芯片控制 T12-K 电烙铁的基本功能,包括:
温度检测与显示(通过 ADC 读取温度传感器,三位数码管显示)
温度调节(通过上下按键调整设定温度)
恒温控制(采用 PID 算法保持设定温度)
空闲休眠(5 分钟无操作自动降低温度)
代码使用了定时器 0 来实现数码管的动态扫描和 PWM 输出,使用 ADC 通道 0 来读取温度传感器的值。PID 控制参数 (Kp=20, Ki=0.1, Kd=5) 需要根据实际情况调整以获得最佳的温度控制效果。
硬件连接方面,代码假设温度传感器连接到 P1.0,加热器控制连接到 P1.1,三个按键分别连接到 P3.2-P3.4,数码管段选连接到 P2 口,位选连接到 P4.0-P4.2。实际应用中请根据具体电路连接调整引脚定义。
#include <STC8H8K64U.h>
#include <intrins.h>
// 引脚定义
#define TEMP_SENSOR_PIN P1.0 // 温度传感器输入引脚
#define HEATER_PIN P1.1 // 加热器控制引脚
#define KEY_UP_PIN P3.2 // 温度上升按键
#define KEY_DOWN_PIN P3.3 // 温度下降按键
#define KEY_POWER_PIN P3.4 // 电源/模式切换按键
#define LED_A P2.0 // 数码管段A
#define LED_B P2.1 // 数码管段B
#define LED_C P2.2 // 数码管段C
#define LED_D P2.3 // 数码管段D
#define LED_E P2.4 // 数码管段E
#define LED_F P2.5 // 数码管段F
#define LED_G P2.6 // 数码管段G
#define LED_DP P2.7 // 数码管小数点
#define DIG1 P4.0 // 数码管位选1
#define DIG2 P4.1 // 数码管位选2
#define DIG3 P4.2 // 数码管位选3
// 全局变量
unsigned int setTemperature = 300; // 默认设定温度300℃
unsigned int currentTemperature = 0; // 当前温度
unsigned char displayValue[3] = {0}; // 显示缓冲区
unsigned char displayIndex = 0; // 当前显示位
unsigned char workingMode = 0; // 工作模式:0-正常,1-休眠
unsigned int idleTime = 0; // 空闲计时器
unsigned int pwmDuty = 0; // PWM占空比
// 数码管段码表(0-9)
code unsigned char segTable[10] = {
0x3F, // 0
0x06, // 1
0x5B, // 2
0x4F, // 3
0x66, // 4
0x6D, // 5
0x7D, // 6
0x07, // 7
0x7F, // 8
0x6F // 9
};
// 函数声明
void SystemInit();
void Timer0Init();
void ADCInit();
void KeyScan();
void DisplayUpdate();
void DisplayDigit(unsigned char pos, unsigned char num);
unsigned int ReadTemperature();
void PIDControl();
void PWMOutput(unsigned int duty);
void main() {
SystemInit(); // 系统初始化
Timer0Init(); // 定时器0初始化(用于数码管扫描和PID控制)
ADCInit(); // ADC初始化(用于温度采集)
EA = 1; // 开总中断
while(1) {
KeyScan(); // 按键扫描
// 空闲计时
if (idleTime < 60000) // 最大计时约10分钟(60000/1000*100ms)
idleTime++;
// 休眠判断
if (idleTime >= 30000 && workingMode == 0) { // 5分钟无操作进入休眠
workingMode = 1;
setTemperature = 200; // 休眠温度设为200℃
}
// 温度读取和控制
currentTemperature = ReadTemperature();
PIDControl();
// 显示更新
DisplayUpdate();
// 延时
Delay1ms(100); // 100ms循环
}
}
// 系统初始化
void SystemInit() {
// 设置IO口模式
P1M1 = 0x00; P1M0 = 0x02; // P1.0输入,P1.1推挽输出
P2M1 = 0x00; P2M0 = 0xFF; // P2口全部推挽输出(数码管段选)
P3M1 = 0x00; P3M0 = 0x00; // P3口全部准双向口(按键输入)
P4M1 = 0x00; P4M0 = 0x07; // P4.0-P4.2推挽输出(数码管位选)
// 其他初始化
HEATER_PIN = 0; // 初始关闭加热器
}
// 定时器0初始化(用于数码管扫描和PID控制)
void Timer0Init() {
TMOD &= 0xF0; // 清除定时器0模式位
TMOD |= 0x01; // 设置定时器0为模式1(16位定时器)
TH0 = 0xFC; // 设置定时初值
TL0 = 0x66; // 设置定时初值(1ms@11.0592MHz)
ET0 = 1; // 使能定时器0中断
TR0 = 1; // 启动定时器0
}
// ADC初始化(用于温度采集)
void ADCInit() {
ADC_CONTR = 0x80; // 开启ADC,ADC时钟为系统时钟/2
ADC_RES = 0; // 清除ADC结果寄存器
ADC_RESL = 0;
}
// 按键扫描
void KeyScan() {
static unsigned char keyUpState = 1;
static unsigned char keyDownState = 1;
static unsigned char keyPowerState = 1;
// 检测温度上升按键
if (KEY_UP_PIN == 0 && keyUpState == 1) {
Delay1ms(20); // 消抖
if (KEY_UP_PIN == 0) {
if (workingMode == 0) { // 仅在正常模式下可调节温度
if (setTemperature < 450)
setTemperature += 5;
} else { // 休眠模式下按任意键唤醒
workingMode = 0;
setTemperature = 300; // 恢复默认温度
}
idleTime = 0; // 重置空闲计时
}
while (KEY_UP_PIN == 0); // 等待按键释放
keyUpState = 0;
} else if (KEY_UP_PIN == 1) {
keyUpState = 1;
}
// 检测温度下降按键
if (KEY_DOWN_PIN == 0 && keyDownState == 1) {
Delay1ms(20); // 消抖
if (KEY_DOWN_PIN == 0) {
if (workingMode == 0) { // 仅在正常模式下可调节温度
if (setTemperature > 150)
setTemperature -= 5;
} else { // 休眠模式下按任意键唤醒
workingMode = 0;
setTemperature = 300; // 恢复默认温度
}
idleTime = 0; // 重置空闲计时
}
while (KEY_DOWN_PIN == 0); // 等待按键释放
keyDownState = 0;
} else if (KEY_DOWN_PIN == 1) {
keyDownState = 1;
}
// 检测电源/模式切换按键
if (KEY_POWER_PIN == 0 && keyPowerState == 1) {
Delay1ms(20); // 消抖
if (KEY_POWER_PIN == 0) {
workingMode = !workingMode; // 切换工作模式
if (workingMode == 0)
setTemperature = 300; // 恢复默认温度
idleTime = 0; // 重置空闲计时
}
while (KEY_POWER_PIN == 0); // 等待按键释放
keyPowerState = 0;
} else if (KEY_POWER_PIN == 1) {
keyPowerState = 1;
}
}
// 显示更新
void DisplayUpdate() {
// 更新显示缓冲区
displayValue[0] = currentTemperature / 100;
displayValue[1] = (currentTemperature % 100) / 10;
displayValue[2] = currentTemperature % 10;
}
// 数码管显示函数
void DisplayDigit(unsigned char pos, unsigned char num) {
// 关闭所有数码管
DIG1 = 1;
DIG2 = 1;
DIG3 = 1;
// 清除段选
LED_A = 0;
LED_B = 0;
LED_C = 0;
LED_D = 0;
LED_E = 0;
LED_F = 0;
LED_G = 0;
LED_DP = 0;
// 设置位选
switch(pos) {
case 0: DIG1 = 0; break;
case 1: DIG2 = 0; break;
case 2: DIG3 = 0; break;
}
// 设置段选
unsigned char segData = segTable[num];
LED_A = (segData & 0x01) ? 1 : 0;
LED_B = (segData & 0x02) ? 1 : 0;
LED_C = (segData & 0x04) ? 1 : 0;
LED_D = (segData & 0x08) ? 1 : 0;
LED_E = (segData & 0x10) ? 1 : 0;
LED_F = (segData & 0x20) ? 1 : 0;
LED_G = (segData & 0x40) ? 1 : 0;
LED_DP = (segData & 0x80) ? 1 : 0;
}
// 读取温度
unsigned int ReadTemperature() {
unsigned int adcValue;
// 选择ADC通道(P1.0)
ADC_CONTR &= 0xF8;
ADC_CONTR |= 0x00; // 通道0
// 启动ADC转换
ADC_CONTR |= 0x40;
_nop_();
_nop_();
// 等待转换完成
while (!(ADC_CONTR & 0x20));
ADC_CONTR &= ~0x20; // 清除ADC完成标志
// 读取ADC结果
adcValue = (ADC_RES << 2) | (ADC_RESL & 0x03);
// 简单线性转换(实际应用中需要根据温度传感器特性校准)
// 假设10位ADC (0-1023)对应0-500℃
return (adcValue * 500) / 1023;
}
// PID控制算法
void PIDControl() {
static int error, lastError = 0;
static int integral = 0;
int derivative;
int output;
// 计算误差
error = setTemperature - currentTemperature;
// 积分项计算(带积分限幅)
integral += error;
if (integral > 1000) integral = 1000;
if (integral < -1000) integral = -1000;
// 微分项计算
derivative = error - lastError;
// PID输出计算(Kp=20, Ki=0.1, Kd=5,需要根据实际情况调整)
output = 20 * error + 0.1 * integral + 5 * derivative;
// 输出限幅(0-1000对应PWM占空比0-100%)
if (output > 1000) output = 1000;
if (output < 0) output = 0;
// 更新PWM占空比
pwmDuty = output;
// 更新上一次误差
lastError = error;
// 输出PWM
PWMOutput(pwmDuty);
}
// PWM输出控制
void PWMOutput(unsigned int duty) {
static unsigned int pwmCount = 0;
pwmCount++;
if (pwmCount > 1000) pwmCount = 0;
if (pwmCount < duty)
HEATER_PIN = 1; // 打开加热器
else
HEATER_PIN = 0; // 关闭加热器
}
// 定时器0中断服务函数(用于数码管扫描)
void Timer0_ISR() interrupt 1 {
TH0 = 0xFC; // 重新加载初值
TL0 = 0x66;
// 数码管动态扫描
DisplayDigit(displayIndex, displayValue[displayIndex]);
displayIndex++;
if (displayIndex >= 3) displayIndex = 0;
}
// 延时函数(@11.0592MHz)
void Delay1ms(unsigned int ms) {
unsigned int i, j;
for (i = 0; i < ms; i++)
for (j = 0; j < 110; j++);
}
作者:
WL0123
时间:
2025-6-27 21:11
100%玩不转
欢迎光临 (http://www.51hei.com/bbs/)
Powered by Discuz! X3.1