51单片机入门教程(2)——实现流水灯

51单片机入门教程(2)——实现流水灯51单片机入门教程(2)——实现流水灯一、搭建流水灯电路二、流水灯程序2.1延时程序2.2延时函数2.3按字节寻址2.4逻辑移位2.5条件判断一、搭建流水灯电路在Proteus中搭建流水灯电路如图二、流水灯程序我们可以把流水灯看作依次点亮若干个灯。程序如下:#include<reg52.h>sbitled1=P2^0;sbitled2=P2^1…

大家好,又见面了,我是你们的朋友全栈君。

一、搭建流水灯电路

在Proteus中搭建流水灯电路如图
在这里插入图片描述

二、流水灯程序

我们可以把流水灯看作依次点亮若干个灯。
程序如下:

#include <reg52.h>
sbit led1 = P2^0;
sbit led2 = P2^1;
sbit led3 = P2^2;
sbit led4 = P2^3;
sbit led5 = P2^4;
sbit led6 = P2^5;
sbit led7 = P2^6;
sbit led8 = P2^7;

void main()
{ 
   
    //点亮第一个灯
    led1 = 1;
    led2 = 0;
    led3 = 0;
    led4 = 0;
    led5 = 0;
    led6 = 0;
    led7 = 0;
    led8 = 0;
    //点亮第二个灯
    led1 = 0;
    led2 = 1;
    led3 = 0;
    led4 = 0;
    led5 = 0;
    led6 = 0;
    led7 = 0;
    led8 = 0;    
    //点亮剩余的灯
    //省略……
    while(1);
}

编译并下载程序到仿真中,观察现象发现只有第二个灯是亮的???
什么鬼???

2.1 延时程序

单片机的执行指令速度非常快,一个晶振是12MHz的单片机执行一条指令的速度是微秒级的,所以点亮第一个灯的时间太短了,以至于我们根本没有察觉。
因此我们需要一个延时的语句。
实现延时的方法就是循环执行很多次空指令。程序如下:

//延时一秒的程序
int i,j;
for(i = 0;i < 110; ++i)
{ 
   
  for(j = 0; j < 1000; ++j)
  { 
   
    ;//什么也不做
  }
}

然后我们就可以把流水灯的程序改成这样的:

#include <reg52.h>
sbit led1 = P2^0;
sbit led2 = P2^1;
sbit led3 = P2^2;
sbit led4 = P2^3;
sbit led5 = P2^4;
sbit led6 = P2^5;
sbit led7 = P2^6;
sbit led8 = P2^7;

void main()
{ 
   
    int i,j;
   //点亮第一个灯
    led1 = 1;
    led2 = 0;
    led3 = 0;
    led4 = 0;
    led5 = 0;
    led6 = 0;
    led7 = 0;
    led8 = 0;
    //延时1秒 
	for(i = 0;i < 110; ++i)
	{ 
   
	  for(j = 0; j < 1000; ++j)
	  { 
   
	    ;//什么也不做
	  }
	}
    //点亮第二个灯
    led1 = 0;
    led2 = 1;
    led3 = 0;
    led4 = 0;
    led5 = 0;
    led6 = 0;
    led7 = 0;
    led8 = 0;    
    //点亮剩余的灯
    //省略……
    while(1);
}

编译并下载程序到仿真中,观察现象发现首先第一个灯亮,过了一会儿第二个灯亮。

2.2 延时函数

我们剩下的任务就是依次点亮每个灯,但是每次点亮一个灯就需要写一段延时程序,很麻烦!
为了程序的可读性(toulan),可以把延时程序写成一个子函数,随时供我们使用。
C语言中子函数的定义方式如下

返回值类型 函数名 (参数1,参数2,……)
{ 
   
  函数体;
}

这样我们就可以把延时函数写成这样:

void delay1s()
{ 
   
  int i,j;
  for(i = 0; i<110;++i)
  { 
   
    for(j = 0; j<1000;++j)
    { 
   
      //什么也不做
    }
  }
}

几点说明:

  • void:因为该延时函数不需要返回值,所以写为void
  • delay1s:该函数的函数名,命名需要符合C语言的标识符命名规则。
  • (): 不需要传入参数,所以括号中为空
    至此我们可以把流水灯程序写为以下形式:
#include <reg52.h>

sbit led1 = P2^0;
sbit led2 = P2^1;
sbit led3 = P2^2;
sbit led4 = P2^3;
sbit led5 = P2^4;
sbit led6 = P2^5;
sbit led7 = P2^6;
sbit led8 = P2^7;

//延时1s
void delay1s()
{ 
   	 
    int i ,j;
   	for(i = 0;i<110; ++i){ 
   
	  for(j = 0;j<1000;++j){ 
   
	    ;
	  }
	}
}

void main()
{ 
   
    //点亮第一个灯
    led1 = 1;
    led2 = 0;
    led3 = 0;
    led4 = 0;
    led5 = 0;
    led6 = 0;
    led7 = 0;
    led8 = 0;

	//延时1s
	delay1s();

    //点亮第二个灯
    led1 = 0;
    led2 = 1;
    led3 = 0;
    led4 = 0;
    led5 = 0;
    led6 = 0;
    led7 = 0;
    led8 = 0;    
    //点亮剩余的灯
    //省略……
    while(1);
}

2.3 按字节寻址

