大家好,又见面了,我是你们的朋友全栈君。
LED流水灯的开发在51单片机中再常见不过了,主要是让大家掌握IO的操作是单片机控制最基本的要求。根据开发流程,我们先查看选型的单片机的资源和控制寄存器,然后在软件上实现控制。
在这里芯片我们采用STC15W404AS作为开发的硬件平台,在这里我们用该单片机的P1引脚来驱动LED实现流水灯的功能。
如图是LED连接的硬件电路,可以看到8颗LED是阳极通过限流电阻共同连接到VCC上的,也就是说当LED的阴极有高电平的时候LED灯是灭的,相反当 LED的阴极是低电平的时候LED就会亮起来,并且由于限流电阻为510欧姆,电源VCC电压是5V,红光LED的电压在1.9V左右,因此LED在亮起 来的时候大概有6mA的电流。有了上面的参数我们可以有目的的去看单片机的手册了。
从手册中可以看到,P1口可以设置成I/O(输入输出)模式,输入模式以及输出模式,并且每种模式的下能够承受的电流大小也不相同,在这里根据我们的电 路连接方式可以看到我们需要的是输入的模式,并且输入的电流最大在3mA左右,因此根据说手册上的说明,我们将P1口设置成准双向口就可以,当然如果想用 单片机端口来作为LED电流源驱动的话那就设置成推挽输出就可以。因此我们将P1端口的控制寄存器P1M1和P1M0相应的位(对应端口)设置成0即可, 也就是P1M1=0x00;P1M0=0x00;
将P1的控制位设置好后,我们就可以通过P1的直接赋值来控制LED的亮和灭了,其对应关系是:P1=~0x01,第一颗亮;~0x02第二颗亮……由此,我们可以有很多种方法来实现流水的功能。
1.数组的方法
由前面的结果类推我们可以制作一个表{0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}这样我们只要选择这个表格中 的一位数然后取反赋值给P1就能控制相应的灯的亮灭,因此我们把这个表格做成一个数组然后进行循环来实现LED的流水功能。其程序如下:
#include
unsigned char code LED[]={0x01,0x02,0x04,0x08,0x10,0x20};
void delay()
{
unsigned int i;
for(i=0;i
}
void IO_Config()
{
P1M0=0x00;//设置P1端口的类型
P1M1=0x00;//可以根据不同的电路来修改
}
void main()
{
unsigned char j;
IO_Config();
while(1)
{
for(j=0;j
{
P1=~LED[j];
delay();
}
}
}
复制代码
由于我们选择的这款芯片P1引脚只有6个(P1.0~P1.5)所以我们只设置了6个值,对于8个的也使用只需要将数组拓展到8位即可,编译后可以通过下载到单片机调试即可
2.移位的方法
在C语言中我们有个>(左移)功能,这也在单片机中常用到,值得注意的是,当数据移动后会用0来填补,这点要非常注意,因此上面的程序可以简化为下面的结构:
#include
//unsigned char code LED[]={0x01,0x02,0x04,0x08,0x10,0x20};
void delay()
{
unsigned int i;
for(i=0;i
}
void IO_Config()
{
P1M0=0x00;//设置P1端口的类型
P1M1=0x00;//可以根据不同的电路来修改
}
void main()
{
unsigned char j,temp;
IO_Config();
while(1)
{
temp=0xfe;
for(j=0;j
{
//P1=~LED[j];
P1=temp;
temp=(temp
delay();
}
}
}
复制代码
同样将改程序编译后生成HEX文件烧写到单片机内然后就可以调试了
3.循环移位法
从 上面可以看到用C语言的移位的方法移动后都自动会用0来填补,这样我们必须在后面通过或的方法来补齐原来的数据,我们知道在汇编语言中有RR和 RL(rotate right和rotateleft)的移位方法,这种以为的好处是从移动过程中数据是圆形的移位,这样不需要填补就可以,但是这个以为的方法只是对ACC 进行移位的。那么我们要实现这个函数怎么办呢,第一种方法就是在C语言中嵌入汇编通过# pragma asm和# pragma endasm来实现嵌入汇编,格式如下:
……
#pragma asm
RL A
……
#pragma end asm
……
当 然在这里这不是我们要讲的重点,我们要说的是第二种方法,那么我们是不是能够编写一个和RL,RR功能一样的函数呢?其实完全没有必要,这些函数都在 intrins头文件中,根据前面几章介绍的知识,我们只需要在文件中加入这个头文件就可以用了,那么这个函数是怎么个形式呢?我们就看看这个函数吧。
externvoid _nop_ (void);
externbit _testbit_ (bit);
externunsigned char _cror_ (unsigned char,unsigned char);
externunsigned int _iror_ (unsigned int, unsigned char);
externunsigned long _lror_ (unsigned long,unsigned char);
externunsigned char _crol_ (unsigned char,unsigned char);
externunsigned int _irol_ (unsigned int, unsigned char);
externunsigned long _lrol_ (unsigned long,unsigned char);
externunsigned char _chkfloat_(float);
externvoid _push_ (unsigned char _sfr);
externvoid _pop_ (unsigned char _sfr);
以上界函数是在intrins头文件中的函数,这些函数的源码keil封装了起来,只能看到函数的接口,其具体用法在我的附件中有说明,在这里我们需要的 就是_crol_字符循环左移的指令,这个函数有两个入口参数分别是要移动的数据和移动的位数着这里我们移动的数据是0xfe,移动的位数为j(和所在的 位有关)位,因此我们可以将上面的程序改写:
#include
#include
//unsigned char code LED[]={0x01,0x02,0x04,0x08,0x10,0x20};
void delay()
{
unsigned int i;
for(i=0;i
}
void IO_Config()
{
P1M0=0x00;//设置P1端口的类型
P1M1=0x00;//可以根据不同的电路来修改
}
void main()
{
unsigned char j;
IO_Config();
while(1)
{
for(j=0;j
{
//P1=~LED[j];//第一种方法
//P1=temp; //第二种方法
//temp=(temp
P1=_crol_(0xfe,j);//第三种方法
delay();
}
}
}
可以看到其时序和前面的都一样。
最后总结一下,在这个小设计的开发过程中我们主要根据前面的开发流程加深了对单片机的开发,首先设计硬件电路根据硬件电路来将参数提取出来,再根据所选单片 机手册来选择相应的功能,最后根据要求来编写程序,再编程过程中善于利用已经编写好的源码来是实现自己的功能,如果现成的源码不能满足自己的要求就根据自 己的需求来编写相应的程序。
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/139900.html原文链接:https://javaforall.cn