第14届蓝桥杯单片机模拟题(我也不知道第几套)(4T)

该代码段展示了在嵌入式系统中如何处理串口接收,包括初始化串口、接收中断处理以及数据的存储和显示。系统还包含按键扫描功能,用于改变显示内容。此外,有一个特定的函数用于从串口获取噪音数据并发送响应。

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

串口有点难住我了,这里借鉴了一下这位博主的串口接收函数

代码如下:

main.c

#include "sys.h"

void KeyScans(void)
{
	l3 = l4 = h4 = 1;
	h3 = 0;
	if(l4 == 0)    //s17
	{
		Delay(10);
		if(interface)
		{
			para -= 5;
			if(para == -5)
				para = 0;
		}
		while(l4 == 0)
		{
			DisplaySMG();
			Function();
		}
	}
	l3 = l4 = h3 = 1;
	h4 = 0;
	if(l3 == 0)   //s12
	{
		Delay(10);
		interface = ~interface;
		while(l3 == 0)
		{
			DisplaySMG();
			Function();
		}
	}
	
	if(l4 == 0)  //s16
	{
		Delay(10);
		if(interface == 1)
		{
			para += 5;
			if(para == 95)
				para = 90;
		}
		while(l4 == 0)
		{
			DisplaySMG();
			Function();
		}
	}
}

sys.c

#include "sys.h"

void Select_74hc138(uchar channel)
{
	switch(channel)
	{
		case 0: P2=P2&0X1F; break;
		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;
	}
}

void SystemInit(void)
{
	Select_74hc138(4);
	P0 = 0xff;
	Select_74hc138(5);
	P0 = 0x00;
	Select_74hc138(7);
	P0 = 0xff;
	Select_74hc138(0);
	UartInit();
	Timer0Init();
}

void Delay(uint ms)
{
	uint i;
	for(ms; ms > 0; ms--)
		for(i = 921; i > 0; i--);
}

void OperateLed(uchar who,status)
{
	static uchar temp;
	uchar zt;
	switch(status)
	{
		case 0:
			temp = temp|_crol_(0x01,who-1);
			P0 = ~temp;
			break;
		case 1: 
			zt = (~temp)|_crol_(0x01,who-1);
			if(temp != (~zt))
				temp = ~zt;
			P0 = zt;
			break;
	}
	Select_74hc138(4);
	Select_74hc138(0);
}

void Timer0Init(void)		//1毫秒@12.000MHz
{
	AUXR &= 0x7F;		//定时器时钟12T模式
	TMOD &= 0xF0;		//设置定时器模式
	TL0 = 0x18;		//设置定时初值
	TH0 = 0xFC;		//设置定时初值
	TF0 = 0;		//清除TF0标志
	TR0 = 1;		//定时器0开始计时
	ET0 = 1;
	EA = 1;
}


smg.c

#include "sys.h"

bit interface = 0;
uchar para = 65;
//extern bit uart_flag;//测试

code unsigned char Seg_Table[] =
{
	0xc0, //0
	0xf9, //1
	0xa4, //2
	0xb0, //3
	0x99, //4
	0x92, //5
	0x82, //6
	0xf8, //7
	0x80, //8
	0x90, //9
	0xc1  //U
};

void DisplayBit(uchar pos,value,bit dot)
{
	P0 = 0X00;
	Select_74hc138(6);
	Select_74hc138(0);
	
	if(dot)
		P0 = Seg_Table[value]&0x7f;
	else
		P0 = Seg_Table[value];
	Select_74hc138(7);
	Select_74hc138(0);
	
	P0 = _crol_(0x01,pos-1);
	Select_74hc138(6);
	Select_74hc138(0);
	
	Delay(1);
}

void DisplaySMG(void)
{
	if(interface == 0)
	{
		DisplayBit(1,10,0);
		DisplayBit(2,1,0);
		
//		DisplayBit(4,uart_flag,0);
		DisplayBit(6,noises/100,0);
		DisplayBit(7,noises/10%10,1);
		DisplayBit(8,noises%10,0);
		OperateLed(1,0);
		OperateLed(2,1);
	}
	else
	{
		DisplayBit(1,10,0);
		DisplayBit(2,2,0);
		
		DisplayBit(7,para/10,0);
		DisplayBit(8,para%10,0);
		OperateLed(1,1);
		OperateLed(2,0);
	}
}

