单片机学习笔记——入门51单片机

news2025/1/23 22:37:48

一、单片机基础介绍

1.何为单片机

单片机,英文Micro Controller Unit,简称MCU 。内部集成了中央处理器CPU、随机存储器ROM、只读存储器RAM、定时器/计算器、中断系统和IO口等一系列电脑的常用硬件功能 单片机的任务是信息采集(依靠传感器)、处理(依靠CPU)和硬件设备(例如电机,LED等)的控制 。单片机跟计算机相比,单片机算是一个袖珍版计算机,一个芯片就能构成完整的计算机系统。但在性能上,与计算机相差甚远,但单片机成本低、体积小、结构简单,在生活和工业控制领域大有所用。 同时,学习使用单片机是了解计算机原理与结构的最佳选择。

单片机工作的基本时序

我们都知道在学校是通过铃声来控制所有班级的上下课时间,我们都知道单片机执行指令的过程就是从ROM取出一条指令执行来完成它在各个地方的作用,那它什么时候取指令这个是顺序呢?这里引入一个时序的周期,每访问一次ROM的时间,就是一个机器周期的时间

1个机器周期 = 6个状态周期 = 12个时钟(振荡)周期  

时钟周期:即单片机的基本时间单位,若晶体的频率=12MHZ,那时钟周期 = 1/12MHZ,一个时钟周期 = 1/12MHZ = 1/12000 000每秒

机器周期:即12x1/12 000 000 =0.000001s = 1us,访问一次ROM取指令的时间就是1us

2.单片机命名规则

3.单片机内部结构

重点需记:单片机管脚

1.电源:Vcc:正极     Gnd:负极             2.XTAL:单片机时钟引脚,外接晶振

3.RST:复位

4.开发板介绍

开发板原理图

二、单片机的一些基础项目

2-1、点亮一个led灯

#include <REGX52.H>
void main()
{
	P2=0x7f;//1111 1110  d1   16 15
	        //0111 1111  d8    7 16
	        //1011 1111  d7   11 16
	
	
}

通过高低电平控制led亮否

2-2、led闪烁

#include <REGX52.H>
#include <INTRINS.H>

//延时函数
void Delay500ms()		//@11.0592MHz
{
	unsigned char i, j, k;

	_nop_();
	i = 4;
	j = 129;
	k = 119;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}

void main()
{
	while(1){
	P2=0xfe;//亮
 Delay500ms();//延时500ms
	P2=0xff;//灭
 Delay500ms();//延时500ms
	}
}

通过延时函数使led闪烁

2-3、流水灯

#include <REGX52.H>
#include <INTRINS.H>

void Delay500ms()		//@11.0592MHz
{
	unsigned char i, j, k;

	_nop_();
	i = 4;
	j = 129;
	k = 119;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}

void main()
{
	while(1){
	P2=0xfe;//1111 1110
 Delay500ms();
	P2=0xfd;//1111 1101
 Delay500ms();
		P2=0xfb;//1111 1011
 Delay500ms();
		P2=0xf7;//1111 0111
 Delay500ms();
		P2=0xef;//1110 1111
 Delay500ms();
		P2=0xdf;//1101 1111
 Delay500ms();
		P2=0xbf;//1011 1111
 Delay500ms();
		P2=0x7f;//0111 1111
 Delay500ms();
	}
}

位运算做法:

2-4、流水灯plus

#include <REGX52.H>
#include <INTRINS.H>

//任意延时函数——1ms的延时函数执行x次循环
void Delay1ms(unsigned int xms)		//@11.0592MHz
{
	unsigned char i, j;
  while(xms)
	{
	_nop_();
	i = 2;
	j = 199;
	do
	{
		while (--j);
	} while (--i);

	xms--;}
}

void main()
{
	while(1){
	P2=0xfe;//1111 1110
 Delay1ms(100);
	P2=0xfd;//1111 1101
 Delay1ms(100);
		P2=0xfb;//1111 1011
 Delay1ms(100);
		P2=0xf7;//1111 0111
 Delay1ms(100);
		P2=0xef;//1110 1111
 Delay1ms(100);
		P2=0xdf;//1101 1111
 Delay1ms(100);
		P2=0xbf;//1011 1111
 Delay1ms(100);
		P2=0x7f;//0111 1111
 Delay1ms(100);
	}
}

通过任意延时函数去简化步骤

3-1、通过独立按键控制led闪烁

#include <REGX52.H>

void main()
{
	while(1)
	{
		if(P3_1==0)//低电平 按下按键接地为0
		{
			P2_0=0;//d1亮
		}
		else {
			P2_0=1;
		}
	}
}

3-2、通过独立按键控制led状态

#include <REGX52.H>
#include <INTRINS.H>
void Delay(unsigned int xms)		//@11.0592MHz
{
	unsigned char i, j;
while(xms){
	_nop_();
	i = 2;
	j = 199;
	do
	{
		while (--j);
	} while (--i);
	xms--;
}
}

void main()
{
	while(1)
	{
		if(P3_1==0)
		{
		 Delay(20);//取消前摇
			while(P3_1==0);//判断何时松手
			Delay(20);//取消后摇
			
			P2_0=~P2_0;//按位取反
	}
}
}

取消按键时的抖动,使单片机稳定判断状态。按一次亮,按一次灭。

3-3、通过独立按键控制led完成二进制

#include <REGX52.H>
#include <INTRINS.H>

void Delay(unsigned int xms)		//@11.0592MHz
{
	unsigned char i, j;
while(xms){
	_nop_();
	i = 2;
	j = 199;
	do
	{
		while (--j);
	} while (--i);
	xms--;
}
}

void main()
{
	unsigned int lednum=0;
	while(1)
	{
		if(P3_1==0)
		{
			Delay(20);
			while(P3_1==0);
			Delay(20);
			
			lednum++;
			P2=~lednum;
	}
}
}

也可以直接用P2--;代替最后两句,完成二进制运算。

3-4、用独立按键控制led灯移位

#include <REGX52.H>
#include <INTRINS.H>

void Delay(unsigned int xms)		//@11.0592MHz
{
	unsigned char i, j;
while(xms){
	_nop_();
	i = 2;
	j = 199;
	do
	{
		while (--j);
	} while (--i);
	xms--;
}
}

void main()
{
	unsigned int lednum=0;
	P2=~0x01;
	while(1)
	{
		if(P3_1==0)//k1
		{
			Delay(20);
			while(P3_1==0);
			Delay(20);
			
			lednum++;
			if(lednum>=8)
				lednum=0;
			P2=~(0x01<<lednum);
	}
		if(P3_0==0)//k2
		{
			Delay(20);
			while(P3_0==0);
			Delay(20);
			
			
			if(lednum==0)
				lednum=7;
			else lednum--;
			P2=~(0x01<<lednum);
}
}
	}

4-1 静态数码管

1.常见数码管

2.控制数码管显示的原理图

3.管脚定义(对应字母控制对应位置亮):上面的为共阴极、下面的为共阳极(可以理解为3,8管脚处为供电,三角形尖尖有一横的是负极,所有共阴极),两个图中的数字为引脚:

