LQB新-2023省赛参考代码
#include "stc15.h"
#include <intrins.h>
#include <stdio.h>
//-----------------------------------------------
#define FOSC 12000000L
//#define T1MS (65536-FOSC/1000) //1T模式
#define T1MS (65536-FOSC/12/1000) //12T模式
//全局变量
unsigned int cnt1ms=0;
//RTC变量************
u8 shi=11;
u8 fen=43;
u8 miao=54;
void ReadRTC(void);
void InitRTC();
//报警
u8 led4flag=0;
u8 led5flag=0;
u8 led6flag=0;
//*****************温度
float wendumax=0.0f;//温度最大值
float wenducs=30.0f;//温度参数
float wendunow=0.0f;//温度当前值
float wenduave=0.0f;//温度平均值
float readwendu();
float wendusum=0;
float shidusum=0;
//PCF8591数据
u8 in1dat=0;
u8 readIN1(void);
//湿度计算,NE555
unsigned int freqnum=0;
unsigned int pulnum=0;
float shidunow=0.0f;
float shidumax=0.0f;
float shiduave=0.0f;
u8 shiduerr=0;
u8 flag555=0;
float shiducal(unsigned int x);
//界面
u8 pagex=1;//1 20 21 22 3 4
u8 pageM=0;
u8 page1cnt=0;
u8 page2cnt=0;
void PageProc();
unsigned int cfcnt=0;
u8cfshi=0;//触发小时
u8cffen=0;//触发分
u8 cfflag=0;//触发标记
void cfread();//触发判断
int cftime=0;
//延时函数
void Delay1ms()//@11.0592MHz
{
unsigned char i, j;
_nop_();
_nop_();
_nop_();
i = 11;
j = 190;
do
{
while (--j);
} while (--i);
}
void delayms(int nms)
{
int i;
for(i=0;i<nms;i++)
{
Delay1ms();
}
}
//****************************************************************************
//位操作函数,置1清0取反位读
u8 bitSet(u8 x,u8 pos)
{
u8 res;
res=x|(1<<pos);
return res;
}
u8 bitReset(u8 x,u8 pos)
{
u8 res;
res=x&(~(1<<pos));
return res;
}
u8 bitRead(u8 x,u8 pos)
{
u8 res;
res=(x>>pos)&0x01;
return res;
}
u8 bitTog(u8 x,u8 pos)
{
u8 res;
res=x^(1<<pos);
return res;
}
//***************
#define LEDY 0 //定义主时钟
#define SMGABCY 1 //定义主时钟
#define SMGCOMY 2 //定义主时钟
#define ULNY 3 //定义主时钟
#define NONE 4 //定义主时钟
//选择573
void s573(u8 x)
{
switch(x)
{
case LEDY:P2=(P2&0x1F)|0x80;break;//y4 100 0 0000
case SMGABCY:P2=(P2&0x1F)|0xe0;break;//7 1110 0000
case SMGCOMY:P2=(P2&0x1F)|0xc0;break;//6 1100 0000
case ULNY:P2=(P2&0x1F)|0xa0;break;//5
case NONE:P2=P2&0x1F;break;
}
}
//****************************************************************************
//LED函数
u8 u8LED=0xff;//熄灭初始
#define L1 0
#define L2 1
#define L3 2
#define L4 3
#define L5 4
#define L6 5
#define L7 6
#define L8 7
void LEDonx(u8 i) //0点亮
{
u8LED=bitReset(u8LED,i);
}
void LEDoffx(u8 i) //1mie
{
u8LED=bitSet(u8LED,i);
}
void LEDshow(u8 x) //1mie
{
u8LED=x;
s573(NONE);
P0=u8LED;
s573(LEDY);
s573(NONE);
}
//ULN2003函数,本次比赛没用到
u8 u8ULN=0x00;//熄灭初始
//void LEDonx(u8 i) //0点亮
//{
//u8LED=bitReset(u8LED,i);
//}
//void LEDoffx(u8 i) //1mie
//{
//u8LED=bitSet(u8LED,i);
//}
void ULNshow(u8 x) //1mie
{
u8ULN=x;
s573(NONE);
P0=u8ULN;
s573(ULNY);
s573(NONE);
}
//****************************************************************************
//数码管函数
//位码,段码,转换函数,显示函数,stc-isp
/************* 本地常量声明 **************/
u8 code t_display[]={ //标准字库,共阴极,要取反
// 0 1 2 3 4 5 6 7 8 9 A B C D E F
0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71,
//black - H J K L N o P U t G Q r M y
0x00,0x40,0x76,0x1E,0x70,0x38,0x37,0x5C,0x73,0x3E,0x78,0x3d,0x67,0x50,0x37,0x6e,
0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x46}; //0. 1. 2. 3. 4. 5. 6. 7. 8. 9. -1
u8 code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}; //位码
/************* 本地变量声明 **************/
u8 smgbuf[20];
u8 LED8[20]; //显示缓冲,改为20,实际显示需要前面8个
u8 display_index; //显示位索引
bit B_1ms; //1ms标志
/********************** 显示扫描函数 ************************/
void DisplayScan(void)
{
//消隐
s573(NONE);
P0=0x00;
s573(SMGCOMY);
s573(NONE);
s573(NONE);
P0=0xff;
s573(SMGABCY);
//显示,位选,段码
s573(NONE);
P0=T_COM[display_index];//我们不用取反,
s573(SMGCOMY);
s573(NONE);
P0=LED8[display_index];
s573(SMGABCY);
s573(NONE);
//锁存输出数据
if(++display_index >= 8) display_index = 0; //8位结束回0
}
//数码管转换函数
void smgconv(void)
{
u8 i=0,j=0,temp=0;
while(smgbuf[i]!=0)
{
switch(smgbuf[i])
{
case '0':temp=~t_display[0];break;//记得取反
case '1':temp=~t_display[1];break;//记得取反
case '2':temp=~t_display[2];break;//记得取反
case '3':temp=~t_display[3];break;//记得取反
case '4':temp=~t_display[4];break;//记得取反
case '5':temp=~t_display[5];break;//记得取反
case '6':temp=~t_display[6];break;//记得取反
case '7':temp=~t_display[7];break;//记得取反
case '8':temp=~t_display[8];break;//记得取反
case '9':temp=~t_display[9];break;//记得取反 CHPEF-
case 'C':temp=~0x39;break;//记得取反 CHPEF-
case 'H':temp=~0x76;break;//记得取反 CHPEF-
case 'P':temp=~0x73;break;//记得取反 CHPEF-
case 'E':temp=~0x79;break;//记得取反 CHPEF-
case '-':temp=~0x40;break;//记得取反 CHPEF-
case 'F':temp=~0x71;break;//记得取反 CHPEF-
default :temp=0xFF;break;
}
if(smgbuf[i+1]=='.')
{
temp &=0x7f;
i++;
}
i++;
LED8[j]=temp;
j++;
}
}
//****************************************************************************
//按键函数,S4 5 8 9
//长按,短按
int longtime=0;
int longtimeflag=0;
u8 key10ms=0;
u8 key10msflag=0;
u8 readkey(void)
{
//列扫描
u8 keyx=0;
P44=0;P42=1;
if(P32==0)keyx=5;
else if(P33==0)keyx=4;
P44=1;P42=0;
if(P32==0)keyx=9;
else if(P33==0)keyx=8;
return keyx;
}
//按键判断,重要步骤
void KeyProc()
{
//按键当前值,按键下降,按键上升沿,上一次按键值
static u8 keyold=0;
u8 keynow=0;
u8 keydown=0;
u8 keyup=0;
//去抖动,10ms进来一次
if(key10msflag==1)
{
key10msflag=0;
}
else return;
//读取按键
keynow=readkey();
keydown=keynow &(keyold^keynow);
keyup=~keynow &(keyold^keynow);
keyold=keynow;
if(keydown)
{
longtime=0;
//sprintf(smgbuf,"CCCCCCCC");//8个字符长度,占据完数码管
//smgconv();
}
//**********************************按键松开
if(keyup)
{
if(longtimeflag==0)
{
//只有4个按键,因此不用switch了
if(keyup==4)
{
//sprintf(smgbuf,"CHCHH%3d",(unsigned int)keyup);//8个字符长度,占据完数码管
//smgconv();
flag555=1;
freqnum=0;
pulnum=0;
TR0=1;//开始计算频率
//1 2 3 4
//if(pagex==0)pagex=1;
if(pagex==1)
{
pagex=2;
page2cnt=0;//每次切入界面2,都是温度回显
}
else if(pagex==2)pagex=3;
else if(pagex==3)pagex=1;
}
else if(keyup==5)
{
if(pagex==2)
{
if(page2cnt==0)page2cnt=1;
else if(page2cnt==1)page2cnt=2;
else if(page2cnt==2)page2cnt=0;
}
}
//***************************按键8
else if(keyup==8)
{
if(pagex==3)
{
if(wenducs>90)wenducs=0;
else wenducs++;
}
}
//*********************按键9
else if(keyup==9)
{
if(pagex==3)
{
if(wenducs<0)wenducs=90;
else wenducs--;
}
}
}
else if(longtimeflag==1)longtimeflag=0;
}
//***********长按,只有判断长按9
if(keyold &&(longtime>2000))
{
longtime=0;
longtimeflag=1;
if(keyold==9)
{
if(pagex==2 &&page2cnt==2)//回显界面,时间回显之
{
cfcnt=0;//触发次数=0
wendumax=0;
shidumax=0;
wenduave=0;
shiduave=0;
cfshi=0;
cffen=0;
}
}
}
}
void LedProc();
//定时器0计数脉冲NE555,定时器1滴答定时器
void main()
{
//测试一下上面的函数,一个个模块来。
float wendu=12.52;
int shidumax=123;
LEDshow(0xFF);//开机关闭LED灯;
ULNshow(0x00);//开机关闭LED灯;
sprintf(smgbuf," ");//清屏
smgconv();
readwendu();
readwendu();
delayms(800);//消除一开始的85
InitRTC();//初始化时间
TMOD |= 0x04; //设置定时器0为16位自动重装载外部记数模式
TH0 = TL0 = 0xff; //设置定时器0初始值
TR0 = 1; //定时器0开始工作,一开始不工作
ET0 = 1; //开定时器0中断
TMOD |= 0x00; //设置定时器为模式0(16位自动重装载)
TL1 = T1MS; //初始化计时值
TH1 = T1MS >> 8;
TR1 = 1; //定时器1开始计时
ET1 = 1; //使能定时器0中断
EA=1;
while(1)
{
if(cnt1ms%100==0)
{
ReadRTC();
LedProc();
LEDshow(u8LED);
if(pageM==0)KeyProc();//温湿度器件才有效
}
if(cnt1ms%250==0)//250ms采样一次频率数据
{
TR0=0;
freqnum=pulnum*4;
pulnum=0;
TR0=1;
}
if(cnt1ms%300==0)
{
PageProc();
}
if(cnt1ms%500==0)
{
if(cfflag==0)cfread();//触发判断
else if(cfflag==1 && cftime>3000)
{
cfflag=0;
cftime=0;
pageM=0;
}
}
}
}
//-----------------------------------------------
//中断服务程序.NE555
void t0int() interrupt 1 //中断入口
{
pulnum++;
}
/* Timer1 interrupt routine */
void tm1_isr() interrupt 3
{
cnt1ms++;
if(cnt1ms==60000)cnt1ms=0;//边界处理
cftime++;
key10ms++;
longtime++;
if(key10ms==10)
{
key10msflag=1;
key10ms=0;
}
DisplayScan(); //1ms扫描显示一位
}
//****************led_Proc
void LedProc()
{
if(led4flag==1)
{
u8LED=bitTog(u8LED,L4);//u8LED^(1<<L4);
}
else LEDoffx(L4);
if(pagex==1&&pageM==0)
{
LEDonx(L1);LEDoffx(L2);LEDoffx(L3);//L1=0,L2=1,L3=1;
}
else if(pagex==2&&pageM==0)
{
LEDoffx(L1);LEDonx(L2);LEDoffx(L3);
}
else if(pagex==3&&pageM==0)
{
LEDoffx(L1);LEDoffx(L2);LEDoffx(L3);
}
if(pageM==1)
{
LEDoffx(L1);LEDoffx(L2);LEDonx(L3);
}
}
//************************************界面
void PageProc()
{
//&******************时间见面
if(pagex==1&&pageM==0)
{
sprintf(smgbuf,"%02u-%02u-%02u",(unsigned int)shi,(unsigned int)fen,(unsigned int)miao);
smgconv();
}
//***********************************************
else if(pagex==2&&pageM==0)
{
if(page2cnt==0) //温度回显
{
if(cfcnt==0)
{
sprintf(smgbuf,"Cxxxxxxx");
//smgconv();
}
else
{
sprintf(smgbuf,"C %02u-%2.1f",(unsigned int)wendumax,wenduave);
//smgconv();
}
}
//*************湿度回显
else if(page2cnt==1) //湿度回显
{
if(cfcnt==0)
{
sprintf(smgbuf,"Hxxxxxxx");
//smgconv();
}
else
{
sprintf(smgbuf,"H %02u-%2.1f",(unsigned int)shidumax,shiduave);
//smgconv();
}
}
//*************
else if(page2cnt==2) //时间回显
{
if(cfcnt==0)
{
sprintf(smgbuf,"F%02uXXXXX",(unsigned int)cfcnt);
//smgconv();
}
else
{
sprintf(smgbuf,"F%02u%02u-%02u",(unsigned int)cfcnt,(unsigned int)cfshi,(unsigned int)cffen);
//smgconv();
}
}
}
//**********参数界面
else if(pagex==3&&pageM==0)
{
sprintf(smgbuf,"Pxxxxx%02u",(unsigned int)wenducs);
//smgconv();
}
//*********************温湿度界面
if(pageM==1)
{
if(shiduerr==1)
{
sprintf(smgbuf,"Exx%02u-AA",(unsigned int)wendunow);
//smgconv();
}
else if(shiduerr==0)
{
sprintf(smgbuf,"Exx%02u-%02u",(unsigned int)wendunow,(unsigned int)shidunow);
//smgconv();
}
}
smgconv();
}
//***********************************************************************ds1302
/*# DS1302代码片段说明
1. 本文件夹中提供的驱动代码供参赛选手完成程序设计参考。
2. 参赛选手可以自行编写相关代码或以该代码为基础,根据所选单片机类型、运行速度和试题
中对单片机时钟频率的要求,进行代码调试和修改。
*/
//
sbit SCK=P1^7;
sbit SDA=P2^3;
sbit RST=P1^3;
void Write_Ds1302(unsigned char temp)
{
unsigned char i;
for (i=0;i<8;i++)
{
SCK = 0;
SDA = temp&0x01;
temp>>=1;
SCK=1;
}
}
//
void Write_Ds1302_Byte( unsigned char address,unsigned char dat )
{
RST=0;_nop_();
SCK=0;_nop_();
RST=1; _nop_();
Write_Ds1302(address);
Write_Ds1302(dat);
RST=0;
}
//
unsigned char Read_Ds1302_Byte ( unsigned char address )
{
unsigned char i,temp=0x00;
RST=0;_nop_();
SCK=0;_nop_();
RST=1;_nop_();
Write_Ds1302(address);
for (i=0;i<8;i++)
{
SCK=0;
temp>>=1;
if(SDA)
temp|=0x80;
SCK=1;
}
RST=0;_nop_();
SCK=0;_nop_();
SCK=1;_nop_();
SDA=0;_nop_();
SDA=1;_nop_();
return (temp);
}
//**********初始化时间,设置时间,读取时间
void InitRTC()
{
//写保护关闭,往时分秒写数据,打开写保护
Write_Ds1302_Byte(0x8E,0x00);
Write_Ds1302_Byte(0x84,0x11);//时
Write_Ds1302_Byte(0x82,0x46);
Write_Ds1302_Byte(0x80,0x55);
Write_Ds1302_Byte(0x8E,0x80);
}
//不要求设置时间题目,只要读取时分秒
/********************** 读RTC函数 ************************/
void ReadRTC(void)
{
u8 tmp[3];
tmp[0] =Read_Ds1302_Byte(0x81);
tmp[1] =Read_Ds1302_Byte(0x83);
tmp[2] =Read_Ds1302_Byte(0x85);
miao = ((tmp[0] >> 4) & 0x07) * 10 + (tmp[0] & 0x0f);
fen = ((tmp[1] >> 4) & 0x07) * 10 + (tmp[1] & 0x0f);
shi = ((tmp[2] >> 4) & 0x03) * 10 + (tmp[2] & 0x0f);
}
//****************温度显示
/*# 单总线代码片段说明
1. 本文件夹中提供的驱动代码供参赛选手完成程序设计参考。
2. 参赛选手可以自行编写相关代码或以该代码为基础,根据所选单片机类型、运行速度和试题
中对单片机时钟频率的要求,进行代码调试和修改。
*/
//
sbit DQ=P1^4;//引脚
void Delay_OneWire(unsigned int t)
{
unsigned char i;
while(t--){
for(i=0;i<12;i++);
}
}
//
void Write_DS18B20(unsigned char dat)
{
unsigned char i;
for(i=0;i<8;i++)
{
DQ = 0;
DQ = dat&0x01;
Delay_OneWire(5);
DQ = 1;
dat >>= 1;
}
Delay_OneWire(5);
}
//
unsigned char Read_DS18B20(void)
{
unsigned char i;
unsigned char dat;
for(i=0;i<8;i++)
{
DQ = 0;
dat >>= 1;
DQ = 1;
if(DQ)
{
dat |= 0x80;
}
Delay_OneWire(5);
}
return dat;
}
//
bit init_ds18b20(void)
{
bit initflag = 0;
DQ = 1;
Delay_OneWire(12);
DQ = 0;
Delay_OneWire(80);
DQ = 1;
Delay_OneWire(10);
initflag = DQ;
Delay_OneWire(5);
return initflag;
}
//读取温度
float readwendu()
{
u8 LSB=0;
u8 MSB=0;
float x0,x1;
float wendux;
init_ds18b20();
Write_DS18B20(0xcc);
Write_DS18B20(0x44);//改为44,转换温度
init_ds18b20();
Write_DS18B20(0xcc);
Write_DS18B20(0xbe);//改为44,转换温度
LSB=Read_DS18B20();
MSB=Read_DS18B20();
//高位
x0=bitRead(MSB,2)*64+bitRead(MSB,1)*32+bitRead(MSB,0)*16;
x1=bitRead(LSB,7)*8+bitRead(LSB,6)*4+bitRead(LSB,5)*2+bitRead(LSB,4)*1+bitRead(LSB,3)*0.5+bitRead(LSB,2)*0.25;
wendux=x0+x1;
return wendux;
}
//PCF8591*****************************
#define SLAW 0x90
#define SLAR 0x91
sbit SDA0 = P2^1; //定义SDA PIN5
sbit SCL0 = P2^0; //定义SCL PIN6
/****************************/
void I2C_Delay(void) //for normal MCS51, delay (2 * dly + 4) T, for STC12Cxxxx delay (4 * dly + 10) T
{
u8 dly=5;
dly = 12000000 / 2000000UL; //按2us计算
while(--dly) ;
}
/****************************/
void I2C_Start(void) //start the I2C, SDA High-to-low when SCL is high
{
SDA0 = 1;
I2C_Delay();
SCL0 = 1;
I2C_Delay();
SDA0 = 0;
I2C_Delay();
SCL0 = 0;
I2C_Delay();
}
void I2C_Stop(void) //STOP the I2C, SDA Low-to-high when SCL is high
{
SDA0 = 0;
I2C_Delay();
SCL0 = 1;
I2C_Delay();
SDA0 = 1;
I2C_Delay();
}
//void S_ACK(void) //Send ACK (LOW)
//{
// SDA0 = 0;
// I2C_Delay();
// SCL0 = 1;
// I2C_Delay();
// SCL0 = 0;
// I2C_Delay();
//}
void S_NoACK(void) //Send No ACK (High)
{
SDA0 = 1;
I2C_Delay();
SCL0 = 1;
I2C_Delay();
SCL0 = 0;
I2C_Delay();
}
void I2C_Check_ACK(void) //Check ACK, If F0=0, then right, if F0=1, then error
{
SDA0 = 1;
I2C_Delay();
SCL0 = 1;
I2C_Delay();
F0 = SDA0;
SCL0 = 0;
I2C_Delay();
}
/****************************/
void I2C_WriteAbyte(u8 dat) //write a byte to I2C
{
u8 i;
i = 8;
do
{
if(dat & 0x80) SDA0 = 1;
else SDA0 = 0;
dat <<= 1;
I2C_Delay();
SCL0 = 1;
I2C_Delay();
SCL0 = 0;
I2C_Delay();
}
while(--i);
}
/****************************/
u8 I2C_ReadAbyte(void) //read A byte from I2C
{
u8 i,dat;
i = 8;
SDA0 = 1;
do
{
SCL0 = 1;
I2C_Delay();
dat <<= 1;
if(SDA0) dat++;
SCL0 = 0;
I2C_Delay();
}
while(--i);
return(dat);
}
/****************************/
//题目只要求光敏电阻,AIN1,读IIC器件的数据,可以看
u8 readIN1(void)
{
u8 res=0;
I2C_Start();
I2C_WriteAbyte(0x90);
I2C_Check_ACK();
I2C_WriteAbyte(0x01);//01
I2C_Check_ACK();
I2C_Start();
I2C_WriteAbyte(0x91);
I2C_Check_ACK();
res= I2C_ReadAbyte();
S_NoACK(); //send no ACK
I2C_Stop();
return res;
}
//湿度变化
float shiducal(unsigned int x)
{
float y;
if(x<200||x>2000)y=999.0f;
else
{
y=0.044f*x+1.12f;
}
return y;
}
void cfread()//触发判断
{
static u8 inold=0;
u8 innow=0,indown=0,inup=0;
static float wenduold=0.0f,shiduold=0.0f;
innow=readIN1();
if(innow<100)innow=1;//档光的情况下
else innow=0;//亮光的情况下0-255是大于90的
indown=innow &(inold ^innow);
inold=innow;
if(indown)
{
wendunow= readwendu();
if(wendunow>wenducs)
{
led4flag=1;
}
else
led4flag=0;
shidunow=shiducal(freqnum);
if(shidunow>95)
{
shiduerr=1;
LEDonx(L5);
}
else
{
shiduerr=0;
LEDoffx(L5);
}
if(shiduerr==0)
{
cfcnt++;
ReadRTC();
cfshi=shi;
cffen=fen;
//计算最大温度和最大湿度,平均温度,平均湿度。
if(wendunow>wenduold)wendumax=wendunow;
else wendumax=wenduold;
if(shidunow>shiduold)shidumax=shidunow;
else shidumax=shiduold;
if(cfcnt>=2)
{
if(wendunow>wenduold &&shidunow>shiduold)
LEDonx(L6);
else
LEDoffx(L6);
}
wenduold=wendunow;
shiduold=shidunow;
wendusum+=wendunow;
shidusum+=shidunow;
shiduave=shidusum/cfcnt;
wenduave=wendusum/cfcnt;
}
pageM=1;//调到温湿度界面页面
cfflag=1;//触发标记等于1
cftime=0;//温湿度界面时间请0,在定时器1中计算时间3秒
}
}
原文地址:https://blog.csdn.net/lmf666/article/details/136386612
免责声明:本站文章内容转载自网络资源,如侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!