【蓝桥杯】第十届蓝桥杯单片机国赛 代码程序

第十届蓝桥杯单片机国赛 程序

题目

历届的省赛和国赛的题目我已经在前面的文章(点击查看)里给大家分享了(网盘资源),需要的话,直接去下载,我在这里就不再赘述了。

hex文件

读者下载这个文件然后用烧录软件直接烧入单片机就可以用了!

有问题直接找我哟。

链接: https://pan.baidu.com/s/1hFaq6CYmBsAeLWXMOafXWQ?pwd=kdrj
提取码: kdrj

代码

main.c

#include "stc15.h"
#include "onewire.h"
#include "string.h"
#include "stdio.h"
#include "iic.h"
#define u8 unsigned char
#define u16 unsigned int
#define get() (P3&0x3f)|((P4&0x10)<<3)|((P4&0x04)<<4)|0xc3

//	C:12	L:13		A:14		   P:15
u8 code Tab[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0xbf,0xff,0xc6,0xc7,0xc8,0x8c};
u8 smg_cnt=0;
u8 disbuff[9];
u16 T;
u16 T_tt=0;
bit T_flag=0;

u8	 key_buf=0;

sbit TX=P1^0;
sbit RX=P1^1;
u8 distance=0;
u16 wave_tt=0;
bit wave_flag=1;


u8  mode=1;


u8 change=0;

u8 T_canshu=30;
u8 distance_canshu=35;

u8 T_canshu_temp=30;
u8 distance_canshu_temp=35;

bit downflag1=0;
bit downflag2=0; 
u16 down_tt1=0;


bit downflag3=0;
bit downflag4=0; 
u16 down_tt2=0;
u8 rx_cnt=0;
u8 buffer[30];
u8 rx_tt=0;

bit rx_flag=0;
bit send_flag=0;


bit DAC_flag=1;


u8 led_state=0xff;
void Delay5ms()		//@12.000MHz
{
	unsigned char i, j;

	i = 59;
	j = 90;
	do
	{
		while (--j);
	} while (--i);
}

void Timer2Init(void)		//1毫秒@12.000MHz
{
	AUXR |= 0x04;		//定时器时钟1T模式
	T2L = 0x20;		//设置定时初值
	T2H = 0xD1;		//设置定时初值
	AUXR |= 0x10;		//定时器2开始计时
	IE2|=0x04;
	EA=1;
}
void Sevice_Timer2() interrupt 12
{
	 P2&=0x1f;

	 P0=0xff;
	P2|=0xe0;
	P2&=0x1f;

	P0=1<<smg_cnt;
	P2|=0xc0;
	P2&=0x1f;
	if(mode==1&&smg_cnt==5)
		P0=Tab[disbuff[smg_cnt+1]]&0x7f;
	else 
	P0=Tab[disbuff[smg_cnt+1]];

	P2|=0xe0;
	P2&=0x1f;

	if(++smg_cnt>=8)
	smg_cnt=0;


   if(++T_tt>=100)
{
	T_tt=0;
	T_flag=1;
}

	  if(++wave_tt>=200)
		{
			wave_tt=0;
			wave_flag=1;
		}
		if(!downflag1&&!downflag2&&key_buf==0xd7&&++down_tt1>1000)
		{
			downflag2=1;
			down_tt1=0;
		}
		if(!downflag3&&!downflag4&&key_buf==0xdb&&++down_tt2>1000)
		{
			downflag4=1;
			down_tt2=0;
		}

		if(rx_flag&&++rx_tt>=50)
	{
		
		rx_tt=0;
		send_flag=1;
	}


}
void Init_System()
{
	P2&=0x1f;
	P0=0xff;
	P2|=0x80;
	P2&=0x1f;

   	P0=0x00;
	P2|=0xa0;
	P2&=0x1f;

	P0=0xff;
	P2|=0xc0;
	P2&=0x1f;

	P0=0xff;
	P2|=0xe0;
	P2&=0x1f;

}