4.下面为多个数码管,PCB板的4个为一体,同样上面为共阴极、下面为共阳极的原理图:

5.STC89C52实现数字显示

①原理图是共阴极(上面给0、下给1亮)

②138译码器:输入3(ABC,读的时候是从下读 C B A )个口,控制输出8个口,输出口连接共阴极的,是0还是1,在这里控制:使能端连接(按下图给1和0就可以用了)

通过CBA给数字0和1二进制转换10进制(得到数字几)就控制Y几,Y0头上“—”是表示低电平有效(即给0):

③双向缓冲,高电平往低电平送数据

CC2电容作用:滤波电容,稳定电源,确定电路稳定性,提高电路工作性能可靠运行;

RP4:排阻,限流,防止电流过大

④这里的P01......P07,就是用P0口,后面代码就是通过P0口控制灯的

只有Y5为0,其他Y0...Y7都为1;

读取顺序都是从下到上

⑤代码控制公共端,从下往上写:

二进制101转换为1十进制为5,控制Y5,即公共端的LED6;

要显示下图的数字6

代码实现如下(P2控制共阴极,P0控制显示数字)及结果;

 ⑥优化操作代码:通过数组,子函数来优化代码
#include <REGX52.H>
#include <INTRINS.H>

unsigned char NixieTable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x00};

void Nixie(unsigned char Location,Number)
{
	switch(Location)
	{
		case 1:P2_4=1;P2_3=1;P2_2=1;break;
		case 2:P2_4=1;P2_3=1;P2_2=0;break;
		case 3:P2_4=1;P2_3=0;P2_2=1;break;
		case 4:P2_4=1;P2_3=0;P2_2=0;break;
		case 5:P2_4=0;P2_3=1;P2_2=1;break;
		case 6:P2_4=0;P2_3=1;P2_2=0;break;
		case 7:P2_4=0;P2_3=0;P2_2=1;break;
		case 8:P2_4=0;P2_3=0;P2_2=0;break;
	}
	P0=NixieTable[Number];
}

void main()
{
	Nixie(7,10);
	while(1);
	}

要显示的数字对应的值

4-2、动态数码管 

#include <REGX52.H>
#include <INTRINS.H>

unsigned char NixieTable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x00};

void Delay (unsigned int xms)		//@11.0592MHz
{
	unsigned char i, j;
while(xms--){
	_nop_();
	i = 2;
	j = 199;
	do
	{
		while (--j);
	} while (--i);
}
}

void Nixie(unsigned char Location,Number)
{
	switch(Location)
	{
		case 1:P2_4=1;P2_3=1;P2_2=1;break;
		case 2:P2_4=1;P2_3=1;P2_2=0;break;
		case 3:P2_4=1;P2_3=0;P2_2=1;break;
		case 4:P2_4=1;P2_3=0;P2_2=0;break;
		case 5:P2_4=0;P2_3=1;P2_2=1;break;
		case 6:P2_4=0;P2_3=1;P2_2=0;break;
		case 7:P2_4=0;P2_3=0;P2_2=1;break;
		case 8:P2_4=0;P2_3=0;P2_2=0;break;
	}
	P0=NixieTable[Number];
	Delay(1);//保证亮度
  P0=0x00;//清零
}

void main()
{
	while(1){
	Nixie(2,1);
 // Delay(20);
	Nixie(3,2);
 // Delay(20);
	Nixie(4,3);
 //	Delay(20);
	}
}

注释掉上面的延时调用,旁边的管会有些影响,需要消影,段选、位选影响造成串位,如下代码消除

了解

5-1、模块化编程 

1)驱动,先会用,后续有详细内容:

2)模块化,功能函数用点C文件写,点H文件声明函数,在main函数文件引入头文件直接调用:

3)注意事项

4)预编译

5)延时函数文件

6)头文件延迟

7)主函数文件程序入口:

8)数码管模块,用到的头文件要引用:

9)数码管模块头文件

10)函数调用

11)显示

5-2 、LCD1602调试工具-------

1)调试工具原理图

2)模块化代码,下完放到自己工程目录下:



3)将下好的两个文件添加到工程:

4)文件主要内容如下:

5)main函数调用:

6)显示其他管脚冲突,所有会一起显示:

7)其他函数的调用及功能,可以设置显示位置和范围:

8)需要用到延迟函数:可以直接将前面模块化文件复制到工程目录下,添加进来引用即可;

9)娱乐:小计时器

#include <REGX52.H>
#include "LCD1602.h"
#include "Delay.h"

int Result=0;

void main()
{
	LCD_Init();//初始化
	
	while(1)
		{
		Result++;
			Delay(1000);
			LCD_ShowNum(1,1,Result,3);
		}//计时器



6-1.矩阵键盘

1)基础介绍:

P14-P17给0就代表扫描,其他给1(没选中),一次只能扫描一行;P10-P13给0表示按下,给1表示没按下;(逐列扫描)

2.代码实现:

①:创建工程并把“Delay”与“LCD1602”的模块加入此工程中。
②:编写MatrixKey(矩阵)代码
#include <REGX52.H>
#include "Delay.h"

unsigned char MatrixKey()
{
	unsigned char KeyNumber=0;
	
	P1=0xff;      //按列扫描
	P1_3=0;       //控制扫描的列
	if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=1;}    //while判断何时松手
	if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=5;}
	if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=9;}
	if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=13;}
	
	P1=0xff;
	P1_2=0;
	if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=2;}
	if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=6;}
	if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=10;}
	if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=14;}
	
	P1=0xff;
	P1_1=0;
	if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=3;}
	if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=7;}
	if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=11;}
	if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=15;}
	
	P1=0xff;
	P1_0=0;
	if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=4;}
	if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=8;}
	if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=12;}
	if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=16;}
	
	
	return KeyNumber;
}
	
③.编写主函数
#include <REGX52.H>
#include "Delay.h"
#include "LCD1602.h"
#include "matrixKey.h"

unsigned char KeyNum;

void main()
{
	LCD_Init();   //LCD上电初始化
	LCD_ShowString(1,1,"Helloworld");
	while(1)
	{
		KeyNum=MatrixKey();
		if(KeyNum){
			LCD_ShowNum(2,1,KeyNum,2);
	}
}
	}
④.软件使用小技巧

快速生成常用格式代码:

设置,完成后双击就可以生成了:

6-2.矩阵键盘密码锁

1):把6-1文件全部cv到6-2工程文件中

2):代码实现

#include <REGX52.H>
#include "Delay.h"
#include "LCD1602.h"
#include "matrixKey.h"

unsigned char KeyNum;
unsigned int Password,count;


