蓝桥杯单片机第九届省赛题详细讲解(彩灯控制器)

news/2024/7/24 12:42:01 标签: 单片机, 蓝桥杯, 嵌入式硬件

看之前强烈建议先自己做一遍!!!

    • 演示视频
    • 题目讲解
    • 完整程序
      • main.c
      • iic.c
      • iic.h
    • 工程文件

演示视频

题目讲解

首先还是从题目的程序框图准备起。
在这里插入图片描述
将程序框图的PCF8591(模拟输入),按键,数码管,EEPROM都先调试好。
在这里插入图片描述
在这里插入图片描述
然后看一下基本功能,让自己对整个流程有一个了解。再将彩灯控制的LED的四种模式准备好。这里我建议是用数组把状态保存起来

uchar LED1[8]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}; //模式1
uchar LED2[8]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe}; //模式2
uchar LED3[4]={0x7e,0xbd,0xdb,0xe7}; //模式3
uchar LED4[4]={0xe7,0xdb,0xbd,0x7e}; //模式4

然后继续看题,这里亮度调节需要调节pwm占空比来实现,所以先放一下,先把Rb2电压转化为4个等级。
在这里插入图片描述

uchar SMG_mode=0,move; //数码管模式定义,滑动变阻器Rb2定义
uchar shine_mode=0; //定义光强
void main(void)
{
	move=IIC_read(PCF8591_address,Move_address); //读取Rb2电阻阻值
	move=move*1.9608; //500/255 将255转换到500		
	if(move<125) shine_mode=1;
	else if(move<250) shine_mode=2;
	else if(move<375) shine_mode=3;
	else shine_mode=4;
}

然后接着往下看题目。首先设置
在这里插入图片描述
S7:控制LED流转,所以设置一个变量控制就行。
S6:为数码管状态切换按键,同样控制一个变量来切换。由于设置的单元需要闪,所以需要用到定时器。注意还要将数据存入eeprom,注意eeprom的大小为255,所以只能将时间变小存

uchar SMG_mode=0,move; //数码管模式定义,滑动变阻器Rb2定义
uchar shine_mode=0; //定义光强
char led_mode=1; //led模式
int gap=400; //设置流转间隔
void main(void)
{
	init(); //初始开发板
	Timer0Init(); //定时器初始化
//	threshold=IIC_read(0xA0,0x00); //上电读取EEPROM
	while(1)
	{
		if(SMG_mode==0)//数码管全息,模式一
		{
			SMG[0]=SMG[1]=SMG[2]=SMG[3]=SMG[4]=SMG[5]=20;
			SMG[6]=SMG[7]=20;
		}
		else if(SMG_mode==1) //模式控制,将模式赋值放定时器闪烁
		{
			SMG[0]=SMG[2]=21;
			SMG[3]=20;
			if(gap<1000){SMG[4]=20;
			SMG[5]=gap/100;SMG[6]=gap%100/10;SMG[7]=gap%10;} //1000以下显示
			else
			{SMG[4]=gap/1000;SMG[5]=gap%1000/100; //1000以上显示
			SMG[6]=gap%100/10;SMG[7]=gap%10;}
		}
		else if(SMG_mode==2)//间隔控制,将间隔赋值放定时器闪烁
		{
			SMG[0]=SMG[2]=21;SMG[1]=led_mode;
			SMG[3]=20;		
		}
		move=IIC_read(PCF8591_address,Move_address); //读取Rb2电阻阻值
		move=move*1.9608; //500/255 将255转换到500		
		if(move<125) shine_mode=1;
		else if(move<250) shine_mode=2;
		else if(move<375) shine_mode=3;
		else shine_mode=4;
		
		SMG_output();
		Dkey_scan();
	}
}
//定时器中断服务函数
uchar t=0; //定时器内计数变量
bit state=0; //闪控制变量
void time0() interrupt 1
{
	if((SMG_mode==1)||(SMG_mode==2))
	{
		t++;
		if(t>=160) //800ms/5ms=160次
		{
			t=0;
			if(state==0)
			{
				state=1;
				switch(SMG_mode)
				{
					case 1:SMG[1]=led_mode;break;
					case 2:
							if(gap<1000){SMG[4]=20;
							SMG[5]=gap/100;SMG[6]=gap%100/10;SMG[7]=gap%10;} //1000以下显示
							else
							{SMG[4]=gap/1000;SMG[5]=gap%1000/100; //1000以上显示
							SMG[6]=gap%100/10;SMG[7]=gap%10;}
						break;
				}
			}
			else
			{
				state=0;
				switch(SMG_mode)
				{
					case 1:SMG[1]=20;break;
					case 2:SMG[4]=SMG[5]=SMG[6]=SMG[7]=20;break;
				}
			}
		}
	}
}
bit ledenable=0; //led工作使能
void Dkey_scan(void)
{
	static uchar keybyte=0;
	static uchar key;
	if(((P3&0X0F)!=0X0F)&&(keybyte==0))
	{
		delay5ms();
		if((P3&0X0F)!=0X0F)
		{
			keybyte=1;key=P3&0x0f;
		}
	}
	if((keybyte==1)&&((P3&0X0F)==0X0F))
	{
		if((P3&0X0F)==0X0F)
		{
			switch(key)
			{
				case 0x0e: //S7
					if(ledenable==0)ledenable=1;
				else ledenable=0;
					break;
				case 0x0d: //S6
				if(SMG_mode==0){SMG_mode=1;P2=0X80;P0=0xff;}//进入设置关灯
				else if(SMG_mode==1)SMG_mode=2;
				else if(SMG_mode==2)
				{SMG_mode=0;IIC_write(AT24C02_address,0x00,led_mode);
				IIC_write(AT24C02_address,0x00,(gap/100));} //保存模式和间隔
					break;
				case 0x0b: //S5
				
					break;
				case 0x07: //S4
				
					break;
			}
			keybyte=0;
		}
	}
}