void Show_SMG_C()
{
	disbuff[1]=12;
	disbuff[2]=11;
	disbuff[3]=11;
	disbuff[4]=11;
	disbuff[5]=T/1000%10;
	disbuff[6]=T/100%10;
	disbuff[7]=T/10%10;
	disbuff[8]=T%10;
}
void Show_SMG_L()
{
	disbuff[1]=13;
	disbuff[2]=11;
	disbuff[3]=11;
	disbuff[4]=11;
	disbuff[5]=11;
	disbuff[6]=11;
	disbuff[7]=distance/10%10;
	disbuff[8]=distance%10;
}
void Show_SMG_P()
{
	disbuff[1]=15;
	disbuff[2]=11;
	disbuff[3]=11;
	disbuff[4]=mode-3;
	disbuff[5]=11;
	disbuff[6]=11;
	if(mode==5)
	{
	disbuff[7]=distance_canshu/10%10;
	disbuff[8]=distance_canshu%10;
	}
	else  if(mode==4)
	{
		disbuff[7]=T_canshu/10%10;
		disbuff[8]=T_canshu%10;
	}
}
void Show_SMG_A()
{
	disbuff[1]=14;
	disbuff[2]=11;
	disbuff[3]=11;
	if(change>=10000)
	disbuff[4]=change/10000%10;
	else
	disbuff[4]=11;
	if(change>=1000)
	disbuff[5]=change/1000%10;
	else
	disbuff[5]=11;
	if(change>=100)
	disbuff[6]=change/100%10;
	else
	disbuff[6]=11;
	if(change>=10)
	disbuff[5]=change/10%10;
	else
	disbuff[7]=11;


	disbuff[8]=change%10;
}
void Show_SMG()
{
	if(mode==1)
		Show_SMG_C();
	else  if(mode==2)
		Show_SMG_L();
	else  if(mode==3)
		Show_SMG_A();
	else  if(mode<=5)
		Show_SMG_P();
}
//************************ 矩阵键盘 ********************************************
void Scan_Key_B()
{
		u8 key_temp;
		P35=1;P34=1;P33=0;P32=0;
		key_temp=get();
		P35=0;P34=0;P33=1;P32=1;
		key_temp|=get();
		if(key_temp!=0xff&&!key_buf)
		{
			Delay5ms();
			P35=1;P34=1;P33=0;P32=0;
			key_temp=get();
			P35=0;P34=0;P33=1;P32=1;
			key_temp|=get();
			if(key_temp!=0xff&&!key_buf)
			{
				key_buf=key_temp;
			}
				
		}
		else if(key_temp==0xff&&key_buf)
		{
			Delay5ms();
			P35=1;P34=1;P33=0;P32=0;
			key_temp=get();
			P35=0;P34=0;P33=1;P32=1;
			key_temp|=get();
			if(key_temp==0xff&&key_buf)
			{
				switch(key_buf)
				{
				


					case 0xd7:
					if(!downflag1)
					{
						if(mode<=3)
						{
							if(++mode>=4)
								mode=1;
						}
						else if(mode<=5)
						{
							if(mode==4)
							mode=5;
							else if(mode==5)
							mode=4;
						}
					}
					downflag1=0;
					down_tt1=0;

					break;//S12
					case 0xdb:
					if(!downflag3)
					{
						if(mode<=3)
						{
							mode=4;
						}
						else if(mode<=5)
						{
							mode=1;
							
							if((distance_canshu_temp!=distance_canshu) ||(T_canshu_temp!=T_canshu))
							{
	
								change++;
								 distance_canshu_temp=distance_canshu;
								T_canshu_temp=T_canshu;
								Write_AT24C02(0x00,change/256);Delay5ms();
								Write_AT24C02(0x01,change%256);Delay5ms();
	
							}
						}
					  }
					 downflag3=0;
					down_tt2=0;
					break;//S13
				

					case 0xe7:
					if(mode==4)
					{  if(T_canshu>=2)
							T_canshu-=2;
					}
					else if(mode==5)
					{  if(distance_canshu>=5)
							distance_canshu-=5;
					}
					break;//S16
					case 0xeb:
					if(mode==4)
					{  if(T_canshu<=97)
							T_canshu+=2;
					}
					else if(mode==5)
					{  if(distance_canshu<=94)
							distance_canshu+=5;
					}
					break;//S17
					

				}
				key_buf=0;
			}
				
		}
	if(downflag2&&key_buf==0xd7)
	{
		downflag2=0;
		downflag1=1;
		change=0;
		Write_AT24C02(0x00,change/256);Delay5ms();
		Write_AT24C02(0x01,change%256);Delay5ms();	

	}
	if(downflag4&&key_buf==0xdb)
	{
		downflag4=0;
		downflag3=1;
		DAC_flag=!DAC_flag;
	 }

}
//************************ 串口 程序 ********************************************
void UartInit(void)		//4800bps@12.000MHz
{
	SCON = 0x50;		//8位数据,可变波特率
	AUXR |= 0x40;		//定时器1时钟为Fosc,即1T
	AUXR &= 0xFE;		//串口1选择定时器1为波特率发生器
	TMOD &= 0x0F;		//设定定时器1为16位自动重装方式
	TL1 = 0x8F;		//设定定时初值
	TH1 = 0xFD;		//设定定时初值
	ET1 = 0;		//禁止定时器1中断
	TR1 = 1;		//启动定时器1





	ES=1;
}
void Sevice_Uart() interrupt 4
{
	if(RI)
	{
		RI=0;
		buffer[rx_cnt++]=SBUF;
		rx_flag=1;
	}
}

