蓝桥杯单片机第十四届省赛模拟题2程序参考

该程序使用单片机实现对温度和电压的读取与显示,通过DS18B20传感器读取温度,PCF8591读取电压,并用Nixie管进行数据显示。同时,系统支持串口通信,可以接收命令切换显示模式,并有按键控制和LED状态显示。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

十四届省赛模拟题2题目:

 

 

 

示例代码(reg52.h):

main.c

#include <REG52.H>
#include "stdio.h"
#include "onewire.h"
#include "IIC.h"
sfr P4=0xc0;
sfr AUXR=0x8e;

sbit L1=P3^0;
sbit L2=P3^1;
sbit L3=P3^2;
sbit L4=P3^3;

sbit R1=P4^4;
sbit R2=P4^2;
sbit R3=P3^5;
sbit R4=P3^4;
sbit LED3=P0^2;
sbit Buzzer=P0^6;
sbit jidian=P0^4;

void HC573(unsigned char channel,unsigned char value);
void Nixie(unsigned char column,unsigned char value);
void Delay(unsigned int x);
void System_init();

unsigned char code Array[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
0x80,0x90,0x88,0x80,0xc6,0xc0,0x86,0x8e,
0xbf,0x7f};
unsigned char code Array_pot[]={0x40,0x79,0X24,0x30,0x19,0x12,0x02,0x78,0x00,0x10};//小数点数组
unsigned int temperature=0;//温度
unsigned int voltage=0;//电压
float T_V=0;//电压计算
unsigned char Mode=0;//模式
unsigned char Data=0;//串口接受数据
unsigned char lock=0;//s4锁定标志位
unsigned char Send_Data=0;//S12按键控制位
unsigned char LED_stat=0xff;//LED控制
unsigned int Count=0;//0.1s计数
unsigned char Flag=0;//LED3翻转标志位
//温度读取
unsigned int Read_temperature()
{
	unsigned char LSB,MSB;
	unsigned int temp;
	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;
	temp=temp>>4;
	temp*=10;
	temp=temp+(LSB&0x0f)*0.625;
	return temp;
}
//温度显示
void Display_temp()
{
	Nixie(1,0xc1);
	Nixie(2,Array[1]);
	Nixie(3,0xff);
	Nixie(4,0xff);
	Nixie(5,0xff);
	Nixie(6,Array[temperature/100%10]);
	Nixie(7,Array_pot[temperature/10%10]);
	Nixie(8,Array[temperature%10]);
}
//电压读取
void PCF8591_Read()
{
	unsigned char t;
	I2CStart();
	I2CSendByte(0x90);
	I2CWaitAck();
	I2CSendByte(0x43);
	I2CWaitAck();
	
	I2CStart();
	I2CSendByte(0x91);
	I2CWaitAck();
	t=I2CReceiveByte();
	I2CWaitAck();
	I2CStop();
	T_V=t*(5.0/255);
	voltage=T_V*100;
}
//电压显示
void Display_V()
{
	Nixie(1,0xc1);
	Nixie(2,Array[2]);
	Nixie(3,0xff);
	Nixie(4,0xff);
	Nixie(5,0xff);
	Nixie(6,Array_pot[voltage/100%10]);
	Nixie(7,Array[voltage/10%10]);
	Nixie(8,Array[voltage%10]);
}

//串口
void UartInit(void)		//9600bps@12.000MHz
{
	SCON = 0x50;		
	AUXR &= 0xBF;		
	AUXR &= 0xFE;		
	TMOD &= 0x0F;		
	TL1 = 0xE6;			
	TH1 = 0xFF;			
	ET1 = 0;			
	TR1 = 1;
	EA=1;
	ES=1;	
}
//串口发送
void Send_Byte(unsigned char Dat)
{
	SBUF=Dat;
	while(TI==0);
	TI=0;
}

//按键扫描
void Key_Scan()
{
	L4=0;
	L1=L2=L3=1;
	R1=R2=R3=R4=1;
	if(R1==0)
	{
		if(lock==0)lock=1;
		while(R1==0)
		{
			if(Mode==0)
			{
				Display_temp();
			}
			else if(Mode==1)
			{
				Display_V();
			}
		}
	}
	else if(R3==0)
	{
		if(Send_Data==0)Send_Data=1;
		while(R3==0)
		{
			if(Mode==0)
			{
				Display_temp();
			}
			else if(Mode==1)
			{
				Display_V();
			}
		}
	}
	L3=0;
	L1=L2=L4=1;
	R1=R2=R3=R4=1;
	if(R1==0)
	{
		if(lock==1)lock=0;
		while(R1==0)
		{
			if(Mode==0)
			{
				Display_temp();
			}
			else if(Mode==1)
			{
				Display_V();
			}
		}
	}
}