void main()
{
	LCD_Init();
	LCD_ShowString(1,1,"Helloworld");
	while(1)
	{
		KeyNum=MatrixKey();
		if(KeyNum){
			if(KeyNum<=10){  				//s1~s10按下密码
				if(count<4){
				Password*=10;
				Password+=KeyNum%10;       //这两句是用来实现四位密码的显示
				count++; 		//计数  防止按下的密码数超过四位	
				}
				LCD_ShowNum(2,1,Password,4);//更新显示
			}
			
			if(KeyNum==11){    //s11设置为确认键
				if(Password==2345){
					LCD_ShowString(1,14,"OK ");
					Password=0;  //密码清零
					count=0;      //计数清零
					LCD_ShowNum(2,1,Password,4);//更新显示
			}
				else {LCD_ShowString(1,14,"ERR");
					Password=0;  //密码清零
					count=0;      //计数清零
					LCD_ShowNum(2,1,Password,4);//更新显示
				}
	}
			
	   if(KeyNum==12){  //定义S12为取消键
			 Password=0;
			 count=0;
			 LCD_ShowNum(2,1,Password,4);
		 }
}
	}
}

7.定时器介绍

1)介绍,Delay前面CPU是一直在等的,用定时器在Delay时可以去检测按键,提高CPU利用率:

2)模式1最常用:

3)模式:时钟--计数最大65535(计数系统TL0\TH0:每来一个脉冲+1方法计数)-TF0(标志位,到最大了回到0)-中断:

4)非门与门图形为控制部分(TR0是否启动暂定)

5)定时器部分:

6)时钟可以由系统提供(上图,晶振),也可以由外部T0P提供(如下图引脚)

7)C/T,给1连上面为控制器,给0连接下面为定时器(如下图):

8)中断系统:

9)中断资源

10)定时器和中断系统

11)定时器相关寄存器

7-1.独立按键控制流水灯的模式

1.TMOD寄存器工作模式,定时器0配置使用(不可位寻址,只能整体赋值)

2.定时器模式1:门控端给0,就是tr0单独控制:C/T,T这里有一横表示低电平有效,就给0是用T(定时器模式),给1用C(控制器模式),M1,M0工作模式选择

3.TCON控制寄存器(可位寻址,可以单独每一位赋值)

中断溢出标志位:

TF=0(等于1产生中断);

TR0=1(定时器是否开启,给1开始,电机开始工作);

IE0、IT0:控制外部中断引脚,可以不配置

4.定时器配置  TH0\TL0

TH0\TL0 分开储存

代码优化,TMOD问题(不可位寻址)配置两个的时候,后面的会把前面的覆盖;

因此,可采用“与或”法设定TMOD

 

5.中断配置T0-->ET0=1--EA=1,PT0=0

6.定时器配置完成,模块化编程

#include <REGX52.H>

void Timer0_Init(void)		//1ms@11.0592MHz
{
	TMOD &= 0xF0;		//设置定时器模式
	TMOD |= 0x01;		//设置定时器模式
	TL0 = 0x66;		//设置定时器初值
	TH0 = 0xFC;		//设置定时器初值
	TF0 = 0;		//清除TF0标志
	TR0 = 1;		//定时器0开始计时
	ET0=1;    //下面三行为中断的配置
	EA=1;
	PT0=0;
}


 7.定时器中断函数模板


void Timer0_Routine() interrupt 1  //中断函数
{
	static unsigned int T0Count; 
	TL0 = 0x66;		//设置定时器初值
	TH0 = 0xFC;		//设置定时器初值
	T0Count++;
	if(T0Count>=1000)   //每隔1s
	{		
         T0Count=0;

	}
}

8.独立按键模块

#include <REGX52.H>
#include "Delay.h"



unsigned char Key()
{
	
	unsigned char KeyNumber=0;
	
	if(P3_1==0){Delay(20);while(P3_1==0);Delay(20);KeyNumber=1;}
	if(P3_0==0){Delay(20);while(P3_0==0);Delay(20);KeyNumber=2;}
	if(P3_2==0){Delay(20);while(P3_2==0);Delay(20);KeyNumber=3;}
	if(P3_3==0){Delay(20);while(P3_3==0);Delay(20);KeyNumber=4;}
	
	return KeyNumber;
}

9.主函数

#include <REGX52.H>
#include "Timer0.h"
#include "key.h"
#include <INTRINS.H>

unsigned char KeyNumber,LEDmode;
void main()
{
	P2=0xfe;  //与INTRINS.H中的循环左右移函数共同实现流水灯
	Timer0_Init(); //上电初始化
	while(1)
	{
		KeyNumber=Key();
		if(KeyNumber)
		{
			if(KeyNumber==1)
			{
				LEDmode++;
				if(LEDmode>=2) LEDmode=0;
			}
			
		}
		
	}
}


 
void Timer0_Routine() interrupt 1  //中断函数
{
	static unsigned int T0Count; 
	TH0=64535/256;
	TL0=64535%256; 	
	T0Count++;
	if(T0Count>=1000)   //每隔1s
	{
		T0Count=0;
	 if(LEDmode==0) P2=_crol_(P2,1);
	 if(LEDmode==1) P2=_cror_(P2,1);
	}
}

7-2.时钟 

1.把LCD1602液晶显示、延迟、定时器、的代码复制到工程目录下,导入;

2.主函数包含其他模块头文件并初始化;

3.复制定时器中断函数到main函数下:

4.定义变量,秒计数、分、小时并显示:

5.代码综合

#include <REGX52.H>
#include "Delay.h"
#include "LCD1602.h"
#include "Timer0.h"

unsigned char Sec,Min,Hour;
void main()
{
	LCD_Init();//³õʼ»¯
	Timer0_Init();
	LCD_ShowString(1,1,"CLOCK:");
	LCD_ShowString(2,1,"  :  :");
	while(1)
	{
	LCD_ShowNum(2,1,Hour,2);
	LCD_ShowNum(2,4,Min,2);
	LCD_ShowNum(2,7,Sec,2);
	}
}

void Timer0_Routine() interrupt 1  
{
	static unsigned int T0Count; 
	TL0 = 0x66;		
	TH0 = 0xFC;		
	T0Count++;
	if(T0Count>=1000)   
	{
		T0Count=0;
    Sec++;
		if(Sec>=60)
		{
			Sec=0;
			Min++;
			if(Min>=60)
			{
				Min=0;
				Hour++;
				if(Hour>=24)
				{
					Hour=0;}
				}
			}
		}
		
	}

8-1串口介绍

1)介绍

2)向单片机发送数据(下面框),返回(上框)

3)DB9串口传输数据(注意使用的电压是否一致)使用RS232或RS485电平

4)知识点

①硬件电路

注:最少需要三根线实现双向通信:TXD,RXD,GND。VCC不一定需要,可独立供电。

②电平标准 

 

 差分信号优点:传送距离远(1km+)   TTL与RS232(10m)

③常见通信接口比较

 CAN总线:常用于汽车领域,因为使用的是差分信号传输,传输距离远、稳定。

通信方式的相关术语:

④51单片机里的UART串口 

 

 

   中间部分用来控制波特率,依靠定时器来约定速率,T1的溢出率通过分频后来控制收发器的采样时间。

  SBUF:收发数据后,会产生相应的TI(发送中断)/RI(接收中断),继而进入中断函数,进行相应的中断函数内部的操作。

 配置ES、EA,PS此时不需要配置,因为只有一个中断,不需要进行优先级判断。

 配置好SCON和PCON,读SBUF,配置定时器T1,打开EA和ES,即串口可以开始工作。

