有关题目






实现代码
注意:
代码实现方面:①从eeprom读出数据的过程中要注意不跟和从pcf8591读取数据一样,不需要rd_eeprom(0x00) * 5.0 / 255,同时在wr_eeprom(uchar addr, da)结束时延时5ms,能够使得eeprom有一个完整的写入周期。
源文件修改方面:官方给的iic.h中使用的时C51的头文件"reg52.h",我们需要修改为对应的15系列头文件"STC15F2K60S2.h",这样才可以使用其中的一些特殊位寄存器
底层代码:①rd_pcf8591()函数读取最后需调用iic.c中IIC_SendAck()发送非应答信号,即SDA发送一个高电平
重点:计数值加1 可以利用定时器每100ms扫描比较Vp与Vain3的关系,实现代码如下:
void N_handle()
{
if (Vain3 > Vp)
h = 1;
else
l = 1;
if (h & l)//上一次为高(低),这一次为(低)高,则N 加1
{
if (++N > 99)
N = 0;
h = 0;
l = 0;
}
}
main.c
写法一:
#include "STC15F2K60S2.h"
#include "iic.h"
#define uchar unsigned char
#define uint unsigned int
sbit L1 = P0^0;
sbit L2 = P0^1;
sbit L3 = P0^2;
uchar jm = 0;//界面初始化数据界面
code uchar tab[] = {
0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xff, 0xc1, 0x8c, 0xc8};
//U P n 11 12 13
bit flag_5s;//Vain3 < Vp是否够 5s,初始化为0,不够
bit flag_100ms;//电压数据采样时间,用来判断N是否增加
bit h, l;//类比零点存在法则判断是否存在Vp 值在100ms前为高(低) 100ms后变为低(高),存在则+1,反之N不变化
float Vp, Vain3;
uchar key_val = 21, inv_key;//inv_key无效按键数 初始化为0
uchar N;//计数值初始化0
void sys_init();
uchar rd_eeprom(uchar addr);//
uchar rd_pcf8591(uchar addr);
void wr_eeprom(uchar addr, da);
void key_scanf();
void key_handle();
void led();
void N_handle();
void dsp_smg_bit(uchar pos, val);
void display();//显示功能,分三个大块
void dsp_dat();
void dsp_para();
void dsp_cnt();
void delay_k(uchar t);//延时t * 10us
void Delay1ms(); //1ms@12.000MHz,延时1ms用于给足数码管足够显示时间
void Delay5ms(); //@12.000MHz
void main()
{
Vp = rd_eeprom(0x00) / 10.0; //存储的是Vp * 10
sys_init();
while(1)
{
Vain3 = rd_pcf8591(0x03) * 5.0 / 255;
if (flag_100ms)
{
flag_100ms = 0;
N_handle();
}
key_handle();
display();
led();
}
}
void N_handle()
{
if (Vain3 > Vp)
h = 1;
else
l = 1;
if (h & l)//上一次为高(低),这一次为(低)高,则N 加1
{
if (++N > 99)
N = 0;
h = 0;
l = 0;
}
}
void led()
{
if (flag_5s)
{
P2 = (P2 & 0x1f) | 0x80;
L1 = 0;
}
if (N % 2)
{
P2 = (P2 & 0x1f) | 0x80;
L2 = 0;
}
else
{
P2 = (P2 & 0x1f) | 0x80;
L2 = 1;
}
if (inv_key >= 3)
{
P2 = (P2 & 0x1f) | 0x80;
L3 = 0;
}
}
void key_handle()
{
key_scanf();
if (15 == key_val)//s12
{
key_val = 21;//初始化键值
if (++jm >= 3)
jm = 0;
inv_key = 0;//无效变为有效
if (2 == jm)//从参数界面保存Vp值
wr_eeprom(0x00, (uchar)(Vp * 10));
}
if (11 == key_val)//s13
{
key_val = 21;
if (2 == jm)
{
N = 0;
inv_key = 0;
}
else
inv_key++;
}
if (12 == key_val)//s17-
{
key_val = 21;
if (1 == jm)
{
if (Vp >= 0.5)
Vp -= 0.5;
else
Vp = 5.0;
inv_key = 0;
}
else
inv_key++;
}
if (16 == key_val)//s16+
{
key_val = 21;
if (1 == jm)
{
if (Vp <= 4.5)
Vp += 0.5;
else
Vp = 0.0;
inv_key

本文介绍了在2020年单片机省赛中,如何改进eeprom读取代码以避免rd_eeprom(0x00)的多余操作,并确保写入周期完整。同时,提供了两种写法的main.c,涉及矩阵键盘处理、定时器中断和Vp/Vain3比较的计数功能。重点在于调整iic.h和iic.c文件以适应STC15F2K60S2系列单片机,并优化数码管显示和按键交互。
最低0.47元/天 解锁文章
4万+

被折叠的 条评论
为什么被折叠?