//串口发送数据
void Send_uart()
{
	if(Send_Data==1)
	{
		if(Mode==0)
		{
			printf("TEMP(温度):%.1F°C\r\n",(float)temperature/10);
		}
		else if(Mode==1)
		{
			printf("Voltage(电压):%.2fV\r\n",(float)voltage/100);
		}
		Send_Data=0;
	}
}
//灯光控制
void LED_Run()
{
	P0=0xff;//防止继电器部分干扰
	if(Mode==0)
	{
		LED_stat=P0&0xfe;
		HC573(4,LED_stat);
	}
	else if(Mode==1)
	{
		LED_stat=P0&0xfd;
		HC573(4,LED_stat);
	}
	//LED3控制
	if(lock==1)
	{
		TR0=1;//锁定状态下开启定时器0.1s计数
		LED3=Flag;//通过定时器翻转引脚
		P2=(P2&0x1f)|0x80;
		P2=(P2&0x1f)|0x00;
	}
	else
	{
		TR0=0;
		LED3=1;
		P2=(P2&0x1f)|0x80;
		P2=(P2&0x1f)|0x00;
	}
}
//定时器0初始化
void Time_init()
{
	TMOD=0x01;
	TH0=(65535-1000)/256;
	TL0=(65535-1000)%256;
	TF0=0;
	ET0=1;
	EA=1;
}
//继电器蜂鸣器控制
void Buzzer_ctrl()
{
	if(temperature>=280)
	{
		P2=(P2&0x1f)|0xa0;
		jidian=1;
		P2=(P2&0x1f)|0x00;
	}
	else if(temperature<280)
	{
		P2=(P2&0x1f)|0xa0;
		jidian=0;
		P2=(P2&0x1f)|0x00;
	}
	if(voltage>=360)
	{
		P2=(P2&0x1f)|0xa0;
		Buzzer=1;
		P2=(P2&0x1f)|0x00;
	}
	else if(voltage<360)
	{
		P2=(P2&0x1f)|0xa0;
		Buzzer=0;
		P2=(P2&0x1f)|0x00;
	}
}

void main()
{
	System_init();
	while(1)
	{
		temperature=Read_temperature();//温度读取
		PCF8591_Read();//电压读取
		if(Mode==0)
		{
			Display_temp();
		}
		else if(Mode==1)
		{
			Display_V();
		}
		Send_uart();
		Key_Scan();
		Buzzer_ctrl();
		LED_Run();
	}
}
void Time0() interrupt 1
{
	TH0=(65535-1000)/256;
	TL0=(65535-1000)%256;
	if(lock==1)//锁定情况下计数,和上面开启定时器双重判断
	{
		Count++;
		if(Count>100)
		{
			Flag++;
			Flag%=2;
			Count=0;
		}
	}
}
void UART() interrupt 4
{
	if(RI)
	{
		Data=SBUF;
		if(lock==0)//未锁定情况下执行命令
		{
			if(Data=='A')
			{
				Mode=0;
			}
			else if(Data=='B')
			{
				Mode=1;
			}
		}
		RI=0;
	}
}

void System_init()
{
	HC573(5,0x00);
	HC573(4,0xff);
	Read_temperature();
	Delay(700);//上电之前延时700ms,将初始85读取掉
	Time_init();
	UartInit();

}
void HC573(unsigned char channel,unsigned char value)
{
	P2=(P2&0x1f)|0x00;
	P0=value;
	switch(channel)
	{
		case 4:P2=(P2&0x1f)|0x80;break;
		case 5:P2=(P2&0x1f)|0xa0;break;
		case 6:P2=(P2&0x1f)|0xc0;break;
		case 7:P2=(P2&0x1f)|0xe0;break;
		case 0:P2=(P2&0x1f)|0x00;break;
	}
	P2=(P2&0x1f)|0x00;
}
void Delay(unsigned int x)
{
	unsigned char i,j;
	while(x--)
	{
		i=11;
		j=190;
		do
		{
			while(--j);
		}while(--i);
	}
}
void Nixie(unsigned char column,unsigned char value)
{
	HC573(6,0x01<<(column-1));
	HC573(7,value);
	Delay(2);
	HC573(6,0x01<<(column-1));
	HC573(7,0xff);
}
//printrf重定向
char putchar(char ch)
{
	Send_Byte(ch);
	return ch;
}

 iic.h

#ifndef __IIC_H_
#define __IIC_H_

void I2CStart(void);
void I2CStop(void);
void I2CSendByte(unsigned char byt);
unsigned char I2CReceiveByte(void);
unsigned char I2CWaitAck(void);

#endif

iic.c

#include "reg52.h"
#include "intrins.h"

sbit scl=P2^0;
sbit sda=P2^1;

#define DELAY_TIME	5

//
static void I2C_Delay(unsigned char n)
{
    do
    {
        _nop_();_nop_();_nop_();_nop_();_nop_();
        _nop_();_nop_();_nop_();_nop_();_nop_();
        _nop_();_nop_();_nop_();_nop_();_nop_();		
    }
    while(n--);      	
}