我们可以看到,上面的代码十分冗长,每次点亮一个灯需要8条语句,那么如何简化?
比如把
led1 = 1;led2 = 0; led3 = 0; led4 = 0; led5 = 0; led6 = 0; led7 = 0; led8 = 0;
这8条语句替代为P2 = 0000 0001???
答案是可以的。代码如下

unsigned char a = 0x01;  //0x01是0000 0001的16进制形式
P2 = a;//相当于led1 = 1;led2 = 0; led3 = 0; led4 = 0; led5 = 0; led6 = 0; led7 = 0; led8 = 0;

至此,我们可以把流水的代码优化为如下形式:

#include <reg52.h>

//延时1s
void delay1s()
{ 
   	 
    int i ,j;
   	for(i = 0;i<110; ++i){ 
   
	  for(j = 0;j<1000;++j){ 
   
	    ;
	  }
	}
}

void main()
{ 
   
	unsigned char a1 = 0x01 ;    // 0000 0001
	unsigned char a2 = 0x02;    // 0000 0010
    //点亮第一个灯
	P2 = a1;

	//延时1s
	delay1s();

    //点亮第二个灯
	P2 = a2;    
    //点亮剩余的灯
    //省略……
    while(1);
}

2.4 逻辑移位

依次点亮8个灯,每点亮一个灯都需要一句赋值语句还是很麻烦 。
所以可以使用逻辑移位语句,每次赋值后,将数值左移一位。
C语言逻辑左移代码如下:

unsigned char a = 0x01;  //a = 0000 0001
unsigned char b = a<<1;  // b = 0000 0010
usingned char c = a<<3;   //c = 0000 1000

至此,我们可以把流水灯的代码优化如下:

#include <reg52.h>
//延时1s
void delay1s()
{ 
   	 
    int i ,j;
   	for(i = 0;i<110; ++i){ 
   
	  for(j = 0;j<1000;++j){ 
   
	    ;
	  }
	}
}

void main()
{ 
   
    //初始化
	unsigned char a = 0x01;
    while(1)
	{ 
   
	  //循环点亮流水灯
	  P2 = a;   
	  a = a<<1;
	  delay1s();
	}
}

编译并下载程序到仿真中,观察现象发现8个灯依次亮过之后不再亮了。

2.5 条件判断

因为在移位操作中,当变量a的值为1000 0000时,再次执行左移操作,a 中的1就溢出了,因此a的值变为0000 0000,此时我们需要加一个判断,使a再次恢复为0000 0001
C语言中,if条件判断使用方式如下

if(判断条件)
{ 
   
  //语句
}

当判断条件为真时,执行{ }中的语句。
至此,流水灯代码可改成如下形式:

#include <reg52.h>

//延时1s
void delay1s()
{ 
   	 
    int i ,j;
   	for(i = 0;i<110; ++i){ 
   
	  for(j = 0;j<1000;++j){ 
   
	    ;
	  }
	}
}

void main()
{ 
   
	unsigned char a = 0x01;
    while(1)
	{ 
   
	  if(a == 0x00)   //如果高位溢出
	  { 
   
	    a = 0x01;      //则恢复
	  }
	  //循环点亮led灯
	  P2 = a;
	  a = a<<1;
	  delay1s();
	}
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/132711.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • カード名義_acm题

    カード名義_acm题原题链接给定一棵包含 n 个节点的有根无向树,节点编号互不相同,但不一定是 1∼n。有 m 个询问,每个询问给出了一对节点的编号 x 和 y,询问 x 与 y 的祖孙关系。输入格式输入第一行包括一个整数 表示节点个数;接下来 n 行每行一对整数 a 和 b,表示 a 和 b 之间有一条无向边。如果 b 是 −1,那么 a 就是树的根;第 n+2 行是一个整数 m 表示询问个数;接下来 m 行,每行两个不同的正整数 x 和 y,表示一个询问。输出格式对于每一个询问,若 x 是 y 的祖先则输

    2022年8月8日
    8
  • HTTP_POST请求的数据格式

    HTTP_POST请求的数据格式在HTTP的请求头中,可以使用Content-type来指定不同格式的请求信息。Content-type的类型常见的媒体格式类型:&nbsp;&nbsp;&nbsp;text/html:HTML格式&nbsp;&nbsp;&nbsp;text/plain:纯文本格式&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbs…

    2022年4月1日
    31
  • idea2019最新可用激活码_通用破解码

    idea2019最新可用激活码_通用破解码,https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月16日
    1.0K
  • Pytest(16)随机执行测试用例pytest-random-order[通俗易懂]

    Pytest(16)随机执行测试用例pytest-random-order[通俗易懂]前言通常我们认为每个测试用例都是相互独立的,因此需要保证测试结果不依赖于测试顺序,以不同的顺序运行测试用例,可以得到相同的结果。pytest默认运行用例的顺序是按模块和用例命名的ASCII编码

    2022年7月29日
    8
  • 从零开始学习Prometheus监控报警系统[通俗易懂]

    从零开始学习Prometheus监控报警系统[通俗易懂]Prometheus是一个开源的监控报警系统,它被纳入了由谷歌发起的Linux基金会旗下的云原生基金会,并成为仅次于Kubernetes的第二大开源项目。

    2022年6月3日
    34
  • ICEM二维网格

    ICEM二维网格非结构网格结构网格拓扑拓扑完建立part边界条件,然后创建block拓扑完后进行边界条件关联全局网格设置转载于:https://www.cnblogs.com/Jay-CFD/p/8795203.html…

    2022年5月13日
    41

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号