导读:《蓝桥杯单片机组》专栏文章是博主2018年参加蓝桥杯的单片机组比赛所做的学习笔记,在当年的比赛中,博主是获得了省赛一等奖,国赛二等奖的成绩。成绩虽谈不上最好,但至少问心无愧。如今2021年回头再看该系列文章,仍然感触颇多。为了能更好地帮助到单片机初学者,今年特地抽出时间对当年的文章逻辑和结构进行重构,以达到初学者快速上手的目的。需要指出的是,由于本人水平有限,如有错误还请读者指出,非常感谢。那么,接下来让我们一起开始愉快的学习吧。
在前面的那一篇《【蓝桥杯单片机组模块】1、硬件电路基础知识 与 蜂鸣器模块上手》文章中,我们详细介绍了 CT107D 开发板的各个外设驱动方式。本节通过LED流水灯的例子再次熟悉CT107D外设驱动套路,为后文更复杂的外设驱动搭桥铺路。
一、LED 外设基本知识复习
前面我们总结了CT107D的常用外设地址图,如下所示。
LED 外设对应的地址为:0x8000
。
IO 编程点亮LED核心代码如下:
P2 = ((P2&0x1F)|0x80);
P0 = 0x00; //点亮所有LED
P2 = P2 & 0x1F;
二、LED 循环流动实验
LED流水灯花样多了去了,咱还是只挑经典的来就行就可以啦。 : )
那就写一个左右循环流动的 Demo 吧。
#include <stc15.h>
typedef unsigned char u8;//注意类型定义用typedef,养成习惯最好不过了!
typedef unsigned int u16;
void CloseFucker();
void FlowLed();
void main()
{
CloseFucker();
while(1)
{
FlowLed();
}
}
void CloseFucker()
{
P2 = (P2 & 0x1F) | 0xA0;
P0 = P0 & 0xAF;
P2 = P2 & 0x1F;
}
void FlowLed()
{
u16 i;
static bit dir = 0;//单独定义成函数需要static
static u8 shift = 0x01;
P2 = ((P2&0x1F)|0x80);
P0 = ~shift;
if(dir == 0)
{
shift <<= 1;
if(shift == 0x80)
dir = 1;
}
else
{
shift >>= 1;
if(shift == 0x01)
dir = 0;
}
for(i=0; i<60000; i++);
P2 = P2 & 0x1F;
}
程序总体来说的话不难理解,这里总结一下初学者常问的几个问题。
三、初学者常见问题
Q1:LED流动的延时间隔是怎么实现的?具体延时多少?
A:延时间隔通过软件延时实现。延时间隔我们一般用定时器,这里考虑到新手嘛,刚刚上手使用CT107D,肯定想直观看到程序的效果,所以此处的延时是用for循环软件延时来做的,更加简单方便。但是for循环的缺点也就很明显了,我们无法直观的知道延时间隔,for(i=0; i<60000; i++);
这段延时也是试出来的,LED 流动停顿刚好挺合适,读者直接拿去用便是了,不用过分纠结它到底延时了多少。
Q2:点亮第一颗LED为什么要写成shift = 0x01
取反(相当于P0 = 0x01
取反),而不直接写shift = 0xFE
(相当于P0 = 0xFE
)?
A:方便移位。首先明白一点,LED是低电平点亮,shift
中为0
的位,对应的LED会亮。写成shift = ~(0x01)
和shift = 0xFE
这两种形式都可以点亮第一颗LED,但是第二颗LED呢?难道要重新再重新算一下:1111 1101
,得出shift = 0xFD
?难道不麻烦么?如果写成 shift = 0x01
的形式,将其左移一位shift = shift << 1
,然后再取反shift = ~shift
。这样不就可以点亮第二颗LED了么,第三、四....颗LED都是同理。这样岂不是更加直观方便,通过一点小算法,屏蔽了里面的计算细节。
Q3:FlowLed
函数中的变量为什么要使用静态变量(static)?
A:这样可以做到每次进入该函数时对应的static变量保持上一次的值。这个应该是C语言的基础知识,调用某个函数时,其中的staic变量保持上一次的值不变。
小结:本篇文章先是回忆了CT107D开发板的LED外设驱动知识,接着以 LED 流水灯的demo再次熟悉外设驱动套路,并在该demo后详细列出并解答了初学者常见的几个问题。
希望大家多多支持我的原创文章。如有错误,请大家及时指正,非常感谢。