//
void I2CStart(void)
{
    sda = 1;
    scl = 1;
	I2C_Delay(DELAY_TIME);
    sda = 0;
	I2C_Delay(DELAY_TIME);
    scl = 0;    
}

//
void I2CStop(void)
{
    sda = 0;
    scl = 1;
	I2C_Delay(DELAY_TIME);
    sda = 1;
	I2C_Delay(DELAY_TIME);
}

//
void I2CSendByte(unsigned char byt)
{
    unsigned char i;
	
    for(i=0; i<8; i++){
        scl = 0;
		I2C_Delay(DELAY_TIME);
        if(byt & 0x80){
            sda = 1;
        }
        else{
            sda = 0;
        }
		I2C_Delay(DELAY_TIME);
        scl = 1;
        byt <<= 1;
		I2C_Delay(DELAY_TIME);
    }
	
    scl = 0;  
}

//
unsigned char I2CReceiveByte(void)
{
	unsigned char da;
	unsigned char i;
	for(i=0;i<8;i++){   
		scl = 1;
		I2C_Delay(DELAY_TIME);
		da <<= 1;
		if(sda) 
			da |= 0x01;
		scl = 0;
		I2C_Delay(DELAY_TIME);
	}
	return da;    
}

//
unsigned char I2CWaitAck(void)
{
	unsigned char ackbit;
	
    scl = 1;
	I2C_Delay(DELAY_TIME);
    ackbit = sda; 
    scl = 0;
	I2C_Delay(DELAY_TIME);
	
	return ackbit;
}

//
//void I2CSendAck(unsigned char ackbit)
//{
//    scl = 0;
//    sda = ackbit; 
//	I2C_Delay(DELAY_TIME);
//    scl = 1;
//	I2C_Delay(DELAY_TIME);
//    scl = 0; 
//	sda = 1;
//	I2C_Delay(DELAY_TIME);
//}

onewire.h

#ifndef __ONEWIRE_H_
#define __ONEWIRE_H_

void Write_DS18B20(unsigned char dat);
unsigned char Read_DS18B20(void);
bit init_ds18b20(void);

#endif

onewire.c

#include "reg52.h"

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

### 关于蓝桥杯单片机第十四届模拟的试题与解题思路 #### 背景介绍 蓝桥杯作为一项面向全国高校学生的科技竞活动,其嵌入式方向尤其注重考察参者在硬件设计、软件编程以及综合应用能力方面的水平。对于第十四届蓝桥杯嵌入式比而言,无论是还是国阶段,都涉及到了大量的实际工程问题解决能力和算法优化技巧[^1]。 #### 题目概述 根据以往的比经验及公开资源分析,可以推测出第十四届蓝桥杯嵌入式中的部分典型题目可能包括但不限于以下几个方面: - **定时器中断控制LED闪烁频率调整** - 此类题目通常要求选手能够熟练掌握微控制器内部定时器模块的工作原理及其配置方法,并通过改变预分频系数或者计数值来实现不同周期下的信号输出。 ```c void TIM2_Init(void){ RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); //使能TIM2时钟 TIM_TimeBaseStructure.TIM_Period = 999; //自动重装载值ARR=999 TIM_TimeBaseStructure.TIM_Prescaler = 7199; //预分频数PSC=7199(即系统时钟8MHz/(7200+1)=1KHz) TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure); NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x0F;//抢占优先级设为最低 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x0F; //子优先级也设为最低 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE); //允许更新事件产生中断请求 TIM_Cmd(TIM2,ENABLE); //启动定时器 } ``` - **串口通信协议解析与数据帧处理** - 这一类型的考题旨在测试学生是否具备构建稳定可靠的数据传输链路的能力,同时还需要考虑如何高效地对接收到的信息包进行校验和提取有效负载等功能。 - **ADC采样电压计算并显示到LCD屏上** - ADC(模数转换器)的应用场景非常广泛,在工业自动化领域尤为常见。因此该知识点成为了历真题的重点覆盖区域之一。它不仅考验了考生对传感器接口电路的理解程度,还进一步延伸至图形界面开发层面的知识迁移运用之上。 #### 解题策略建议 针对上述提到的各种类型挑战项目,这里给出一些通用性的应对措施供参考学习之用: - 对于涉及到外设初始化的部分,务必仔细阅读官方手册文档说明,确保每一个寄存器位域设置准确无误; - 当面临复杂逻辑运算需求的时候,则应该尝试采用结构化程序设计理念将其拆分成若干个小功能单元分别予以实现后再组合起来形成最终解决方案; - 如果遇到需要长时间运行的任务调度情况的话,那么引入RTOS操作系统将会是一个不错的选择方案,这样既可以提高代码可维护性又能保障实时响应性能指标达成预期目标值范围之内。 ---
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

brepot

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值