【蓝桥杯】第十三届蓝桥杯单片机第二次省赛 代码程序

第十三届蓝桥杯单片机第二次省赛

题目

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

hex文件

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

有问题直接找我哟。

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

代码

main.c

#include "stc15.h"
#include "iic.h"

#define u8 unsigned char
#define u16 unsigned int


//			12:U 		13:P		14:L		 15:A
u8 code Tab[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0xbf,0xff,0xc1,0x8c,0xc3,0x88};
u8 smg_cnt=0;
u8 disbuff[9];


u8	 key_buf=0;


u16 dat_rb2=0;


u8 rb2_s=45;
u8 rb2_x=5;

u8 rb2_s_temp=45;
u8 rb2_x_temp=5;
u8 mode=1;


bit d_flag=0;//0:未启动 1:连续测量	

bit rb2_flag=0;//0:上限	   1:下限

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


bit L8_flag=0;
u16 L8_tt=0;

//************************ 超声波模块 ********************************************

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 Delay5ms()		//@12.000MHz
{
	unsigned char i, j;

	i = 59;
	j = 90;
	do
	{
		while (--j);
	} while (--i);
}
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 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  if(mode==3&&smg_cnt==6)
	  P0=Tab[disbuff[smg_cnt+1]]&0x7f;
	else  if(mode==3&&smg_cnt==3)
	  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(++wave_tt>=200)
{
	wave_tt=0;
	wave_flag=1;
}


	 if(d_flag)
	 {
	 	if(++L8_tt>=100)
		{
			 L8_tt=0;
			 L8_flag=!L8_flag;
		}
		
	 }
	 else
	 {
	 	L8_tt=0;
		L8_flag=0;
	 }

}




void Show_SMG_11()
{
	disbuff[1]=11;
	disbuff[2]=11;
	disbuff[3]=11;
	disbuff[4]=11;
	disbuff[5]=11;
	disbuff[6]=11;
	disbuff[7]=11;
	disbuff[8]=11;
}

void Show_SMG_U()
{
	disbuff[1]=12;
	disbuff[2]=11;
	disbuff[3]=11;
	disbuff[4]=11;
	disbuff[5]=11;
	disbuff[6]=dat_rb2/100%10;
	disbuff[7]=dat_rb2/10%10;
	disbuff[8]=dat_rb2%10;
}
void Show_SMG_P()
{
	disbuff[1]=13;
	disbuff[2]=11;
	disbuff[3]=11;

	disbuff[4]=rb2_s/10%10;
	disbuff[5]=rb2_s%10;

	disbuff[6]=11;
	disbuff[7]=rb2_x/10%10;
	disbuff[8]=rb2_x%10;
}
void Show_SMG_L()
{
	disbuff[1]=14;
	disbuff[2]=11;
	disbuff[3]=11;
	disbuff[4]=11;
	disbuff[5]=11;
	if(d_flag)
	{
		if(distance>=100)
			disbuff[6]=distance/100%10;
		else
			disbuff[6]=11;
		if(distance>=10)
			disbuff[7]=distance/10%10;
		else		
		disbuff[7]=11;
	
		disbuff[8]=distance%10;
	   }
	  else
	  {
	  	disbuff[6]=15;
		disbuff[7]=15;
		disbuff[8]=15;
		}
}
void Show_SMG()
{
	if(mode==1)
		 Show_SMG_U();
	else  if(mode==3)
		 Show_SMG_P();
	else  if(mode==2)
		 Show_SMG_L();
}

//************************ 按键 程序 ********************************************
void Scan_Key()
{
	u8 key_temp;
	key_temp=P3&0x0f;
	if(key_temp!=0x0f&&!key_buf)
	{
		Delay5ms();
		key_temp=P3&0x0f;
		if(key_temp!=0x0f&&!key_buf)
		{
			  key_buf=key_temp;
		}	
	}
	else if(key_temp==0x0f&&key_buf)
	{
		Delay5ms();
		key_temp=P3&0x0f;
		if(key_temp==0x0f&&key_buf)
		{
			  switch(key_buf)
			  {
			  	case 0x07:
				if(mode<=3)
				{
					if(++mode>=4)
					{
						mode=1;
						rb2_s_temp=rb2_s;
						rb2_x_temp=rb2_x;
						}
					if(mode==3)
					{
						rb2_flag=0;
					}
				}
				break;//S4
				case 0x0b:
				rb2_flag=!rb2_flag;

				break;//S5
				case 0x0d:
				if(mode==3)
				{
					if(rb2_flag==0)
					{
						rb2_s+=5;
						if(rb2_s>50)
						{
							 rb2_s=5;
						}
					}
					else
					{
						 rb2_x+=5;
						if(rb2_x>=55)
						{
							 rb2_x=5;
						}
					}
				}
				break;//S6
				case 0x0e:
				if(mode==3)
				{
					if(rb2_flag==0)
					{
						rb2_s-=5;
						if(rb2_s==0)
						{
							 rb2_s=50;
						}
					}
					else
					{
						 rb2_x-=5;
						if(rb2_x==0)
						{
							 rb2_x=50;
						}
					}
				}
				break;//S7
			  }

			  key_buf=0;
		}
			
	}

}
//************************ 主函数 程序 ********************************************
void main()
{
	Init_System();
	 Timer0Init();
	Timer2Init();
	while(1)
	{
	
		dat_rb2=Read_PCF8591(3)/255.0*500;
		dat_rb2=Read_PCF8591(3)/255.0*500;
		if(wave_flag)
		{
			wave_flag=0;
			Wave_Distance();
		}


		if((dat_rb2<(rb2_s*10))&&(dat_rb2>(rb2_x*10)))
		{
			d_flag=1;
		}
		else
			d_flag=0;


		if(d_flag)
		{
			if(distance<=20)
			{
				   Write_PCF8591(51);
			}
			else   if(distance<=20)
			{
				  Write_PCF8591((distance/15.0-1/3.0)/5.0*255);
			}
			else
				   Write_PCF8591(255);
			
		
		Scan_Key();
		Show_SMG();

	  }
		led_state=0xff;


		if(mode==1)
			led_state&=0xfe;
		if(mode==2)
			led_state&=0xfd;
		if(mode==3)
			led_state&=0xfb; 
		if(L8_flag)
			led_state&=0x7f; 
		
		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_PCF8591(unsigned char channel);
void Write_PCF8591(unsigned char dat);
#endif

ii.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_PCF8591(unsigned char channel)
{
	unsigned char temp;
	IIC_Start();
	IIC_SendByte(0x90);
	IIC_WaitAck();
	IIC_SendByte(0x40|channel);
	IIC_WaitAck();

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



	return temp;
}
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();

}	

工程文件

链接: https://pan.baidu.com/s/1jRhMNoBHzTNTYV4sMyU6-A?pwd=ac8w
提取码: ac8w

B站视频

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

更多资料

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

省赛

国赛

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值