然后设置S5和S4两个按键。
在这里插入图片描述

bit ledenable=0; //led工作使能
void Dkey_scan(void)
{
	static uchar keybyte=0;
	static uchar key;
	if(((P3&0X0F)!=0X0F)&&(keybyte==0))
	{
		delay5ms();
		if((P3&0X0F)!=0X0F)
		{
			keybyte=1;key=P3&0x0f;
		}
	}
	if((keybyte==1)&&((P3&0X0F)==0X0F))
	{
		if((P3&0X0F)==0X0F)
		{
			switch(key)
			{
				case 0x0e: //S7
					if(ledenable==0)ledenable=1;
				else ledenable=0;
					break;
				case 0x0d: //S6
				if(SMG_mode==0)SMG_mode=1;
				else if(SMG_mode==1)SMG_mode=2;
				else if(SMG_mode==2)
				{SMG_mode=0;IIC_write(AT24C02_address,0x00,led_mode);
				IIC_write(AT24C02_address,0x00,gap);} //保存模式和间隔
					break;
				case 0x0b: //S5
				switch(SMG_mode)
				{
					case 1:led_mode++;break;
					case 2:gap+=100;break;
				}
				if(gap>=1200)gap=1200;  //限幅
				if(led_mode>=4)led_mode=4;  //限幅
					break;
				case 0x07: //S4
				switch(SMG_mode)
				{
					case 1:led_mode--;break;
					case 2:gap-=100;break;
				}
				if(gap<=400)gap=400;  //限幅
				if(led_mode<=1)led_mode=1;  //限幅
					break;
			}
			keybyte=0;
		}
	}
}

做完这些,接下来就是调节LED模式和亮度度,当然这些肯定是都需要在定时器里进行调节的。
间隔调节就是按照规定的间隔进行亮灭,亮度调节就是对间隔里面亮的状态进行细分,就是说你亮的时候不能一直亮,要闪亮,闪亮的间隔决定亮度。
这里我拿20ms的周期的pwm,分为四个占空比,11ms/20ms,14ms/20ms,17/20ms,20ms/20ms,这样就能控制亮度,

else //led模式
	{
		if(ledenable==1) //led使能有效才进行
		{
			tt++;
			if(tt>=gap)//gap到达间隔时间控制led间隔
			{
				tt=0;
				if(state==1){state=0;}
				else {state=1;num+=1;}
			}
			
			if(state==1)//亮
			{
				if(ttt<((shine_mode*3)+8))
				{
					switch(led_mode)
					{
						case 1:if(num>=8)num=0;led=LED1[num];break;//led1
						case 2:if(num>=8)num=0;led=LED2[num];break;//led2
						case 3:if(num>=4)num=0;led=LED3[num];break;//led3
						case 4:if(num>=4)num=0;led=LED4[num];break;//led4
					}
				}
				else
				{
					led=0xff;//模式0熄灭
				}
				ttt++;
				if(ttt>=20)ttt=0;
			}
			else//灭
			{
				led=0xff;//模式0控制间隔设置熄灭
			}
		}
	}