8-2.实际配置串口 

1)将延迟函数复制过来并导入工程里面;

2)配置串口控制寄存器,配置模式1最常用,REN允许接收给1,不允许接收给0(也可以给1外面不给发就行);

3)TI、RI发送完置1(硬件只负责),但必须软件复位置0;

 故SCON=0x40

PCON

4)配置定时器

这里定时器1,没有定时器0,所有要把高位修改(不影响高低位配置用“”& |“”这两个方式)

选择8位自动重载模式

5)可以直接用STC-isp来配置串口 

系统频率根据板子选择,波特率4800,波特率发生器选择8位自动重载,时钟12T

6)发送数据的函数

7)发送单项数据

8)模块化

9)数据显示模式 

8-3.串口实例实现 

1)每一秒发送一个递增的数字

 2)电脑通过串口控制LED灯,并且返回电脑读入的数据

1.需要打开串口的中断功能

SCON = 0x50;  EA=1;ES=1; 

注:要分清这里是禁止了定时器1的中断功能,只是让它的溢出率去开启串口收发的功能,中断的产生是由于串口收发数据产生的中断。

2.编写串口中断函数

串口中断函数模板:

void UART_Routine() interrupt 4
{
	 if(RI==1)
	 {
		 
		 RI=0;   //复位清0
	 }
}

3.整体代码

9-1LED点阵屏

1.介绍:


2.显示原理

3.相关图

①:led矩阵图——经测试,P0控制列,D控制行

②:开发版引脚对应关系:

③:74HC595 

 加-,低电平有效

OE: 使能,output enable,接低电平工作,高电平不工作。

RCLK:寄存器时钟,register clock

SRCLR:串行清零端

SRCLK:串行时钟 

SER:串行数据

运行方式:类似队列

④:总结:使用步骤:进行行列的选择——列由P0口控制,因此给P0赋值就能控制列。行需通过74HC595来间接控制。

9-2.点阵屏的驱动代码 

示例代码1-点阵屏显示图形

1)sfr与sbit,可位寻址与不可位寻址。

2) 进行位声明,方便操作。

3)编写子函数——控制74HC595把字节数据输出给D的8个引脚。

4) 编写LED矩阵显示的子函数

5)  自己设计图形,调用函数输入数据,即可显示;

示例代码2-点阵屏显示动画

1)把示例1的代码模块化
2)定义要显示的动画的数组,并初步检测是否显示正常

3)实现动态显示

 

上述代码实现的是流水式的动画,也可以改变offset去实现逐帧动画。

4)定义的Animation是放在RAM中的,有空间限制,浪费RAM空间。所以可以加个code,把它放在flash里,但此时不可在主函数里改变数组内容。

10-1 DS1302实时时钟介绍

1.介绍

定时器计时的缺点:1.精度不高  2.消耗单片机的CPU  3.断电不能继续计时

而时钟芯片精度高,且有备用电源,掉电可以用备用电源继续计时。

2.时钟芯片的引脚定义和应用电路

①:两种封装模式:直插封装和贴片封装  ②:分为三部分:电源部分,时钟部分,数据交互部分

3. 内部结构框图

4.内部寄存器的定义

秒,分,时,日,月,年,星期,年,wp(write protect,写保护),涓流充电

前两列的为 :命令字——

5. 时序图

前八为指定读还是写,后八位读出或者写入指定的数据。

上升沿写入,下降沿读出。

6.BCD码

10-2 实例代码

1.时钟

1)创建工程,cvLCD1602的文件,并开始模块化编写DS1302的模块

2) DS1302模块的子函数编写

3)初步测试 

结果会发现,数字到9后会直接到16……,这是因为寄存器数据是以BCD码存储的,只需将Second改为Second/16*10+Second%16,就可以转化成十进制,正常显示。

4) 定义一下各个寄存器的地址,方便操作;再定义一个存储时间的数组;
5) 编写DS1302的另外两个子函数——①:设定时间 ②:读出时间     并声明为外部可调用的函数和数组;

6) 编写主函数,调用函数在LCD显示屏上显示时间

2.功能化时钟——可以用独立按键设置要显示的时间

1)需要用到的模块

DS1302模块,LCD1602模块,独立按键key模块,定时器Timer0模块,延时函数模块;

2)编写主函数

主要分为四大部分:时钟模式选择,显示时间,设置时间(选择设置的时间位置,调整时间大小,利用定时器中断去实现选择的时间的位置的动态显示),更新显示新的时钟。

#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "Key.h"
#include "Timer0.h"
#include "Delay.h"

unsigned char KeyNum,MODE,TimeSetSelect,TimeFlashFlag;


/**
  * @brief  显示时间
  * @param 无
  * @retval 无
  */
void TimeShow(void)
{
		DS1302_ReadTime();
		LCD_ShowNum(1,1,DS1302_Time[0],2);
		LCD_ShowNum(1,4,DS1302_Time[1],2);
		LCD_ShowNum(1,7,DS1302_Time[2],2);
		LCD_ShowNum(2,1,DS1302_Time[3],2);
		LCD_ShowNum(2,4,DS1302_Time[4],2);
		LCD_ShowNum(2,7,DS1302_Time[5],2);
}

/**
  * @brief  设置时间,按键2选择设置的时间位置,按键3让时间++,按键4让时间--
  * @param  无
  * @retval 无
  */