key.c

#include "sys.h"

void KeyScans(void)
{
	l3 = l4 = h4 = 1;
	h3 = 0;
	if(l4 == 0)    //s17
	{
		Delay(10);
		if(interface)
		{
			para -= 5;
			if(para == -5)
				para = 0;
		}
		while(l4 == 0)
		{
			DisplaySMG();
			Function();
		}
	}
	l3 = l4 = h3 = 1;
	h4 = 0;
	if(l3 == 0)   //s12
	{
		Delay(10);
		interface = ~interface;
		while(l3 == 0)
		{
			DisplaySMG();
			Function();
		}
	}
	
	if(l4 == 0)  //s16
	{
		Delay(10);
		if(interface == 1)
		{
			para += 5;
			if(para == 95)
				para = 90;
		}
		while(l4 == 0)
		{
			DisplaySMG();
			Function();
		}
	}
}

 uart.c

#include "sys.h"
uchar urdat;
uchar buf[10];  //接收到的数据
uchar index = 0;  //buf数组索引
bit uart_flag = 0;  //接收完成标志 1:接收完成
uchar tem[32];

void UartInit(void)		//9600bps@12.000MHz
{
	PCON &= 0x7F;		//波特率不倍速
	SCON = 0x50;		//8位数据,可变波特率
	AUXR &= 0xBF;		//定时器1时钟为Fosc/12,即12T
	AUXR &= 0xFE;		//串口1选择定时器1为波特率发生器
	TMOD &= 0x0F;		//清除定时器1模式位
	TMOD |= 0x20;		//设定定时器1为8位自动重装方式
	TL1 = 0xFD;		//设定定时初值
	TH1 = 0xFD;		//设定定时器重装值
	ET1 = 0;		//禁止定时器1中断
	TR1 = 1;		//启动定时器1
	ES = 1;
	EA = 1;
}

void SendByte(uchar dat)
{
	SBUF = dat;
	while(TI == 0);
	TI = 0;
}

void SendString(uchar *buf)
{
	while(*buf != '\0')
	{
		SendByte(*buf++);
	}
}

void GetNoisesFromUart(void)
{
	if(uart_flag == 1)  //接收完成
	{
		uart_flag = 0;
		if(strcmp(buf,"Return") == 0)
		{
			sprintf((char *)tem,"\n\rNoises:%.1fdB\n\r",noises/10.0);
			SendString(tem);	
		}
	}
}

void Uart(void) interrupt 4
{
	//接收中断
	
	if(RI)
	{
		RI = 0;
		urdat = SBUF;
		if(urdat == 'n')
		{
			OperateLed(8,0);
			uart_flag = 1;
			buf[index] = 'n';
			buf[index+1] = '\0';
			index = 0;
			
		}
		else
		{
			buf[index++] = urdat; //接收的数据暂时存到数组里
		}
	}
}

sys.h

#ifndef _SYS_H_
#define _SYS_H_

#include <STC15F2K60S2.H>
#include "intrins.h"
#include "stdio.h"
#include "string.h"

sbit l3 = P3^5;
sbit l4 = P3^4;
sbit h3 = P3^2;
sbit h4 = P3^3;

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

#define uchar unsigned char
#define uint unsigned int

extern uchar rb2;
extern uint noises;
extern bit interface;
extern uchar para;

void Function(void);
void Select_74hc138(uchar channel);
void SystemInit(void);
void Delay(uint ms);
void OperateLed(uchar who,status);
void Timer0Init(void);
//smg.c
void DisplayBit(uchar pos,value,bit dot);
void DisplaySMG(void);
//key.c
void KeyScans(void);
//iic.c
void ReadAIN3(void);

//uart.c
void UartInit(void);
void SendByte(uchar dat);
void SendString(uchar *buf);
void GetNoisesFromUart(void);