void Send_data(u8 dat)
{
	 SBUF=dat;
	 while(!TI);
	 TI=0;
}
void Send_String(u8 *str)
{
	  while(*str)
	  {
	  	 Send_data(*str++);
	  }
}
//************************ 超声波程序 ********************************************
void Timer0Init(void)		//12微秒@12.000MHz
{
	AUXR &= 0x7F;		//定时器时钟12T模式
	TMOD &= 0xF0;		//设置定时器模式
	TL0 = 0xF4;		//设置定时初值
	TH0 = 0xFF;		//设置定时初值
	TF0 = 0;		//清除TF0标志
	TR0 = 0;		//定时器0开始计时
}

void Wave_Distance()
{
	u8 num=10;
	TX=0;
	TL0 = 0xF4;		//设置定时初值
	TH0 = 0xFF;		//设置定时初值
	TR0=1;
	while(num--)
	{
		 while(TF0==0);
		 TF0=0;
		 TX^=1;
	}
	TR0=0;
	TH0=0;
	TL0=0;
	TR0=1;
	while(RX && !TF0);
   	TR0=0;
	if(TF0)
	{
		TF0=0;
		distance=99;
	}
	else
	{
		distance=((TH0<<8)+TL0)*0.017;	
		if(distance>=99)
			distance=99;
	}


}
void main()
{
	Init_System();
	Timer0Init();
	UartInit();
	Timer2Init();
	if(Read_AT24C02(0x2f)!=123)
	{


		   Write_AT24C02(0x2f,123);Delay5ms(); 

		   change=0;
		 Write_AT24C02(0x00,change/256);Delay5ms();
		Write_AT24C02(0x01,change%256);Delay5ms();


	}
	 else 
	 {
	 	  change=Read_AT24C02(0x00)*256+Read_AT24C02(0x01);

	 }
	
	while(1)
	{
		 if(T_flag)
		{
		T_flag=0;
		T=(u16)(Read_Temp()*100) ;
		}
		if(wave_flag)
		{
			wave_flag=0;
			Wave_Distance();
		}
		if(send_flag)
		{
		  send_flag=0;
		  buffer[rx_cnt++]='\0';
		  rx_cnt=0;
		  rx_flag=0;

		  if(strcmp("ST\r\n",buffer)==0)
		  {
		  	  sprintf(buffer,"$%d,%d.%d\r\n",(u16)(distance),(u16)(T/100),(u16)(T%100));
		  }
		  else if(strcmp("PARA\r\n",buffer)==0)
		  {
		  	  sprintf(buffer,"$%d,%d\r\n",(u16)(distance_canshu_temp),(u16)(T_canshu_temp));
		  }
		  else
		  {
		  	 sprintf(buffer,"ERROR\r\n");
		  }
		  Send_String(buffer);
		  	
		}
		if(DAC_flag)
		{
			if(distance<=distance_canshu_temp)
			{
				Write_PCF8591(102);
			}
			else
			{
				Write_PCF8591(204);
			}
		}
		else
		{
			Write_PCF8591(20.4);
		}
		Scan_Key_B();
		Show_SMG();


		led_state=0xff;

		if(T>T_canshu_temp*100)
		{
			 led_state&=0xfe;
		}
		if(distance<distance_canshu_temp)
		{
			 led_state&=0xfd;
		}
		if(DAC_flag)
		{
			led_state&=0xfb;
		}
		P0=led_state;P2|=0x80;P0=led_state;P2&=0x1f;
	}
}

iic.h

#ifndef _IIC_H
#define _IIC_H

#include "stc15.h"
#include "intrins.h"

sbit SDA = P2^1;
sbit SCL = P2^0;

void IIC_Start(void); 
void IIC_Stop(void);  
bit IIC_WaitAck(void);  
void IIC_SendAck(bit ackbit); 
void IIC_SendByte(unsigned char byt); 
unsigned char IIC_RecByte(void); 
unsigned char Read_AT24C02(unsigned char addr);
void Write_AT24C02(unsigned char addr,unsigned char dat);
void Write_PCF8591(unsigned char dat);
#endif

