根据原理图可以看出控制LED的三个口状态为:P27 = 1;P26 = 0;P25 = 0;即可使用LED控制;
同样使用到对应的锁存器,设置为此状态:P2 = P2 & 0x1f | 0x80;//打开锁存器
延时函数可以在STC里面设置生成c代码复制过来使用,根据实际需要的延时状态设置
#include
void Delay500ms() //@11.0592MHz
{
unsigned char i, j, k;
// _nop_();//这个复制过来的函数需要加相应的头文件,这里用不到可以删掉,不然会报错
// _nop_();
i = 22;
j = 3;
k = 227;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void LED_dalay()
{
int i;
P0 = 0x10;
P2 = P2 & 0x1f | 0x80;//打开锁存器
//0000 0001 << 1 = 0000 0010
for(i=0;i<8;i++)
{
P0 = ~(0x01 << i);
Delay500ms();
}
}
void main()
{
// //1111 1110
// //P0 = 0xfe;
// //P27 = 1;P26 = 0;P25 = 0;
// P0 = 0x10;//0001 0000 低电平点亮,灯呈现的状态
// //X & 0 = 1;
// //X | 1 = 1;与0为0,或1为1
// //0001 1111 0x1f
// //1000 0000 0x80
// P2 = P2 & 0x1f | 0x80;
while(1)
{
LED_dalay();
}
}
在赛点资源包里面找到这个共阳的数码管段码表
封装一个锁存器的函数方便使用
设置一个数码管显示函数,调用锁存器函数和直接编写相应的锁存器使用两种方法
另外再编写一个数码管小数的显示函数,方法和数码管显示函数相同,多了一个语句P0 = Seg_Table[duan] - 0x80;同样有调用锁存器函数和直接编写相应的锁存器使用两种方法
在运行板子时一开始蜂鸣器会一直响,可以使用这三种方法关闭:
1.//Latch(5,P0 & ~(0x01 << 6));//关闭蜂鸣器
2.//Latch(5,0x00);//可以实现同样的功能但是不建议使用
3.Latch(5,P0 & ~(5 << 4));//拉低P06和P04关闭,蜂鸣器和L10提示灯关闭
#include
//#include
//共阳数码管段码
//0xc0, //0
//0xf9, //1
//0xa4, //2
//0xb0, //3
//0x99, //4
//0x92, //5
//0x82, //6
//0xf8, //7
//0x80, //8
//0x90, //9
//0x88, //A
//0x83, //b
//0xc6, //C
//0xa1, //d
//0x86, //E
//0x8e //F
void Delay2ms() //@11.0592MHz
{
unsigned char i, j;
// _nop_();
// _nop_();
i = 22;
j = 128;
do
{
while (--j);
} while (--i);
}
code unsigned char Seg_Table[] = {0xc,0xf9,0xa4, 0xb0,0x99,0x92,0x82, 0xf8,0x80,0x90, 0x88, 0x83, 0xc6, 0xa1, 0x86, 0x8e };//这个code是把这个数据存放在另一个区,可以存放较大的数据
void Latch(unsigned char Yn,unsigned char value)//锁存器函数,使用锁存器可以调用这个函数
{
if(Yn < 4 || Yn > 7)//判断锁存器是否合法,这里实际只用到了Y4—Y7
{
return;//关闭所以不返回值
}
P0 = value;
P2 = P2 & 0x1f |(Yn << 5);//打开锁存器7,想打开哪一个锁存器就修改到对应的值
P2 = P2 & 0x1f;
}
void SEG(unsigned char wei,unsigned char duan)
{
// P0 = Seg_Table[duan];//段选值显示7(段选)
// P2 = P2 & 0x1f |(7 << 5);//打开锁存器7,想打开哪一个锁存器就修改到对应的值
// P2 = P2 & 0x1f;//这几行锁存器的和Latch(7,Seg_Table[duan]);效果一样
Latch(7,Seg_Table[duan]);
//P0 = 0x03;//选择什么位显示,这里选择前两位 //0000 0011=0x03
// P0 = 0x01<< (wei - 1);//按位显示
// P2 = P2 & 0x1f |(6 << 5);//打开锁存器6,想打开哪一个锁存器就修改到对应的值
// P2 = P2 & 0x1f;//这几行锁存器的和上面同理
Latch(6,0x01<< (wei - 1));
Delay2ms();
}
void SEG_Point(unsigned char wei,unsigned char duan)//带小数点的数码管显示
{
//P0 = Seg_Table[duan] - 0x80;//段选值显示7(段选),最高位dp为1,-0x80 是为了让dp小数点显示,详细可以看段码表数据
// P2 = P2 & 0x1f |(7 << 5);//打开锁存器7,想打开哪一个锁存器就修改到对应的值
// P2 = P2 & 0x1f;
//
// //P0 = 0x03;//选择什么位显示,这里选择前两位 //0000 0011=0x03
// P0 = 0x01<< (wei - 1);//按位显示
// P2 = P2 & 0x1f |(6 << 5);//打开锁存器6,想打开哪一个锁存器就修改到对应的值
// P2 = P2 & 0x1f;
Latch(7,Seg_Table[duan] - 0x80);
Latch(6,0x01<< (wei - 1));
Delay2ms();
}
void main()
{
//Latch(5,P0 & ~(0x01 << 6));//关闭蜂鸣器
//Latch(5,0x00);//可以实现同样的功能但是不建议使用
Latch(5,P0 & ~(5 << 4));//拉低P06和P04关闭,蜂鸣器和L10提示灯关闭
while(1)
{
SEG(1,1);
SEG(2,2);
SEG(3,3);
SEG(4,4);
SEG_Point(5,5);
SEG_Point(6,6);
SEG_Point(7,7);
SEG_Point(8,8);
}
这里循环太快数据显示会有问题,要添加2ms的延时函数进行延时调用
添加延时后显示正常
带小数点的状态,这里只设置了后四位来对比,可以根据使用情况修改
Latch(5,P0 & ~(0x01 << 6));//关闭蜂鸣器
按键,以及矩阵按键:
复制过来的延时函数里面//#include
添加一个void scan()函数设置独立按键,根据原理图所示设置按键对应的P口,如这里的按键S4为P33口
添加一个void scan_arr()函数设置矩阵按键,根据原理图所示设置按键对应的P口,如这里的按键S4为P33口
#include
//#include
//共阳数码管段码
//0xc0, //0
//0xf9, //1
//0xa4, //2
//0xb0, //3
//0x99, //4
//0x92, //5
//0x82, //6
//0xf8, //7
//0x80, //8
//0x90, //9
//0x88, //A
//0x83, //b
//0xc6, //C
//0xa1, //d
//0x86, //E
//0x8e //F
void Delay10ms() //@11.0592MHz
{
unsigned char i, j;
i = 108;
j = 145;
do
{
while (--j);
} while (--i);
}
void Delay2ms() //@11.0592MHz
{
unsigned char i, j;
// _nop_();
// _nop_();
i = 22;
j = 128;
do
{
while (--j);
} while (--i);
}
code unsigned char Seg_Table[] = {0xc,0xf9,0xa4, 0xb0,0x99,0x92,0x82, 0xf8,0x80,0x90, 0x88, 0x83, 0xc6, 0xa1, 0x86, 0x8e };//这个code是把这个数据存放在另一个区,可以存放较大的数据
void Latch(unsigned char Yn,unsigned char value)//锁存器函数,使用锁存器可以调用这个函数
{
if(Yn < 4 || Yn > 7)//判断锁存器是否合法,这里实际只用到了Y4—Y7
{
return;//关闭所以不返回值
}
P0 = value;
P2 = P2 & 0x1f |(Yn << 5);//打开锁存器7,想打开哪一个锁存器就修改到对应的值
P2 = P2 & 0x1f;
}
void SEG(unsigned char wei,unsigned char duan)
{
Latch(7,Seg_Table[duan]);
Latch(6,0x01<< (wei - 1));
Delay2ms();
}
void scan()
{
if(P33 == 0)
{
Delay10ms();
if(P33 == 0)//确定按下S4按键
{
SEG(1,4);
}
}
if(P32 == 0)
{
Delay10ms();
if(P32 == 0)//确定按下S5按键
{
SEG(1,5);
}
}
}
void scan_arr()
{
P44 = 0;P42 = 1;P35 = 1;P34 = 1;//扫描第一列
if(P33 == 0)
{
Delay10ms();
if(P33 == 0)//确定按下S4按键
{
SEG(1,4);
}
}
if(P32 == 0)
{
Delay10ms();
if(P32 == 0)//确定按下S5按键
{
SEG(1,5);
}
}
P44 = 1;P42 = 0;P35 = 1;P34 = 1;//扫描第一列
if(P33 == 0)
{
Delay10ms();
if(P33 == 0)//确定按下S8按键
{
SEG(1,8);//测试按键按下数码管显示状态是否正确,这里为S8按下第一位数码管显示8
}
}
if(P32 == 0)
{
Delay10ms();
if(P32 == 0)//确定按下S9按键
{
SEG(1,9);
}
}
}
void main()
{
Latch(4,0xff);//关闭LED灯
Latch(5,P0 & ~(5 << 4));//拉低P06和P04关闭,蜂鸣器、继电器和L10提示灯关闭
while(1)
{
//scan();
scan_arr();
}
这里特别要注意的是:这个J5的连接,在使用独立按键S4-S9时连接的为:2和3,即使用跳线帽BTN这一端的两个;而矩阵按键则是1和2,使用跳线帽连接KBD这一端的两个