然后就是亮度等级显示了。
在这里插入图片描述

	while((key==0x07)&&(SMG_mode==0))
	{
		keybyte=0;key=P3&0x0f;
		SMG[6]=21;SMG[7]=shine_mode;
		SMG[0]=SMG[1]=SMG[2]=SMG[3]=SMG[4]=SMG[5]=20;		
		SMG_output();P2=0X80;P0=led;
	}

完整程序

main.c

#include <stc15f2k60s2.h>
#include "intrins.h"
#include "iic.h"

#define uchar unsigned char
#define uint unsigned int

#define AT24C02_address 0xA0 //EEPROM地址
#define PCF8591_address 0x90 //PCF地址
#define Move_address 0x03 //滑动变阻器地址

void init(void);
void Delay1ms(void);
void delay5ms(void);
void Dkey_scan(void);
void SMG_output(void);
void Timer0Init(void);
uchar tab[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,\
0x40,0x79,0x24,0x30,0x19,0x12,0x02,0x78,0x00,0x10,0xff,0xbf};
uchar SMG[8]={20,20,20,20,20,20,20,20};//初始显示10,全息数码管

uchar LED1[8]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}; //模式1
uchar LED2[8]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe}; //模式2
uchar LED3[4]={0x7e,0xbd,0xdb,0xe7}; //模式3
uchar LED4[4]={0xe7,0xdb,0xbd,0x7e}; //模式4

uchar SMG_mode=0; //数码管模式定义
uint move=0; //滑动变阻器Rb2定义
uchar shine_mode=0; //定义光强
char led_mode=1; //led模式
uint gap=400; //设置流转间隔
uchar led=0xff; //led显示变量
bit ledenable=0; //led工作使能
void main(void)
{
	init(); //初始开发板
	led_mode=IIC_read(AT24C02_address,0x00); //上电读取EEPROM
	delay5ms();
	gap=IIC_read(AT24C02_address,0x01);
	gap *=100;
	Timer0Init(); //定时器初始化
	while(1)
	{
		move=IIC_read(PCF8591_address,Move_address); //读取Rb2电阻阻值
		move=move*1.9608; //500/255 将255转换到500		
		if(move<125) shine_mode=1;
		else if(move<250) shine_mode=2;
		else if(move<375) shine_mode=3;
		else shine_mode=4;
		if(SMG_mode==0)//数码管全息,模式一
		{
			SMG[0]=SMG[1]=SMG[2]=SMG[3]=SMG[4]=SMG[5]=20;
			SMG[6]=SMG[7]=20;
		}
		else if(SMG_mode==1) //模式控制,将模式赋值放定时器闪烁
		{
			SMG[0]=SMG[2]=21;
			SMG[3]=20;
			if(gap<1000){SMG[4]=20;
			SMG[5]=gap/100;SMG[6]=gap%100/10;SMG[7]=gap%10;} //1000以下显示
			else
			{SMG[4]=gap/1000;SMG[5]=gap%1000/100; //1000以上显示
			SMG[6]=gap%100/10;SMG[7]=gap%10;}
		}
		else if(SMG_mode==2)//间隔控制,将间隔赋值放定时器闪烁
		{
			SMG[0]=SMG[2]=21;SMG[1]=led_mode;
			SMG[3]=20;		
		}
		
		P2=0X80;P0=led;
		SMG_output();
		Dkey_scan();
	}
}

uint t=0; //定时器内计数变量
uint tt=0;//控制led间隔变量
uint ttt=0; //控制led光强
bit state=0; //闪控制变量
uchar num=0; //led数组选择变量
void time0() interrupt 1
{
	if((SMG_mode==1)||(SMG_mode==2))
	{
		t++;
		if(t>=800) //800ms
		{
			t=0;
			if(state==0)
			{
				state=1;
				switch(SMG_mode)
				{
					case 1:SMG[1]=led_mode;break; //模式一控制模式设置闪烁
					case 2://模式二控制间隔设置闪烁
							if(gap<1000){SMG[4]=20;
							SMG[5]=gap/100;SMG[6]=gap%100/10;SMG[7]=gap%10;} //1000以下显示
							else
							{SMG[4]=gap/1000;SMG[5]=gap%1000/100; //1000以上显示
							SMG[6]=gap%100/10;SMG[7]=gap%10;}
					break;
				}
			}
			else
			{
				state=0;
				switch(SMG_mode)
				{
					case 1:SMG[1]=20;break;//模式一控制模式设置熄灭
					case 2:SMG[4]=SMG[5]=SMG[6]=SMG[7]=20;break;//模式二控制间隔设置熄灭
				}
			}
		}
	}
	else //led模式
	{
		if(ledenable==1) //led使能有效才进行
		{
			tt++;
			if(tt>=gap)//gap到达间隔时间控制led间隔
			{
				tt=0;
				if(state==1){state=0;}
				else {state=1;num+=1;}
			}
			
			if(state==1)//亮
			{
				if(ttt<((shine_mode*3)+8))
				{
					switch(led_mode)
					{
						case 1:if(num>=8)num=0;led=LED1[num];break;//led1
						case 2:if(num>=8)num=0;led=LED2[num];break;//led2
						case 3:if(num>=4)num=0;led=LED3[num];break;//led3
						case 4:if(num>=4)num=0;led=LED4[num];break;//led4
					}
				}
				else
				{
					led=0xff;//模式0熄灭
				}
				ttt++;
				if(ttt>=20)ttt=0;
			}
			else//灭
			{
				led=0xff;//模式0控制间隔设置熄灭
			}
		}
	}
}