void TimeSet(void)
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		TimeSetSelect %= 6;
	}
	if(KeyNum==3)
	{
		DS1302_Time[TimeSetSelect]++;
		if(DS1302_Time[0]>99){DS1302_Time[0]=0;}
		if(DS1302_Time[1]>12){DS1302_Time[1]=1;}
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 ||DS1302_Time[1]==5 ||DS1302_Time[1]==7 
		|| DS1302_Time[1]==8 || DS1302_Time[1]==10 ||DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
		}else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 ||DS1302_Time[1]==5 ||                     
        DS1302_Time[1]==9 || DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
		}else if(DS1302_Time[1]==2)
		{	if(DS1302_Time[0]%100!=0&&DS1302_Time[0]%4==0||DS1302_Time[0]%100==0&&DS1302_Time[0]%400==0)
			{
				if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
			}else if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
		}
		if(DS1302_Time[3]>23){DS1302_Time[3]=0;}
		if(DS1302_Time[4]>59){DS1302_Time[4]=0;}
		if(DS1302_Time[5]>59){DS1302_Time[5]=0;}
	}
	if(KeyNum==4)
	{
		DS1302_Time[TimeSetSelect]--;
		if(DS1302_Time[0]<0){DS1302_Time[0]=99;}
		if(DS1302_Time[1]<0){DS1302_Time[1]=12;}
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 ||DS1302_Time[1]==5 || DS1302_Time[1]==7 
			|| DS1302_Time[1]==8 || DS1302_Time[1]==10 ||DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]<1){DS1302_Time[2]=31;}
			if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
		}else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 ||DS1302_Time[1]==5 || DS1302_Time[1]==9 
		||	DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]<1){DS1302_Time[2]=30;}
			if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
		}else if(DS1302_Time[1]==2)
		{
			if(DS1302_Time[0]%100!=0&&DS1302_Time[0]%4==0||DS1302_Time[0]%100==0&&DS1302_Time[0]%400==0)
			{
				if(DS1302_Time[2]<1){DS1302_Time[2]=29;}
				if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
			}else 
	{
		if(DS1302_Time[2]<1){DS1302_Time[2]=28;}
		if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
	}
		}
		if(DS1302_Time[3]<0){DS1302_Time[3]=23;}
		if(DS1302_Time[4]<0){DS1302_Time[4]=59;}
		if(DS1302_Time[5]<0){DS1302_Time[5]=59;}
	}
	//动态显示选择位
	if(TimeSetSelect==0 && TimeFlashFlag==1 ){LCD_ShowString(1,1,"  ");}
	else {LCD_ShowNum(1,1,DS1302_Time[0],2);}
	if(TimeSetSelect==1 && TimeFlashFlag==1 ){LCD_ShowString(1,4,"  ");}
	else {LCD_ShowNum(1,4,DS1302_Time[1],2);}
	if(TimeSetSelect==2 && TimeFlashFlag==1 ){LCD_ShowString(1,7,"  ");}
	else {LCD_ShowNum(1,7,DS1302_Time[2],2);}
	if(TimeSetSelect==3 && TimeFlashFlag==1 ){LCD_ShowString(2,1,"  ");}
	else {LCD_ShowNum(2,1,DS1302_Time[3],2);}
	if(TimeSetSelect==4 && TimeFlashFlag==1 ){LCD_ShowString(2,4,"  ");}
	else {LCD_ShowNum(2,4,DS1302_Time[4],2);}
	if(TimeSetSelect==5 && TimeFlashFlag==1 ){LCD_ShowString(2,7,"  ");}
	else {LCD_ShowNum(2,7,DS1302_Time[5],2);}
}



void main()
{
	Timer0_Init();
	LCD_Init();
	DS1302_Init();
	DS1302_WriteByte(0x8e,0x00);  

	LCD_ShowString(1,1,"  -  -  ");
	LCD_ShowString(2,1,"  :  :  ");
	DS1302_SetTime();
	
	while(1)
	{
		//通过按键1控制时钟的模式,模式0为显示时钟,模式1为设置时间
		KeyNum=key();
		if(KeyNum==1)
		{
			if(MODE==0){MODE=1;TimeSetSelect=0;}
			else if(MODE==1) {MODE=0;DS1302_SetTime();}
		}
		switch(MODE)
		{
			case 0:TimeShow();break;
			case 1:TimeSet();break;
		}
	}
	
}

//利用定时器中断来动态显示选择的时间位置
void Timer0_Routine() interrupt 1  
{
	static unsigned int T0Count; 
	TL0 = 0x66;		
	TH0 = 0xFC;		
	T0Count++;
	if(T0Count>=1000)   
	{
		T0Count=0;
		TimeFlashFlag=!TimeFlashFlag;
	}
}

11.蜂鸣器

1)介绍

 NPN:高电平导通     PNP:低电平导通

由P15的高低电平取反控制BZ的高低电平

 2)乐理知识

①:介绍C1-C2升高8度。往右升高,往左降低:相邻半音黑白,1对应中央C1,i表示升即C2部分,降低8度下面加一个点,降低2个8度,下面加两个点(白键);相邻两个键为半音#:升高半音,b:降低半音。

②:-表示时长,如图中的665-,其中5占两个节拍,

③:音符

 一般以四分音符为一个基准

④:音符频率对照

⑤:以a440hz为基准,到另一个a,中间等比12频分,
 ⑥:单片机晶振芯片每秒震荡的次数称为时钟频率,震荡一次所需时间称为振荡周期。12个震荡周期是一个机器周期,机器频率=晶振频率/12,计时周期=1/机器周期,即每过一个计时周期,定时器计数+1;
给定时器的TL和TH赋重装载值,使定时器每过所需周期的一半时计数+1(因为需要给震荡信号,故计数周期=原周期的二分之一),从而使蜂鸣器发出对应频率的声音。

3)实例代码

1.蜂鸣器鸣响
①:蜂鸣器模块函数

②:主函数——按下独立按键,在数码管上显示按的第几个键,并且蜂鸣器鸣响。

 

2.蜂鸣器演奏音乐—有时间再补

12.AT24C02,I2C总线

1.存储器介绍

RAM:存储速度快,但是掉电丢失   ROM:掉电不丢失,但是速度慢。

2.AT24C02介绍

引脚及应用电路

 

3. I2C总线

 

I2C通信 时序结构:

 

 

 

 

 

 

4.示例程序

1——用独立按键设置想要写入的数据,并读出显示在LCD上,断电不丢失。
①:程序编写整体思路:编写I2C模块,AT24C02模块,main模块

②:编写I2C模块

先进行位声明,再分别编写子函数,开始,结束,发送和接受一个字节,发送和接收应答位

 

③:编写AT24C02模块

先定义一下AT24C02的地址,写是0xA0,读是0xA1;

再用I2C里的函数去编写写入一个字节和读出一个字节的函数

 

④:编写main模块

用独立按键控制想要写入的数据,并读出显示在LCD上,断电不丢失。

 

2——秒表(用定时器扫描数码管)
①:先创建新的工程,包含所需的模块——独立按键,数码管,延迟函数,定时器,AT,I2C
②:改造独立按键模块——用定时器扫描独立按键

这样就不用像原来一样在检测时停在while死循环里,影响主函数进程

 

③:改造数码管模块——改造成用定时器扫描数码管

先在main模块的中断函数里加上一个计数定时器的T0Count2,这样每隔一段时间就会调用一下NixieLoop函数,而NixieLoop函数会不断扫描并在数码管上显示每一位数字。NixieSetBuf函数会根据输入的位置和数字改变Buf数组里的值,继而让NixieLoop扫描显示对应的值。

 

④: 编写主函数
#include <REGX52.H>
#include "Timer0.h"
#include "key.h"
#include "Delay.h"
#include "Nixie.h"
#include "AT24C02.h"

unsigned char KeyNum;
unsigned char Min,Sec,Minisec;
unsigned char Runflag;

void main()
{
	Timer0_Init();
	
	while(1)
	{
		KeyNum=Key();
		if(KeyNum==1)   //按键1控制开始和停止计时
		{
			Runflag=!Runflag;
		}
		if(KeyNum==2)    //按键2清0
		{
			Min=0;
			Sec=0;
			Minisec=0;
		}
		if(KeyNum==3)  //按键3将数据写入AT
		{
			AT24C02_WriteByte(0,Min);
			Delay(5);
			AT24C02_WriteByte(1,Sec);
			Delay(5);
			AT24C02_WriteByte(2,Minisec);
			Delay(5);
		}
		if(KeyNum==4)   //按键4读出数据
		{
			Min=AT24C02_ReadByte(0);
			Sec=AT24C02_ReadByte(1);
			Minisec=AT24C02_ReadByte(2);
		}
			Nixie_SetBuf(1,Min/10);
			Nixie_SetBuf(2,Min%10);
			Nixie_SetBuf(3,11);
			Nixie_SetBuf(4,Sec/10);
			Nixie_SetBuf(5,Sec%10);
			Nixie_SetBuf(6,11);
			Nixie_SetBuf(7,Minisec/10);
			Nixie_SetBuf(8,Minisec%10);
		
	}
}