iic.c

#include "iic.h"

#define DELAY_TIME 5

//I2C总线内部延时函数
void IIC_Delay(unsigned char i)
{
    do{_nop_();}
    while(i--);        
}

//I2C总线启动信号
void IIC_Start(void)
{
    SDA = 1;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 0;
    IIC_Delay(DELAY_TIME);
    SCL = 0;	
}

//I2C总线停止信号
void IIC_Stop(void)
{
    SDA = 0;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 1;
    IIC_Delay(DELAY_TIME);
}

//发送应答或非应答信号
void IIC_SendAck(bit ackbit)
{
    SCL = 0;
    SDA = ackbit;  					
    IIC_Delay(DELAY_TIME);
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SCL = 0; 
    SDA = 1;
    IIC_Delay(DELAY_TIME);
}

//等待应答
bit IIC_WaitAck(void)
{
    bit ackbit;
	
    SCL  = 1;
    IIC_Delay(DELAY_TIME);
    ackbit = SDA;
    SCL = 0;
    IIC_Delay(DELAY_TIME);
    return ackbit;
}

//I2C总线发送一个字节数据
void IIC_SendByte(unsigned char byt)
{
    unsigned char i;

    for(i=0; i<8; i++)
    {
        SCL  = 0;
        IIC_Delay(DELAY_TIME);
        if(byt & 0x80) SDA  = 1;
        else SDA  = 0;
        IIC_Delay(DELAY_TIME);
        SCL = 1;
        byt <<= 1;
        IIC_Delay(DELAY_TIME);
    }
    SCL  = 0;  
}

//I2C总线接收一个字节数据
unsigned char IIC_RecByte(void)
{
    unsigned char i, da;
    for(i=0; i<8; i++)
    {   
    	SCL = 1;
	IIC_Delay(DELAY_TIME);
	da <<= 1;
	if(SDA) da |= 1;
	SCL = 0;
	IIC_Delay(DELAY_TIME);
    }
    return da;    
}

unsigned char Read_AT24C02(unsigned char addr)
{
	unsigned char temp;
	IIC_Start();
	IIC_SendByte(0xa0);
	IIC_WaitAck();
	IIC_SendByte(addr);
	IIC_WaitAck();

	IIC_Start();
	IIC_SendByte(0xa1);
	IIC_WaitAck();
	temp=IIC_RecByte();
	IIC_SendAck(1);
	IIC_Stop();



	return temp;
}
void Write_AT24C02(unsigned char addr,unsigned char dat)
{
	
	IIC_Start();
	IIC_SendByte(0xa0);
	IIC_WaitAck();
	IIC_SendByte(addr);
	IIC_WaitAck();
	IIC_SendByte(dat);
	IIC_WaitAck();

	IIC_Stop();

 }
 void Write_PCF8591(unsigned char dat)
{
	
	IIC_Start();
	IIC_SendByte(0x90);
	IIC_WaitAck();
	IIC_SendByte(0x40);
	IIC_WaitAck();
	IIC_SendByte(dat);
	IIC_WaitAck();

	IIC_Stop();

}

onewire.h

#ifndef __ONEWIRE_H
#define __ONEWIRE_H

#include "stc15.h"

sbit DQ = P1^4;  

float Read_Temp();

#endif

onewire.c

#include "onewire.h"

//单总线内部延时函数
void Delay_OneWire(unsigned int t);  

//单总线写操作
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;
}

//DS18B20初始化
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 Read_Temp()
{
	unsigned int temp;
	unsigned char LSB,MSB;
	init_ds18b20();
	Write_DS18B20(0xcc);
	Write_DS18B20(0x44);

	init_ds18b20();
	Write_DS18B20(0xcc);
	Write_DS18B20(0xbe);
	LSB=Read_DS18B20();
	MSB=Read_DS18B20();

	temp=(MSB<<8)+LSB;

	return temp*0.0625;

		
}
void Delay_OneWire(unsigned int t)  
{
	t*=12;
	while(t--);
}

工程文件

链接: https://pan.baidu.com/s/1ZkBuFzP12RAM_QAc1QEltw?pwd=w3ak
提取码: w3ak

B站视频

我在B站发布了自己完成代码的过程,想要了解的朋友可以点击呀,第十届蓝桥杯单片机国赛

更多资料

需要更多蓝桥杯资料的小可爱,可以关注公众号【玛德花】,回复 蓝桥杯 即可,也随时欢迎需要交流的小伙伴一起讨论呀。
蓝桥杯

省赛

国赛

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值