void Dkey_scan(void)
{
	static uchar keybyte=0;
	static uchar key;
	if(((P3&0X0F)!=0X0F)&&(keybyte==0))
	{
		delay5ms();
		if((P3&0X0F)!=0X0F)
		{
			keybyte=1;key=P3&0x0f;
		}
	}
	while((key==0x07)&&(SMG_mode==0))
	{
		keybyte=0;key=P3&0x0f;
		SMG[6]=21;SMG[7]=shine_mode;
		SMG[0]=SMG[1]=SMG[2]=SMG[3]=SMG[4]=SMG[5]=20;		
		SMG_output();P2=0X80;P0=led;
	}
	if((keybyte==1)&&((P3&0X0F)==0X0F))
	{
		if((P3&0X0F)==0X0F)
		{
			switch(key)
			{
				case 0x0e: //S7
					if(ledenable==0)ledenable=1;
				else ledenable=0;
					break;
				case 0x0d: //S6
				if(SMG_mode==0){SMG_mode=1;P2=0X80;P0=0xff;}//进入设置关灯
				else if(SMG_mode==1)SMG_mode=2;
				else if(SMG_mode==2)
				{SMG_mode=0;IIC_write(AT24C02_address,0x00,led_mode);delay5ms();
				IIC_write(AT24C02_address,0x01,(gap/100));} //保存模式和间隔
					break;
				case 0x0b: //S5
				if(SMG_mode==1)led_mode++;
				else if(SMG_mode==2)gap+=100;
				if(gap>=1200)gap=1200;  //限幅
				if(led_mode>=4)led_mode=4;  //限幅
					break;
				case 0x07: //S4
				if(SMG_mode==1)led_mode--;
				else if(SMG_mode==2)gap-=100;
				if(gap<=400)gap=400;  //限幅
				if(led_mode<=1)led_mode=1;  //限幅
					break;
			}
			keybyte=0;
		}
	}
}

void Timer0Init(void)		//1毫秒@11.0592MHz
{
	AUXR |= 0x80;		//定时器时钟1T模式
	TMOD &= 0xF0;		//设置定时器模式
	TL0 = 0xCD;		//设置定时初值
	TH0 = 0xD4;		//设置定时初值
	TF0 = 0;		//清除TF0标志
	TR0 = 1;		//定时器0开始计时
	EA=1;ET0=1; //打开定时器中断
}

void SMG_output(void)
{
	uchar i;
	for(i=0;i<8;i++)
	{
	P2=(P2&0X1F)|0XC0;
	P0=(1 << i);
	P2=(P2&0X1F)|0XE0;
	P0=tab[SMG[i]];			
	Delay1ms();
	}
	P2=(P2&0X1F)|0XC0;
	P0=0XFF;
	P2=(P2&0X1F)|0XE0;
	P0=0XFF;		
}

void init(void)
{
	P2=(P2&0X1F)|0XA0;
	P0=0X00;
	P2=(P2&0X1F)|0X80;
	P0=0Xff;
	P2=(P2&0X1F)|0Xc0;
	P0=0Xff;
	P2=(P2&0X1F)|0Xe0;
	P0=0Xff;	
}

void Delay1ms(void)		//@11.0592MHz
{
	unsigned char i, j;

	_nop_();
	_nop_();
	_nop_();
	i = 11;
	j = 190;
	do
	{
		while (--j);
	} while (--i);
}

void delay5ms(void)		//@11.0592MHz
{
	unsigned char i, j;

	i = 54;
	j = 199;
	do
	{
		while (--j);
	} while (--i);
}

iic.c

#include "iic.h"

#define DELAY_TIME 40