//计时中断函数
void Sec_Loop()
{
	if(Runflag)
	{
	Minisec++;
	if(Minisec>=100)
	{
		Minisec=0;
		Sec++;
		if(Sec>=60)
		{
			Sec=0;
			Min++;
			if(Min>=60)
			{
				Min=0;
			}
		}
	}
}
}


void Timer0_Routine() interrupt 1  //ÖжϺ¯Êý
{
	static unsigned int T0Count1,T0Count2,T0Count3; 
	TL0 = 0x66;		//ÉèÖö¨Ê±Æ÷³õÖµ
	TH0 = 0xFC;		//ÉèÖö¨Ê±Æ÷³õÖµ
	T0Count1++;
	if(T0Count1>=20)  
	{
		T0Count1=0;
		Key_Loop();
	}
	T0Count2++;
	if(T0Count2>=2)   
	{
		T0Count2=0;
		Nixie_Loop();
	}
	T0Count3++;
	if(T0Count3>=10)
	{
		T0Count3=0;
		Sec_Loop();
	}
}

13.1 DS18B20——温度传感器

1.介绍

 2.引脚及应用电路

3.内部结构框图

最后一列分别是:温度传感器(内部的模拟温度传感器,能进行数据的转换),存储高报警位的EEPROM,存储低报警位的EEPROM,调节精度,校验码

前两个字节分别存储数据的低位和高位

总体思路:先发送温度转换的指令,再发送读数据的指令。因此,我们接下来要学习,如何通过单总线来发送数据和接收数据。

4.单总线介绍

①:介绍

  ②:时序结构

 

5.DS18B20操作流程

 本节需要的指令SKIP ROM,CONVERT T:让温度转换器进行数据的转换,READ SCRATCHPAD:读暂存器,依次读出每一个字节。

 

 

包括符号,整数部分,小数部分,其中负数以补码形式存储

13.2示例代码

1-DS18B20温度读取

①:整体思路——先写OneWire模块(包括初始化,发送一位,读出一位,发送一个字节,接收一个字节),再写DS18B20模块(包括温度转换指令函数,读取数据函数),最后编写main函数。

②:OneWire模块编写

③:DS18B20模块编写

④:main函数

实例代码2-DS18B20温度报警器

①:要用到的模块:AT24C02,Delay,DS18B20,LCD1602,OneWire,I2C,key,Timer0
②:main函数
#include <REGX52.H>
#include "DS18B20.h"
#include "key.h"
#include "LCD1602.h"
#include "Delay.h"
#include "AT24C02.h"
#include "Timer0.h"

float T,TShow;
char TLow,THigh;
unsigned char KeyNum;

void main()
{
	DS18B20_ConverT();
	Delay(1000);
	THigh=AT24C02_ReadByte(0);
	TLow=AT24C02_ReadByte(1);
	if(THigh>125 || TLow<-55 || THigh<=TLow)
	{
		THigh=20;
		TLow=15;
	}
	LCD_Init();
	LCD_ShowString(1,1,"T:");
	LCD_ShowString(2,1,"TH:");
	LCD_ShowString(2,9,"TL:");
	LCD_ShowSignedNum(2,4,THigh,3);
	LCD_ShowSignedNum(2,12,TLow,3);
	Timer0_Init();
	while(1)
	{
		KeyNum=Key();
		/*温度读取及显示*/
		DS18B20_ConverT();
		T=DS18B20_ReadT();
		if(T<0)
		{
			LCD_ShowChar(1,3,'-');
			TShow=-T;
		}
		else 
		{
			LCD_ShowChar(1,3,'+');
			TShow=T;
		}
		LCD_ShowNum(1,4,TShow,3);
		LCD_ShowChar(1,7,'.');
		LCD_ShowNum(1,8,(unsigned long)(TShow*100)%100,2);
		
		/*阈值判断及显示*/
		if(KeyNum)
		{
		if(KeyNum==1)
		{
			THigh++;
			if(THigh>125){THigh=125;}
		}
		if(KeyNum==2)
		{
			THigh--;
			if(THigh<=TLow){THigh++;}
		}
		if(KeyNum==3)
		{
			TLow++;
			if(TLow>=THigh){TLow--;}
		}
		if(KeyNum==4)
		{
			TLow--;
			if(TLow<-55){TLow=-55;}
		}
		LCD_ShowSignedNum(2,4,THigh,3);
		LCD_ShowSignedNum(2,12,TLow,3);
		AT24C02_WriteByte(0,THigh);
		Delay(5);
		AT24C02_WriteByte(1,TLow);
		Delay(5);
	}
		if(T>THigh)
		{
			LCD_ShowString(1,13,"OV:H");
		}
		else if(T<TLow)
		{
			LCD_ShowString(1,13,"OV:L");
		}
		else {LCD_ShowString(1,13,"OV: ");}
	}
}


void Timer0_Routine() interrupt 1  //中断函数
{
	static unsigned int T0Count; 
	TL0 = 0x66;		
	TH0 = 0xFC;		
	T0Count++;
	if(T0Count>=20)   
	{
		T0Count=0;
		Key_Loop();
	}
}

为防止定时器扫描按键时会打断OneWire的传送与接收

可以在OneWire里先关闭EA,再打开EA。

14-1 LCD1602液晶显示屏

1.介绍

 2.引脚及应用电路

   3.内部结构框图

 4.存储器结构

 5.时序结构

 6.LCD1602指令集

 

7.LCD1602操作流程

14-2 功能代码编写

之后可以在main函数里调用流动的指令,实现流动字幕

15-1 直流电机

1.介绍

2.电机驱动电路

大功率器件直接驱动:只能使电机朝着一个方向转,不具备调换电机正反方向的功能。

H桥驱动:可以控制电机正反转。

3.电机调速——PWM介绍

给电全速转,不给电不转,利用惯性,可以设置一个在周期内高电平与低电平的时间,使之呈现中间速度

 

15-2 示例代码

1.LED呼吸灯——用来理解PWM

2.电机调速 

①:创建新工程,需要用到的模块:Delay,Timer0,Nixie,key
②:main函数编写
#include <REGX52.H>
#include "Delay.h"
#include "key.h"
#include "Nixie.h"
#include "Timer0.h"

sbit Motor=P1^0;  //位声明

unsigned char Counter,Compare;
unsigned char KeyNum,Speed;