#endif
<think>好的,我需要帮助用户找到2016年蓝桥杯单片机相关的模拟试题或资料。首先,用户提到了第七省赛的题目,涉及的外设有ADC、EEPROM、USART、LED、KEY等,还有温度数据获取的问题。用户提供的三个引用中,引用1和2提到了第七省题的外设使用和头文件,引用3讨论了温度数据读取时的中断问题及解决方法。 用户的需求是查找该年的模拟题或试题解析。根据现有信息,第七省赛的外设配置和问题点已经比较明确,但用户可能需要更具体的试题内容或解析步骤。我应该先整理现有的资料,然后看看能否扩展相关内容,比如提供试题解析的思路或常见考点。 首先,根据引用1,第七题目用到了多个外设,但没有PWM和蜂鸣器。这可能意味着考试重点在于这些外设的配置和综合应用。引用2中的头文件ledAndKey.h显示了LED和按键的函数,这可能涉及扫描按键和控制LED显示。引用3提到的温度数据读取问题,涉及中断处理,需在读取时关闭中断以避免数据错乱,但会影响PWM输出,这可能是一个考点,如何平衡中断和外围设备的稳定性。 接下来,用户可能需要试题的具体实现步骤或常见问题解析。例如,如何配置ADC和EEPROM,如何通过USART传输数据,以及如何处理中断冲突的问题。此外,代码示例可能有帮助,比如温度传感器的读取代码,或者按键扫描的实现。 另外,用户可能需要了解如何综合这些外设,例如,题目可能要求通过ADC读取电压,存储到EEPROM,再通过USART发送到上位机,同时用LED显示状态,按键控制流程。这类综合应用题的解析可能对用户有用。 由于现有引用中没有提供完整的试题内容,可能需要根据已有的信息构建可能的试题结构和解析方向。例如,结合外设的使用情况,分模块解析每个部分的实现方法,并指出常见错误,如中断处理当导致的数据错乱,以及如何优化代码结构。 最后,生成相关问题,帮助用户进一步深入学习,比如涉及外设配置的步骤,中断处理的注意事项,或者如何调试综合应用中的问题。</think>### 2016年蓝桥杯单片机第七省赛相关解析 根据提供的引用资料,2016年第七蓝桥杯嵌入式/单片机省赛题目主要涉及以下核心内容: --- #### **1. 外设模块与考点** - **主要外设**:ADC、EEPROM、USART、LED、按键(Key)[^1][^2]。 - **未涉及外设**:PWM、蜂鸣器。 - **综合应用**:需结合多个外设实现功能,例如数据采集(ADC)、存储(EEPROM)、通信(USART)、人机交互(LED和按键)[^1]。 --- #### **2. 关键问题与解决方法** - **温度数据读取问题**: - **现象**:温度显示跳动,因定时器中断打断了DS18B20的时序[^3]。 - **解决方案**:读取温度前关闭中断,读取完成后恢复中断。但需注意关闭中断可能导致PWM输出稳定,需权衡时序优先级。 - **代码结构优化**: - **示例**:通过模块化编程分离功能(如`ledAndKey.h`中封装LED显示和按键扫描函数)。 - **中断管理**:避免外设操作冲突,如USART通信与定时器中断的协调。 --- #### **3. 试题解析方向** 1. **ADC与EEPROM综合应用**: - 题目可能要求通过ADC采集电压,将结果存储到EEPROM,并通过USART发送到上位机。 - **关键点**:ADC校准、EEPROM页写入、USART协议设计。 2. **LED与按键交互**: - 实现按键控制LED显示模式(如流水灯、呼吸灯)。 - **参考代码**: ```c // 按键扫描函数(引用自头文件) unsigned char Key_Scan(void) { // 扫描逻辑 } ``` 3. **温度传感器与中断冲突**: - 题目可能要求读取DS18B20温度并显示,同时处理定时器中断。 - **解决方法**:在温度读取临界区关闭中断,完成后恢复。 --- #### **4. 模拟试题示例** **题目**:设计一个系统,通过ADC采集0~3.3V电压,按键控制采样频率,采样结果存储到EEPROM,并通过USART发送到PC端显示。 **解析步骤**: 1. 配置ADC为连续扫描模式,DMA传输数据。 2. 按键设置采样间隔(如1s/5s)。 3. EEPROM分页存储数据,避免频繁擦写。 4. USART使用中断接收PC指令,触发数据回传。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Wu小燕呀

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

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

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

打赏作者

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

抵扣说明:

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

余额充值