void IIC_write(uchar hw_address,uchar reg_address,uchar num)
{
	IIC_Start();
	IIC_SendByte(hw_address&0xfe);
	IIC_WaitAck();
	IIC_SendByte(reg_address);
	IIC_WaitAck();
	IIC_SendByte(num);
	IIC_WaitAck();	
	IIC_Stop();	
}	

uchar IIC_read(uchar hw_address,uchar reg_address)
{
	uchar num;
	IIC_Start();
	IIC_SendByte(hw_address&0xfe);
	IIC_WaitAck();
	IIC_SendByte(reg_address);	
	IIC_WaitAck();
	IIC_Stop();
	
	IIC_Start();
	IIC_SendByte(hw_address|0x01);
	IIC_WaitAck();
	num=IIC_RecByte();
	IIC_WaitAck();
	IIC_Stop();	
	
	return num;
}

//
void IIC_Delay(unsigned char i)
{
    do{_nop_();}
    while(i--);        
}

//
void IIC_Start(void)
{
    SDA = 1;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 0;
    IIC_Delay(DELAY_TIME);
    SCL = 0;	
}

//
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;
}

//
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;  
}

//
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;    
}

iic.h

#ifndef _IIC_H
#define _IIC_H

#include "stc15f2k60s2.h"
#include "intrins.h"

#define uchar unsigned char
#define uint unsigned int

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); 
void IIC_write(uchar hw_address,uchar reg_address,uchar num);
uchar IIC_read(uchar hw_address,uchar reg_address);


#endif

工程文件

工程文件有注释


http://www.niftyadmin.cn/n/2237.html

相关文章

【网络原理】网络编程Socket套接字基础知识汇总

目录 1.网络初始&#xff1a; 2.网络编程&#xff1a; 3.UDP数据报套接字&#xff1a; 4.TCP流套接字&#xff1a; 1.网络初始&#xff1a; 局域网&#xff08;LAN&#xff09;广域网&#xff08;WAN&#xff09;IP地址用于定位主机的网络地址。端口号可以标识主机中发送数…

HTML之表格、表单

用 <table> <tr> <td>单元格</td> </tr> </table> 可以创建一个最简单的只有一行、一个单元格的表格。 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>HTML表格…

c++ queue用法 入门必看 超详细

1、queue的作用 说到queue&#xff0c;大家一定会想到stack&#xff0c;同样是简单易用的数据结构之一。queue就是队列的意思&#xff0c;像大家日常排队一样&#xff0c;先排的人先用。stack则是相反的&#xff0c;后来的先用。这就有了queue先进先出&#xff0c;stack后进先…

CMake中include的使用

CMake中的include命令用于从文件或模块(file or module)加载并运行CMake code。其格式如下&#xff1a; include(<file|module> [OPTIONAL] [RESULT_VARIABLE <var>][NO_POLICY_SCOPE]) 从给定的文件加载并运行CMake code。变量读写访问调用者的范围(Variable rea…

SpringBoot整合Javamail实现邮件发送

博客主页&#xff1a;踏风彡的博客 博主介绍&#xff1a;一枚在学习的大学生&#xff0c;希望在这里和各位一起学习。 所属专栏&#xff1a;SpringBoot学习笔记 文章创作不易&#xff0c;期待各位朋友的互动&#xff0c;有什么学习问题都可在评论区留言或者私信我&#xff0c;我…

宁德时代换挡,钠电池“接力”锂电池?

宁德时代以一己之力将钠电池推上了台前&#xff0c;钠电池量产被提上日程&#xff0c;钠离子电池产业化将进一步加速。 去年7月&#xff0c;宁德时代在发布会上宣布&#xff0c;其开发的第一代钠离子电池&#xff0c;电芯单体能量密度已经达到了160Wh/kg&#xff0c;为目前全球…

Java连接mysql数据库

文章目录一、Java连接mysql数据库1.1 流程1.2 一个测试连接的java程序二、优化&#xff1a;创建一个工具类2.1 存在的问题2.2 创建配置文件和工具类2.3 测试使用工具类进行CRUD操作一、Java连接mysql数据库 1.1 流程 java连接mysql大致需要这六步&#xff1a; 导入驱动包&am…

【uniapp小程序开发】—— 组件封装之【自定义弹窗】

文章目录&#x1f34b;前言&#xff1a;&#x1f34d;正文1、探讨需求封装popup自定义弹窗组件2、实战开发弹窗组件2.1 子组件内容 popup.vue文件2.2 父组件引用子组件3、效果图预览3.1 不使用具名插槽的原有样式效果3.2 使用具名插槽之后样式效果&#x1f383;专栏分享&#x…