void main()
{
	Timer0_Init();
	while(1)
	{
		KeyNum=Key();
		if(KeyNum==1)
		{
			Speed++;
			Speed%=4;
			if(Speed==0){Compare=0;}
			if(Speed==1){Compare=50;}
			if(Speed==2){Compare=75;}
			if(Speed==3){Compare=100;}

		}
		Nixie(1,Speed);
	}
}

void Timer0_Routine() interrupt 1  
{
	TL0 = 0xA4;		
	TH0 = 0xFF;		
	Counter++;
	if(Counter>=100) {Counter=0;}
	if(Counter<Compare)
	{
		Motor=1;
	}
	else
	{
		Motor=0;
	}
}

16-1 AD/DA

1.介绍

2.硬件电路模型

3.实际硬件电路

4.运算放大器

四大运算放大器的经典电路

5.DA原理

6.AD

7.AD/DA性能指标

8.XPT2046

SPI通信:CS:片选,DCLK:时钟,DIN:数据输入,DOUT:数据输出

每个从机单独有个CS与主机连接,剩下三根线共用

16-2.实例代码

1.AD模数转换——在LCD上显示滑变,热敏电阻,光敏电阻的值

①:需要的模块:Delay,LCD,XPT2046
②:XPT2046模块的编写

先进行位声明,再写读取AD值的函数,读AD值,要给一个命令字,说明以什么模式读取的什么部位的AD值

#include <REGX52.H>
#include "Delay.h"

sbit XPT2046_CS=P3^5;
sbit XPT2046_DCLK=P3^6;
sbit XPT2046_DIN=P3^4;
sbit XPT2046_DOUT=P3^7;

unsigned int XPT2046_ReadAD(unsigned char Command)
{
	unsigned char i;
	unsigned int ADValue;
	XPT2046_DCLK=0;
	XPT2046_CS=0;
	
	for(i=0;i<8;i++)
	{
	XPT2046_DIN=Command&(0x80>>i);
	XPT2046_DCLK=1;
		Delay(1);
	XPT2046_DCLK=0;
	}
	for(i=0;i<16;i++)
	{
	XPT2046_DCLK=1;
		Delay(1);
	XPT2046_DCLK=0;
	if(XPT2046_DOUT) {ADValue|=(0x8000>>i);}
	}
	XPT2046_CS=1;
	if(Command&0x08)
	{
	return ADValue>>8;
	}
	else
	{
		return ADValue>>4;
	}
}

③:main函数

2.DA数模转换——实现呼吸灯

①:用PWM的工程改造即可
②:换IO口,把Motor换成DA的口,再结合PWM调速,即可实现。

17-1 红外遥控

1.介绍

2.硬件电路

3.基本发送与接收

4.NEC编码

5.51单片机的外部中断

17-2.实例代码

1.红外遥控显示按键键码

①:配置外部中断INT0,并编写设置定时器时间和读取定时器时间,以及控制定时器开启的子函数

②:编写红外遥控模块IR
#include <REGX52.H>
#include "Timer0.h"
#include "Int0.h"

unsigned int IR_Time;
unsigned char IR_State;

unsigned char IR_Data[4];  //数据缓存器
unsigned char IR_pData;   //指向存储的位置

unsigned char IR_DataFlag;
unsigned char IR_RepeatFlag;
unsigned char IR_Address;
unsigned char IR_Command;

void Int0_Routine(void) interrupt 0
{
	if(IR_State==0)
	{
		Timer0_SetCounter(0);
		Timer0_Run(1);
		IR_State=1;
	}
	else if(IR_State==1)
	{	
		IR_Time=Timer0_GetCounter();
		Timer0_SetCounter(0);
		if(IR_Time>12442-500 && IR_Time<12442+500)  //根据时间来判断是否是start信号
		{
			IR_State=2;
		}
		else if(IR_Time>10368-500 && IR_Time<10368+500)  //判断是否是repeat信号
		{
			IR_RepeatFlag=1;
			Timer0_Run(0);
			IR_State=0;
		}
		else   //其它未知的出错情况
		{
			IR_State=1;
		}
	}
	else if(IR_State==2)
	{
		IR_Time=Timer0_GetCounter();
		Timer0_SetCounter(0);
		if(IR_Time >1032-500 && IR_Time<1032+500)
		{
			IR_Data[IR_pData/8]&=~(0x01<<(IR_pData%8));
			IR_pData++;
		}
		else if(IR_Time >2074-500 && IR_Time<2074+500)
		{
			IR_Data[IR_pData/8]|=(0x01<<(IR_pData%8));
			IR_pData++;
		}
		else 
		{
			IR_pData=0;
			IR_State=1;
		}
		if(IR_pData>=32)
		{
			IR_pData=0;
			if((IR_Data[0]==~IR_Data[1]) && (IR_Data[2]==~IR_Data[3]) )
			{
				IR_DataFlag=1;
				IR_Address=IR_Data[0];
				IR_Command=IR_Data[2];
			}
			Timer0_Run(0);
			IR_State=0;
		}
	}
}

③:编写main
#include <REGX52.H>
#include "Delay.h"
#include "LCD1602.h"
#include "Int0.h"
#include "IR.h"

unsigned char Num;
unsigned char Address;
unsigned char Command;

void main()
{
	LCD_Init();
	IR_Init();
	LCD_ShowString(1,1,"ADDR  CMD  NUM");
	LCD_ShowString(2,1,"00    00   000");
	while(1)
	{
		if(IR_GetDataFlag() || IR_GetRepeatFlag())  //可实现连加连减功能
		{
			Address=IR_GetAddress();
			Command=IR_GetCommand();
			
			LCD_ShowHexNum(2,1,Address,2);
			LCD_ShowHexNum(2,7,Command,2);
			
			if(Command==IR_VOL_MINUS)
			{
				Num--;
			}
			if(Command==IR_VOL_ADD)
			{
				Num++;
			}
			
			LCD_ShowNum(2,12,Num,3);
		}
	}
}


2.红外遥控控制电机调速

①:改造电机调速的文件
②:因为按键和红外遥控都用了定时器0,因此我们先把原本按键的定时器0改为定时器1

main函数部分做对应调整

③:Motor模块封装

④:加入红外模块,改造main函数

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2262404.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【青牛科技】D8563是低功耗的CMOS实时时钟/日历电路,它提供一个可编程时钟输出,一个中断输出和掉电检测器,所有的地址和数据通过IC总线接口串行传递。

概述&#xff1a; D8563是低功耗的CMOS实时时钟/日历电路,它提供一个可编程时钟输出&#xff0c;一个中断输出和掉电检测器&#xff0c;所有的地址和数据通过IC总线接口串行传递。最大总线速度为400Kbitss每次读写数据后&#xff0c;内嵌的字地址寄存器会自动产生增量。 主要特…

安卓获取所有可用摄像头并指定预览

在Android设备中&#xff0c;做预览拍照的需求的时候&#xff0c;我们会指定 CameraSelector DEFAULT_FRONT_CAMERA前置 或者后置CameraSelector DEFAULT_BACK_CAMERA 如果你使用的是平板或者工业平板&#xff0c;那么就会遇到多摄像头以及外置摄像头问题&#xff0c;简单的指…

R语言学习笔记-1

1. 基础操作和函数 清空环境&#xff1a;rm(list ls()) 用于清空当前的R环境。 打印输出&#xff1a;print("Hello, world") 用于输出文本到控制台。 查看已安装包和加载包&#xff1a; search()&#xff1a;查看当前加载的包。install.packages("package_na…

Windows如何安装go环境,离线安装beego

一、安装go 1、下载go All releases - The Go Programming Language 通过网盘分享的文件&#xff1a;分享的文件 链接: https://pan.baidu.com/s/1MCbo3k3otSoVdmIR4mpPiQ 提取码: hxgf 下载amd64.zip文件&#xff0c;然后解压到指定的路径 2、配置环境变量 需要新建两个环境…

Mac上使用ln指令创建软链接、硬链接

在Mac、Linux和Unix系统中&#xff0c;软连接&#xff08;Symbolic Link&#xff09;和硬连接&#xff08;Hard Link&#xff09;是两种不同的文件链接方式。它们的主要区别如下&#xff1a; 区别&#xff1a; 硬连接&#xff1a; 不能跨文件系统。不能链接目录&#xff08;为…

Unity A*算法实现+演示

注意&#xff1a; 本文是对基于下方文章链接的理论&#xff0c;并最终代码实现&#xff0c;感谢作者大大的描述&#xff0c;非常详细&#xff0c;流程稍微做了些改动&#xff0c;文末有工程网盘链接&#xff0c;感兴趣的可以下载。 A*算法详解(个人认为最详细,最通俗易懂的一…

博弈论3:图游戏SG函数(Graph Games)

目录 一、图游戏是什么 1.游戏特征 2.游戏实例 二、图游戏的必胜策略 1.SG 函数&#xff08;Sprague-Grundy Function&#xff09; 2.必胜策略&#xff08;利用SG函数&#xff09; 3.拿走游戏转化成图游戏&#xff08;Take-away Game -> Graph Game&#xff09; 一、图…

0101多级nginx代理websocket配置-nginx-web服务器

1. 前言 项目一些信息需要通过站内信主动推动给用户&#xff0c;使用websocket。web服务器选用nginx&#xff0c;但是域名是以前通过阿里云申请的&#xff0c;解析ip也是阿里云的服务器&#xff0c;甲方不希望更换域名。新的系统需要部署在内网服务器&#xff0c;简单拓扑图如…

qt-C++笔记之自定义类继承自 `QObject` 与 `QWidget` 及开发方式详解

qt-C笔记之自定义类继承自 QObject 与 QWidget 及开发方式详解 code review! 参考笔记 1.qt-C笔记之父类窗口、父类控件、对象树的关系 2.qt-C笔记之继承自 QWidget和继承自QObject 并通过 getWidget() 显示窗口或控件时的区别和原理 3.qt-C笔记之自定义类继承自 QObject 与 QW…

Elastic 8.17:Elasticsearch logsdb 索引模式、Elastic Rerank 等

作者&#xff1a;来自 Elastic Brian Bergholm 今天&#xff0c;我们很高兴地宣布 Elastic 8.17 正式发布&#xff01; 紧随一个月前发布的 Elastic 8.16 之后&#xff0c;我们将 Elastic 8.17 的重点放在快速跟踪关键功能上&#xff0c;这些功能将带来存储节省和搜索性能优势…

[C++]类的继承

一、什么是继承 1.定义&#xff1a; 在 C 中&#xff0c;继承是一种机制&#xff0c;允许一个类&#xff08;派生类&#xff09;继承另一个类&#xff08;基类&#xff09;的成员&#xff08;数据和函数&#xff09;。继承使得派生类能够直接访问基类的公有和保护成员&#xf…

Docker 用法详解

文章目录 一、Docker 快速入门1.1 部署 MYSQL1.2 命令解读&#xff1a; 二、Docker 基础2.1 常见命令&#xff1a;2.1.1 命令介绍&#xff1a;2.1.2 演示&#xff1a;2.1.3 命令别名&#xff1a; 2.2 数据卷&#xff1a;2.2.1 数据卷简介&#xff1a;2.2.2 数据卷命令&#xff…

【自动化】Python SeleniumUtil 油猴 工具 自动安装用户脚本

【自动化】Python SeleniumUtil 油猴 工具 【自动化】Python SeleniumUtil 工具-CSDN博客【自动化】Python SeleniumUtil 工具。https://blog.csdn.net/G971005287W/article/details/144565691 油猴工具 import timefrom selenium.webdriver.support.wait import WebDriverW…

盛元广通畜牧与水产品检验技术研究所LIMS系统

一、系统概述 盛元广通畜牧与水产品检验技术研究所LIMS系统集成了检测流程管理、样品管理、仪器设备管理、质量控制、数据记录与分析、合规性管理等功能于一体&#xff0c;能够帮助实验室实现全流程的数字化管理。在水产、畜牧产品的质检实验室中&#xff0c;LIMS系统通过引入…

clickhouse-数据库引擎

1、数据库引擎和表引擎 数据库引擎默认是Ordinary&#xff0c;在这种数据库下面的表可以是任意类型引擎。 生产环境中常用的表引擎是MergeTree系列&#xff0c;也是官方主推的引擎。 MergeTree是基础引擎&#xff0c;有主键索引、数据分区、数据副本、数据采样、删除和修改等功…

GEE+本地XGboot分类

GEE本地XGboot分类 我想做提取耕地提取&#xff0c;想到了一篇董金玮老师的一篇论文&#xff0c;这个论文是先提取的耕地&#xff0c;再做作物分类&#xff0c;耕地的提取代码是开源的。 但这个代码直接在云端上进行分类&#xff0c;GEE会爆内存&#xff0c;因此我准备把数据下…

Docker搭建kafka环境

系统&#xff1a;MacOS Sonoma 14.1 Docker版本&#xff1a;Docker version 27.3.1, build ce12230 Docker desktop版本&#xff1a;Docker Desktop 4.36.0 (175267) 1.拉取镜像 先打开Docker Desktop&#xff0c;然后在终端执行命令 docker pull lensesio/fast-data-dev …

校园点餐订餐外卖跑腿Java源码

简介&#xff1a; 一个非常实用的校园外卖系统&#xff0c;基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化&#xff0c;提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合&am…

Linux文件属性 --- 硬链接、所有者、所属组

三、硬链接数 1.目录 使用“ll”命令查看&#xff0c;在文件权限的后面有一列数字&#xff0c;这是文件的硬链接数。 对于目录&#xff0c;硬链接的数量是它具有的直接子目录的数量加上其父目录和自身。 下图的“qwe”目录就是“abc”目录的直接子目录。 2.文件 对于文件可…

Centos7 部署ZLMediakit

1、拉取代码 #国内用户推荐从同步镜像网站gitee下载 git clone --depth 1 https://gitee.com/xia-chu/ZLMediaKit cd ZLMediaKit #千万不要忘记执行这句命令 git submodule update --init 2、安装编译器 sudo yum -y install gcc 3、安装cmake sudo yum -y install cmake 4…