基于STC12C5A60S2系列1T 8051单片机通过单个按键单击次数实现开关机应用
基于STC12C5A60S2系列1T 8051单片机通过单个按键单击次数实现开关机应用
- STC12C5A60S2系列1T 8051单片机管脚图
- STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置
- STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍
- 基于STC12C5A60S2系列1T 8051单片机通过单个按键单击次数实现开关机功能
STC12C5A60S2系列1T 8051单片机管脚图
STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置
STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍
基于STC12C5A60S2系列1T 8051单片机通过单个按键单击次数实现开关机功能
main.c
#include "STC12C5A60S2.h"
#include "Timer0.h"
#include "Key.h"
#include "Digitron.h"
sbit LED = P1^2;//位定义LED灯为单片机P1.2脚void main()//主函数
{Timer0Init();//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHzDigitronBootDisplay();//数码管开机显示函数while(1)//主循环{KeyScanResult();//按键扫描结果函数}}
Key.c
#include "Key.h"
#include "Timer0.h"
#define uchar unsigned char //定义无符号字符
#define uint unsigned int //定义无符号整形
#define KeyPressDeshakeTime 15//自定义按键按下消抖时间为20ms
#define KeyLongPressDelayTime 100//自定义按键长按延时时间为200ms
uchar KeyTypePressCount = 0;//定义按键类型按下计数变量为0
//uchar KeyTypePressCountFlag = 0;//定义按键类型按下计数标志位变量为0
//uchar ClearKeyPressFlag = 0;//定义清零按键按下标志位变量为0
//uchar SetKeyFlag = 0;//定义设置按键标志位变量为0
//uchar SetKeyPressCountFlag = 1;//定义设置按键按下计数标志位变量为1
uchar SetKeyShortPressLcokFlag = 0;//定义设置按键短按按下锁定标志位变量为0
//uchar SetKeyShortPressCount = 0;//定义设置按键短按按下计数变量为0
//uchar SetKeyShortPressFlag = 0;//定义设置按键短按按下标志位变量为0
//uchar SetKeyShortPressCountFlag = 1;//定义设置按键短按按下计数标志位变量为1
uchar SetKeyLongPressLcokFlag = 0;//定义设置按键长按按下锁定标志位变量为0
//uchar SetKeyLongPressCount = 0;//定义设置按键长按按下计数变量为0
//uchar SetKeyLongPressFlag = 0;//定义设置按键长按按下标志位变量为0
//uint SetKeyLongPressCountFlag = 1;//定义设置按键长按按下计数标志位变量为0
uint KeyPressDelayTime = 0;//定义按键按下延时时间变量为0
uint KeyLiftDelayTime = 0;//定义按键弹起延时时间变量为0
uint KeyPressNumber = 0;//定义按键按下数值变量为0
uint KeyType = 0;//定义按键类型变量为0
// uint KeyScan ()//带按键返回值的按键扫描函数void KeyScan ()//按键扫描函数
{if(SetKey == 0)//设置按键按下{ KeyPressDelayTime++;//按键按下延时时间变量自加if(KeyPressDelayTime > KeyPressDeshakeTime)//判断按键按下延时时间变量是否大于按键按下消抖时间{KeyPressDelayTime = 0;//按键按下延时时间变量清0 SetKeyShortPressFlag = 1;//设置按键短按按下标志位变量置1 }}else//设置按键弹起或没按下{KeyPressDelayTime = 0;//按键按下延时时间变量清0 重启下一步按键按下延时操作KeyLiftDelayTime++;//按键弹起延时时间变量自加if(KeyLiftDelayTime > KeyPressDeshakeTime)//判断按键弹起延时时间变量是否大于按键按下消抖时间{ KeyLiftDelayTime = 0;//按键弹起延时时间变量清0 重启下一步按键弹起延时操作if(SetKeyShortPressFlag == 1)//判断设置按键短按按下标志位变量是否为1 表示设置按键短按按下过 { SetKeyShortPressFlag = 0;//设置按键短按按下标志位变量置0 为了重启下一步设置按键短按按下操作KeyType = 1;//此处是设置按键短按 对于按键计数或按键类型触发操作 建议要放在按键弹起后再计数或触发 此处就是SetKeyShortPressCountFlag++;//设置按键短按按下计数标志位变量自加 对于按键计数或按键类型触发操作 建议要放在按键弹起后再计数或触发 此处就是}}}if(SetKeyShortPressCountFlag > 1)//判断设置按键短按按下计数标志位变量是否大于1 此处是设置按键第2次短按后松手 {SetKeyFlag = 0;//设置按键标志位变量清0 触发关机KeyType = 0;//按键类型清0 为了跳出设置按键短按 让设置按键可以进行下一步短按或再次长按SetKeyShortPressCountFlag = 0;//设置按键短按按下计数标志位变量清0 让设置按键可以进行下一步短按} }void KeyScanResult()//按键扫描结果函数
{switch(KeyType)//按键类型筛选位{case 1 ://单击或连击增加触发位
// KeyTypePressCountFlag = 1;//按键类型计数标志位变量置1 表示设置按键短按过
// KeyPressNumber++;//按键按下数值自加if(SetKeyShortPressCountFlag == 1)//判断设置按键短按按下计数标志位变量是否等于1 此处是设置按键第1次短按后松手{SetKeyFlag = 1;//设置按键标志位变量置1 触发开机 } KeyType = 0;//按键类型清0break;//跳出
// case 2 ://单击或连击减少触发位
// KeyPressNumber--;//按键按下数值自减
// if(KeyPressNumber == 0 | KeyPressNumber == 65535)//如果按键按下数值等于0或65535
// {
// KeyPressNumber = 0;//按键按下数值置0
// }
// KeyType = 0;//按键类型清0
// break;//跳出
// case 3 ://长按触发位
// KeyPressNumber++;//按键按下数值自加
// if(KeyPressNumber > 9999)//如果按键按下数值大于9999
// {
// KeyPressNumber = 0;//按键按下数值清0
// }
// KeyType = 0;//按键类型清0
// break;//跳出default:break;//跳出}}
Key.h
#ifndef _KEY_H
#define _KEY_H
#include "STC12C5A60S2.h"
#define uchar unsigned char //定义无符号字符
#define uint unsigned int //定义无符号整形
//sbit AddKey = P3^5;//增加按键
//sbit DecKey = P3^4;//减少按键
sbit SetKey = P3^3;//设置按键
//sbit ClearKey = P3^2;//复位按键
sbit led0 = P1^5;//短按LED指示灯
sbit led1 = P1^6;//长按LED指示灯
sbit led2 = P1^7;//复位LED指示灯
extern uchar SetKeyFlag;声明设置按键标志位变量
//extern uchar KeyTypePressCount;//声明按键类型按下计数变量
//extern uchar KeyTypePressCountFlag;//声明按键类型按下计数标志位变量
//extern uchar ClearKeyPressFlag;//声明清零按键按下标志位变量
//extern uchar SetKeyFlag;//声明设置按键标志位变量
//extern uchar SetKeyPressCountFlag;//声明设置按键按下计数标志位变量
//extern uchar SetKeyShortPressLcokFlag;//声明设置按键短按按下锁定标志位变量
//extern uchar SetKeyShortPressCount;//声明设置按键短按按下计数变量
extern uchar SetKeyShortPressFlag;//声明设置按键短按按下标志位变量置
extern uchar SetKeyShortPressCountFlag;//声明设置按键短按按下计数标志位变量
//extern uchar SetKeyLongPressCount;//声明设置按键长按按下计数变量
//extern uchar SetKeyLongPressLcokFlag;//声明设置按键长按按下锁定标志位变量
//extern uchar SetKeyLongPressFlag;//声明设置按键长按按下标志位变量
//extern uint SetKeyLongPressCountFlag;//声明设置按键长按按下计数标志位变量
extern uint KeyPressDelayTime;//声明按键按下延时时间变量 可被其他.c文件通过#include "其他.h"引用该变量
extern uint KeyLiftDelayTime;//声明按键弹起延时时间变量
extern uint KeyPressNumber;//声明按键按下数值变量
extern uint KeyType;//声明按键类型变量
void KeyScan ();//按键扫描函数
//extern uint KeyScan ();//带有按键返回值的按键扫描函数
void KeyScanResult();//按键扫描结果函数
//void KeyTypePressCountResult();//按键类型按下计数结果函数
#endif
Digitron.c
#include "Digitron.h"
//#include "Key.h"
#include "Timer0.h"
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
//uchar code DigitronBitCodeArray[] = {0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};//定义八位共阴数码管位码数组变量 为什么不是{0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f} 这才是定义八位共阴数码管位码数组变量 不对吗? 在不使用NPN三极管驱动 用单片机端口直接连接驱动 位码数组是对的 但数码管亮度不够 因此使用了NPN型三极管(比如S8050)来驱动共阴数码管位选 NPN型三极管(比如S8550)基极输入高电平才能导通 解释:共阴数码管 阴极是公共端 对应位选 低电平选通 阳极是显示端 对应段选 高电平选通 由于共阴数码管阴极公共端接单片机来驱动共阴数码管阳极显示端 共阴数码管的亮度会比较低 需要借助NPN型三极管的集电极连接共阴数码管阴极公共端 而NPN型三极管的基电极串个限流电阻连接单片机端口 通过单片机端口输出高电平到NPN型三极管的基电极 从而导通NPN型三极管 放大流过共阴数码管的电流 这样共阴数码管的亮度才会比较亮
//uchar code DigitronSegmentCodeArray[] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x40,0x00};//定义共阴数码管显示0到F数据及符号“—”及熄灭数组变量
//uchar code DigitronSegmentCodeOfPointArray[] = {0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,0x87,0xff,0xef,0xf7,0xfc,0xb9,0xde,0xf9,0xf1,0x40,0x00};//定义带小数点共阴数码管显示0.到F.数据及符号“—”及熄灭数组变量
uchar code DigitronBitCodeArray[] = {0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//定义八位共阳数码管位码数组变量 为什么不是{0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80} 这才是定义八位共阳数码管位码数组变量 不对吗? 在不使用PNP三极管驱动 用单片机端口直接连接驱动 位码数组是对的 但数码管亮度不够 因此使用了PNP型三极管(比如S8550)来驱动共阳数码管位选 PNP型三极管(比如S8550)基极输入低电平才能导通 解释:共阳数码管 阳极是公共端 对应位选 高电平选通 阴极是显示端 对应段选 低电平选通 由于共阳数码管阳极公共端接单片机来驱动共阳数码管阴极显示端 共阳数码管的亮度会比较低 需要借助PNP型三极管的集电极连接共阳数码管阳极公共端 而PNP型三极管的基电极串个限流电阻连接单片机端口 通过单片机端口输出低电平到PNP型三极管的基电极 从而导通PNP型三极管 由外接电源来驱动共阳数码管 这样共阳数码管的亮度才会比较亮
uchar code DigitronSegmentCodeArray[] = {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0xbf,0xff};//定义共阳数码管显示0到F数据及符号“—”及熄灭数组变量
//uchar code DigitronSegmentCodeOfPointArray[] = {0x40,0x79,0x24,0x30,0x19,0x12,0x02,0x78,0x00,0x10,0x08,0x03,0x46,0x21,0x06,0x0e,0xbf,0xff};//定义带小数点共阳数码管显示0.到F.数据及符号“—”及熄灭数组变量
uchar DigitronCacheDataArray[] = {0,0,0,0};//定义共阳数码管缓存数据数组变量
uchar DigitronBootTimerFlag = 1;//定义共阳数码管开机时间标志位变量
uint DigitronBootTimer = 0;//定义数码管开机时间变量
//extern uchar Data;//取用外部定义的数据变量
//extern uint KeyPressNumber;//如果在Key.c文件下已经定义按键按下数值变量KeyPressNumber 则以此语句来引用Key.c文件下的按键按下数值变量KeyPressNumber 否则先在Key.c文件下定义按键按下数值变量KeyPressNumber 接着在Key.h文件下的用extern关键字声明按键按下数值变量KeyPressNumber 最后通过在其他.c文件下#include "Key.h" 就可以引用在Key.c文件下已经定义的按键按下数值变量KeyPressNumbervoid DigitronBootDisplay()//数码管开机显示函数
{do{//if(DigitronBootTimer == 500 )//如果数码管开机时间等于1sLED0 = ~ LED0;//LED灯亮灭更新}while(DigitronBootTimer <= 500);//当数码管开机时间小于5sDigitronBootTimerFlag = 0;//数码管开机时间标志位清0LED0 = 1;//LED灯熄灭}void DigitronDisplayDataSplit()//数码管显示数据分解函数
{DigitronCacheDataArray[0] = KeyPressNumber / 1000;//数码管千位数据显示DigitronCacheDataArray[1] = KeyPressNumber / 100 % 10;//数码管百位数据显示DigitronCacheDataArray[2] = KeyPressNumber / 10 % 10;//数码管十位数据显示DigitronCacheDataArray[3] = KeyPressNumber % 10;//数码管个位数据显示// DigitronCacheDataArray[0] = Data / 1000;//数码管千位数据显示
// DigitronCacheDataArray[1] = Data / 100 % 10;//数码管百位数据显示
// DigitronCacheDataArray[2] = Data / 10 % 10;//数码管十位数据显示
// DigitronCacheDataArray[3] = Data % 10;//数码管个位数据显示if(KeyPressNumber < 1000)//如果累积时间变量小于1000{DigitronCacheDataArray[0] = 17;//数码管千位数据不显示}else{DigitronCacheDataArray[0] = KeyPressNumber / 1000;//数码管千位数据显示}if(KeyPressNumber < 100)//如果累积时间变量小于100{DigitronCacheDataArray[1] = 17;//数码管百位数据不显示}else{DigitronCacheDataArray[1] = KeyPressNumber / 100 % 10;//数码管百位数据显示}if(KeyPressNumber < 10)//如果累积时间变量小于10{DigitronCacheDataArray[2] = 17;//数码管十位数据不显示}else{DigitronCacheDataArray[2] = KeyPressNumber / 10 % 10;//数码管十位数据显示}DigitronCacheDataArray[3] = KeyPressNumber % 10;//数码管个位数据显示}void DigitronDisplayData()//数码管显示数据函数
{ static uchar i = 0;//定义静态数码管管位变化变量switch(i)//数码管管位变化筛选{case 0 ://数码管千位显示DigitronSegmentCode = 0xff;//数码管段码消影DigitronSegmentCode = DigitronSegmentCodeArray[DigitronCacheDataArray[0]];//数码管千位的段码显示DigitronBitCode = DigitronBitCodeArray[0];//数码管千位码显示i++;//数码管管位变化自加1break;//跳出case 1 ://数码管百位显示DigitronSegmentCode = 0xff;//数码管段码消影DigitronSegmentCode = DigitronSegmentCodeArray[DigitronCacheDataArray[1]];//数码管百位的段码显示DigitronBitCode = DigitronBitCodeArray[1];//数码管百位码显示i++;//数码管管位变化自加1break;//跳出 case 2 ://数码管十位显示DigitronSegmentCode = 0xff;//数码管段码消影DigitronSegmentCode = DigitronSegmentCodeArray[DigitronCacheDataArray[2]];//数码管十位的段码显示DigitronBitCode = DigitronBitCodeArray[2];//数码管十位码显示i++;//数码管管位变化自加1break;//跳出case 3 ://数码管个位显示DigitronSegmentCode = 0xff;//数码管段码消影DigitronSegmentCode = DigitronSegmentCodeArray[DigitronCacheDataArray[3]];//数码管个位的段码显示DigitronBitCode = DigitronBitCodeArray[3];//数码管个位码显示i = 0;//数码管管位变化清0break;//跳出default:break;//跳出}}
Digitron.h
#ifndef _DIGITRON_H
#define _DIGITRON_H
#include "STC12C5A60S2.h"
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
#define DigitronSegmentCode P0//自定义共阳数码管段码端口为单片机P0组引脚
#define DigitronBitCode P2//自定义共阳数码管位码端口为单片机P2组引脚
sbit LED0 = P1^0;//位定义LED灯为单片机P1.0脚
extern uchar code DigitronBitCodeArray[];//声明八位共阳数码管位码数组变量 可被其他.c文件通过#include "其他.h"引用该变量
extern uchar code DigitronSegmentCodeArray[];//声明共阳数码管显示0到F数据及符号“—”及熄灭数组变量 可被其他.c文件通过#include "其他.h"引用该变量
extern uchar DigitronCacheDataArray[];//声明共阳数码管缓存数据数组变量 可被其他.c文件通过#include "其他.h"引用该变量
extern uchar DigitronBootTimerFlag;//声明共阳数码管开机时间标志位变量 可被其他.c文件通过#include "其他.h"引用该变量
extern uint DigitronBootTimer;//声明数码管开机时间变量 可被其他.c文件通过#include "其他.h"引用该变量
void DigitronBootDisplay();//声明数码管开机显示函数
void DigitronDisplayDataSplit();//声明数码管显示数据分解函数
void DigitronDisplayData();//声明数码管显示数据函数
#endif
Timer0.c
#include "Timer0.h"
#include "Key.h"
#include "Digitron.h"
/*****关于通过特殊功能寄存器AUXR设定定时器/计数器模式为1T或12T模式不需分频或需12分频8051系列单片机定时器初值(定时计数初值)计算的知识点*****//****时钟周期(又称振荡周期):单片机晶振频率的倒数 例:单片机晶振频率12MHz 则时钟周期=[1/(12*10^6)Hz]s=0.000000083s=0.000083ms=0.083us机器周期:单片机执行一条指令过程中需要完成一个基本操作(如:取指、译码、执行等基本操作)所需的时间 8051系列单片机的一个机器周期由6个S周期(状态周期)组成 一个时钟周期定义为一个节拍(用P表示) 二个节拍定义为一个状态周期(用S表示) 那么8051单片机的机器周期由6个状态周期组成 也就是说一个机器周期=6个状态周期=12个时钟周期=[12x[1/(12*10^6)Hz]s]s=0.000001s=0.001ms=1us指令周期:单片机取出一条指令且执行完这条指令所需的时间以上三者间的关系:指令周期>机器周期>时钟周期一、以下是8051单片机定时器用12分频计算定时器初值的一种计算公式(以单片机晶振频率为12MHz 定时器0工作模式为16位定时模式1 需要定时1ms来计算):0、计算nT单片机机器周期T公式:T=n*(1/晶振频率)=几us1、一个机器周期=12个时钟周期=12乘以单片机晶振频率的倒数=12*[1/(12*10^6)Hz]s=0.000001s=0.001ms=1us2、定时时间=定时计数*一个机器周期 1ms=定时计数*1us 定时计数=1ms/1us=1000us/1us=1000次3、定时器初值(定时计数初值)=2^n-定时计数 n为几位定时器 此处n=16 则定时器初值(定时计数初值)=2^16-1000=65536-1000=64536 把64536转化成十六进制 拆开成高八位和低八位 高八位放TH0=0xfc或(65536-64536)/256 低八位放TL0=0x18或(65536-64536)%256二、以下是8051单片机定时器用12分频或不分频计算定时器初值的另外一种计算公式(以单片机晶振频率为12MHz 定时器0工作模式为16位定时模式1 需要定时1ms来计算):1、综合公式:定时器初值(定时计数初值)=2^n-(晶振频率/几分频/定时频率) n为几位定时器 该公式常用于脉冲宽度调制中运算 例如:利用8051系列单片机晶振频率为12MHz的定时器0的16位定时模式1来产生1KHz方波脉冲 由此可知:定时时间=1/定时频率=1/1000Hz=0.001s=1ms=1000us 进而可得:定时器初值(定时计数初值)=2^n-(晶振频率/几分频/定时频率)=2^16-(12MHz/12/1KHz)=2^16-(12*10^6)Hz/12/1000Hz)=65536-1000=64536 把64536转化成十六进制 拆开成高八位和低八位 高八位放TH0=0xfc或(65536-64536)/256或Value >> 8 低八位放TL0=0x18或(65536-64536)%256或=Value 2、TH0 = Value >> 8;TL0 = Value;该两句代码解释如下:(1)、TH0 = Value >> 8相当于TH0 = (65536-10000)/256=55536/256=216.9375 分析:65536-10000=55536转化成二进制为11011000 11110000 55536/256=216.9375转化成二进制为11011000 由此可看出Value为(65536-10000)=55536的二进制数11011000 11110000右移8位就可以得到55536/256=216.9375的二进制数11011000(2)、TL0 = Value相当于TL0 = (65536-时器初值的另外一种计算公式(以单片机晶振频率为12MHz 定时器0工作模式为16位定时模式1 需要定时1ms来计算):(一)、以下是8051单片机定时器用12分频计算定时器初值:定时器初值(定时计数初值)=2^n-(晶振频率/几分频/定时频率) n为几位定时器 该公式常用于脉冲宽度调制中运算 例如:利用8051系列单片机晶振频率为12MHz的定时器0的16位定时模式1来产生1KHz方波脉冲(相当于定时1ms) 由此可知:定时时间=1/定时频率=1/1000Hz=0.001s=1ms=1000us 进而可得:定时器初值(定时计数初值)=2^n-(晶振频率/几分频/定时频率)=2^16-(12MHz/12/1KHz)=2^16-(12*10^6)Hz/12/1000Hz)=65536-1000=64536 把64536转化成十六进制 拆开成高八位和低八位 高八位放TH0=0xfc或(65536-64536)/256或Value >> 8 低八位放TL0=0x18或(65536-64536)%256或=Value (二)、以下是8051单片机定时器不用分频计算定时器初值:定时器初值(定时计数初值)=2^n-(晶振频率/几分频/定时频率) n为几位定时器 该公式常用于脉冲宽度调制中运算 例如:利用8051系列单片机晶振频率为12MHz的定时器0的16位定时模式1来产生1KHz方波脉冲(相当于定时1ms) 由此可知:定时时间=1/定时频率=1/1000Hz=0.001s=1ms=1000us 进而可得:定时器初值(定时计数初值)=2^n-(晶振频率/几分频/定时频率)=2^16-(12MHz/1/1KHz)=2^16-(12*10^6)Hz/1/1000Hz)=65536-12000=53536 把53536转化成十六进制 拆开成高八位和低八位 高八位放TH0=0xd1或(65536-53536)/256或Value >> 8 低八位放TL0=0x20或(65536-53536)%256或=Value(三)、TH0 = Value >> 8;TL0 = Value;该两句代码解释如下:1、TH0 = Value >> 8相当于TH0 = (65536-10000)/256=55536/256=216.9375 分析:65536-10000=55536转化成二进制为11011000 11110000 55536/256=216.9375转化成二进制为11011000 由此可看出Value为(65536-10000)=55536的二进制数11011000 11110000右移8位就可以得到55536/256=216.9375的二进制数110110002、TL0 = Value相当于TL0 = (65536-10000)%256=55536%256=240 分析:65536-10000=55536转化成二进制为11011000 11110000 55536%256=240转化成二进制为11110000 由此可看出Value为(65536-10000)=55536的二进制数11011000 11110000取低8位就可以得到55536%256=240的二进制数11110000(四)、由定时器定时初值(定时计数初值)推导出定时器定时时间步骤如下:1、如果定时器定时初值(定时计数初值)是拆开成高八位和低八位赋值形式 如:TH0=0xfc TL0=0x18 先把高八位和低八位赋值组成一个十六位数据0xfc18 转化成十进制数据64536 用2^n-64536算出每秒产生的脉冲数 其中n为几位定时器 再根据公式计算定时时间 如:由公式:每秒产生的脉冲数=晶振频率/几分频/定时频率 转换成:每秒产生的脉冲数=晶振频率x定时频率/几分频 可求:定时频率=(每秒产生的脉冲数x几分频)/晶振频率 进而求出:定时时间=1/定时频率=1/[(每秒产生的脉冲数x几分频)/晶振频率] 转换成:晶振频率/(每秒产生的脉冲数x几分频)=定时时间2、如果定时器定时初值(定时计数初值)是十进制数据 如:64536 直接用2^n-64536算出每秒产生的脉冲数 其中n为几位定时器 再根据公式计算定时时间 如:由公式:每秒产生的脉冲数=晶振频率/几分频/定时频率 转换成:每秒产生的脉冲数=晶振频率x定时频率/几分频 可求:定时频率=(每秒产生的脉冲数x几分频)/晶振频率 进而求出:定时时间=1/定时频率=1/[(每秒产生的脉冲数x几分频)/晶振频率] 转换成:晶振频率/(每秒产生的脉冲数x几分频)=定时时间****/
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uintvoid Timer0Init()//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHz
{//AUXR &= 0x7f;//设定定时器/计数器模式为12TTMOD &= 0xf0;//设定定时器/计数器工作模式清0TMOD |= 0x01;//设定定时器/计数器为定时器 工作模式为16位定时器0模式1TH0 = 0xf8;//设定定时器0高8位初值TL0 = 0x30;//设定定时器0低8位初值TF0 = 0;//定时器0溢出中断标志位清0ET0 = 1;//打开定时器0中断开关EA = 1;//打开定时器中断总开关TR0 = 1;//打开定时器0开关} void Timer0() interrupt 1//定时器0的16位定时模式1用12分频定时2ms中断函数 晶振为12MHz
{TR0 = 0;//关定时器0开关if(DigitronBootTimerFlag == 1)//数码管开机时间标志位置1{DigitronBootTimer++;//数码管开机时间自加}if(DigitronBootTimerFlag == 0)//判断共阳数码管开机时间标志位是否等于0{ if(SetKeyFlag == 1)//开关机触发位 1是开机 0是关机{ DigitronDisplayDataSplit();//数码管显示数据分解函数DigitronDisplayData();//数码管显示数据函数}
// SetKeyScan();//设置按键扫描函数 该函数放在定时器定时2ms的中断函数中扫描KeyScan();//按键扫描函数 该函数放在定时器定时2ms的中断函数中扫描}TH0 = 0xf8;//设定定时器0计数高8位初值TL0 = 0x30;//设定定时器0计数低8位初值TR0 = 1;//开定时器0开关}
Timer0.h
#ifndef _TIMER0_H
#define _TIMER0_H
#include "STC12C5A60S2.h"
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
void Timer0Init();//声明定时器0初始化函数
#endif
相关文章:
基于STC12C5A60S2系列1T 8051单片机通过单个按键单击次数实现开关机应用
基于STC12C5A60S2系列1T 8051单片机通过单个按键单击次数实现开关机应用 STC12C5A60S2系列1T 8051单片机管脚图STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍基于STC12C5A60S2系列1T 8051单片机通过单个按…...
静态住宅IP优缺点,究竟要怎么选?
在进行海外 IP 代理时,了解动态住宅 IP 和静态住宅 IP 的区别以及如何选择合适的类型非常重要。本文将介绍精态住宅 IP 特点和,并提供选择建议,帮助您根据需求做出明智的决策。 静态住宅 IP 的特点 静态住宅 IP 是指 IP 地址在一段时间内保…...
day07-缓存商品、购物车
1. 缓存菜品 1.1 问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大。 结果: 系统响应慢、用户体验差 1.2 实现思路 通过Redis来缓存菜品数据,减少数据库查询操作。 缓…...
平台介绍-搭建赛事运营平台(3)
上文介绍了品牌隔离的基本原理,就是通过不同的前端和微服务来实现。但是确实很多功能是类似的,所以从编程角度还是有些管理手段的。 前端部分:前端部分没有什么特别手段,就是两个独立的项目工程,分别维护。相同的部分复…...
数值分析复习:逼近理论的应用——最小二乘问题、解超定、欠定方程组
文章目录 逼近理论的应用——最小二乘问题、解超定、欠定方程组离散平方逼近最小二乘解 本篇文章适合个人复习翻阅,不建议新手入门使用 本专栏:数值分析复习 的前置知识主要有:数学分析、高等代数、泛函分析 逼近理论的应用——最小二乘问题、…...
设计模式-设配器模式
目录 🎊1.适配器模式介绍 🎃2.适配器类型 🎏3.接口适配器 🎐4.类的适配器 🎎5.优缺点 1.适配器模式介绍 适配器模式(Adapter Pattern)是作为两个不兼容的接口之间的桥梁。这种类型的设…...
BEVFormer v2论文阅读
摘要 本文工作 提出了一种具有透视监督(perspective supervision)的新型鸟瞰(BEV)检测器,该检测器收敛速度更快,更适合现代图像骨干。现有的最先进的BEV检测器通常与VovNet等特定深度预训练的主干相连,阻碍了蓬勃发展…...
FFMPEG C++封装(二)
4 详细设计 这章是FFMPEG C封装库的详细设计。 4.1 Init 该模块初始化FFMPEG库。 4.1.1 Init定义 namespace media { namespace sdk { void MEDIASDK_EXPORT Init(); } }函数说明: Init 初始化FFMPEG库,该函数可调用多次。 4.1.2 Init实现 name…...
使用unplugin-auto-import页面不引入api飘红
解决方案:. tsconfig.json文件夹加上 {"compilerOptions": {"target": "ES2020","useDefineForClassFields": true,"module": "ESNext","lib": ["ES2020", "DOM", &q…...
八大技术趋势案例(虚拟现实增强现实)
科技巨变,未来已来,八大技术趋势引领数字化时代。信息技术的迅猛发展,深刻改变了我们的生活、工作和生产方式。人工智能、物联网、云计算、大数据、虚拟现实、增强现实、区块链、量子计算等新兴技术在各行各业得到广泛应用,为各个领域带来了新的活力和变革。 为了更好地了解…...
Vue实现SQL语句关键字高亮显示?
SQL关键字高亮 要在Vue中实现SQL语句中关键字的高亮显示,你可以使用类似的方法,但是你需要根据SQL语法的特点来解析并高亮显示关键字。以下是一个示例代码,演示了如何在Vue中实现SQL语句关键字的高亮显示。 <template><div><…...
开始时间大于结束时间
1.dom中代码,监听所选日期值的变化,并把需要比较的时间字段作为参数传到监听方法中, <el-form-item label"起始日期" prop"startTime"><el-date-picker clearable size"small":disabled"isDisa…...
Java中 List 集合,通过 Stream 流进行排序总结
一、数据准备 public class OrderTest {private String channelCode;private BigDecimal rate;// 省略 getter、setter、toString()、constructor }List<OrderTest> orderTestList new ArrayList<>();OrderTest z09 new OrderTest("Z09", new BigDeci…...
1688中国站按关键字搜索工厂数据 API
公共参数 名称类型必须描述keyString是申请免费调用key(必须以GET方式拼接在URL中)secretString是调用密钥api_nameString是API接口名称(包括在请求地址中)[item_search,item_get,item_search_shop等]cacheString否[yes,no]默认y…...
YOLOV8逐步分解(2)_DetectionTrainer类初始化过程
接上篇文章yolov8逐步分解(1)--默认参数&超参配置文件加载继续讲解。 1. 默认配置文件加载完成后,创建对象trainer时,需要从默认配置中获取类DetectionTrainer初始化所需的参数args,如下所示 def train(cfgDEFAULT_CFG, use_pythonFalse…...
Java是用什么语言写的?PHP呢?
Java底层是C语言。 Sun公司研发人员根据嵌入式软件的要求,对C进行了改造,去除了留在C的一些不太实用及影响安全的成分,并结合嵌入式系统的实时性要求,开发了一种称为Oak的面向对象语言。而后,经过迭代更新,…...
SpringBoot Redis的使用
官方文档: 官方文档:Spring Data Redis :: Spring Data Redis 和jedis一样,SpringBoot Redis 也可以让我在Java代码中使用redis,同样也是通过引入maven依赖的形式。 加速访问github: 使用steam可以免费加速访问github Spring…...
数据仓库——维度表特性
企业信息化工厂 数据集市中的一致性,由于企业信息化工厂的数据集市是从集成仓库中获得信息的,因此至少从维度建模的角度来看,一致性维护的问题减少了。尽管合并不同数据源的问题依然在,但是负担主要在设计者身上。尽管压力降低了…...
从电荷角度理解开关电容中的电荷守恒
目录 一些铺垫电容的电荷量的解释电荷流入流出对节点电压的影响 从电荷角度理解开关电容加法器中的电荷守恒以开关电容积分器为例说明什么样的节点是电荷守恒 一些铺垫 电容的电荷量的解释 对于一个1F的电容,当它的压差为1V时,它所携带的电荷量是QCU1库…...
1.7.1 python 作业 15道
1、求出1 / 1 1 / 3 1 / 5……1 / 99的和 (1分之一1分之三1分支5....) sum0 for i in range(1,100,2): sum 1/i sum; print(sum) 2、用循环语句,计算2 - 10之间整数的循环相乘的值 (2*3*4*5....10) sum 1 for i in range(2,11): sum sum *…...
synchronized 和 ReentrantLock 的区别是什么
该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:synchronized 和 ReentrantLock 的区别是什么 1. 获取锁的方式 synchronized:synchronized的锁获取是隐式的。当线程进入synchronized修饰的代码…...
大话设计模式之迪米特法则
迪米特法则,也称为最少知识原则(Law of Demeter),是面向对象设计中的一个重要原则,其核心思想是降低耦合度、减少对象之间的依赖关系,从而使系统更加灵活、易于维护和扩展。 根据迪米特法则,一…...
KSD测试系统使用方法和注意事项
①下载链接在最顶部; ②安装方法:应该先将测试设备绑定在假人身上,测试设备不能过度往下拉,传感器绑在脖子上,切记最后才开传感器开关!!!开传感器后3秒内不要碰测试设备衣服&#x…...
IT服务营销管理案例分析题
习题一 企业随着业务的蓬勃发展,所投入的基础设施资源不断增加。企业员工数倍数增长,办公场地、办公环境等要求也越来越高。 可是该企业的IT部门人员短缺,对IT管理还处于被动的“救火”阶段,每天至少15个突发故障,故障…...
NRF52832修改OTA升级时的bootloader蓝牙MAC
NRF52832在OTA升级时,修改了APP的蓝牙MAC会导致无法升级,原因是OTA程序的蓝牙MAC没有被修改所以手机扫描蓝牙时无法连接 解决办法 在bootloader的程序里面加入修改蓝牙mac地址的代码实现原理: 在bootloader蓝牙广播开启之前修改蓝牙mac 通…...
# Python 编程入门教程
欢迎来到 Python 编程入门教程!Python 是一种简单易学、功能强大的编程语言,适用于各种应用场景,从简单的脚本到大型软件开发项目。无论你是初学者还是有一定编程经验的开发者,本教程都将为你提供全面的学习路径,帮助你掌握 Python 编程的基础知识和技能。 ## 目录 1. 简…...
Sqoop【实践 02】Sqoop1最新版 全库导入 + 数据过滤 + 字段类型支持 说明及举例代码(query参数及字段类型强制转换)
Sqoop1最新版举例 1.环境说明2.import-all-tables3.query4.字段类型支持 1.环境说明 还是之前的环境: # 不必要信息不再贴出 # JDK [roottcloud ~]# java -version java version "1.8.0_251" # MySQL [roottcloud ~]# mysql -V mysql Ver 14.14 Distrib…...
第十四届蓝桥杯JavaA组省赛真题 - 特殊日期
解题思路: 暴力秒了 public class Main {public static void main(String[] args) {int cnt 0;for (int i 1900; i < 9999; i) {for (int j 1; j < 12; j) {for (int k 1; k < days(i, j); k) {if (sum(i) sum(j) sum(k)) cnt;}}}System.out.print…...
《VulnHub》Lampião:1
title: 《VulnHub》Lampio:1 date: 2024-03-28 21:37:49 updated: 2024-03-28 21:37:50 categories: WriteUp:Cyber-Range excerpt: 关键技术:主机发现,端口扫描、服务探测、操作系统探测,对开放的端口探测漏洞&#x…...
RabbitMq高可用
消息队列高级 服务异步通信-高级篇1.消息可靠性1.1.生产者消息确认1.2.消息持久化1.3.消费者消息确认1.4.消费失败重试机制1.5.总结 2.死信交换机2.1.初识死信交换机2.2.TTL2.3.延迟队列 3.惰性队列3.1.消息堆积问题3.2.惰性队列 4.MQ集群4.1.集群分类4.2.普通集群4.3.镜像集群…...
总部在北京的互联网企业/优化网站排名
很多时候我们都会在Python编程中用print 输出来调试代码,但是我今天想要告诉你的是这种方法过时了,现在大家都在用PySnooper,你可知道? 4 月 23 日,GitHub 每日趋势榜第一位是一个 Python 相关项目:PySnooper。 该项目…...
北京手机网站制作/做百度推广的公司电话号码
问题描述 在使用CupertinoPageScaffold,在child添加ListView时会有20的top-padding(只有在设置navigationBar的backgroundColor时会出现)。 原因深究 ListView底层调用了SliverPadding,而SliverPadding的默认有20的padding /// By…...
wordpress 载入特别慢/竞价推广出价多少合适
英特尔第四代CPU架构(Haswell):Haswell的最高端核芯显卡GT3系列在移动版Core i7使用,而中端的GT2则分配给桌面版的Core i系列处理器,而最低端的奔腾、赛扬搭载GT1。此外,Haswell将会使用LGA1150插座,无法和LGA1155替换…...
网站怎么做留言/b2b电子商务网
◆ ◆ ◆认识关键帧动画帧 Frame指的是单幅影像画面,一帧相当于电影胶片上的一格。任何动画要表现运动或变化,前后至少要给出两个不同的关键状态,称之为关键帧 Keyframe。而中间状态的变化和衔接,则是由计算机通过特定的插值 In…...
牡丹江网络推广/windows优化大师免费
定义在函数内部的变量拥有一个局部作用域,定义在函数外的拥有全局作用域。 局部变量只能在其被声明的函数内部访问,而全局变量可以在整个程序范围内访问。调用函数时,所有在函数内声明的变量名称都将被加入到作用域中。如下实例: …...
荆门做网站公司/淘宝seo搜索优化工具
上一讲 数据结构之线性结构 主要讲数组与链表。这期介绍数据结构中线性结构代表栈与队列,两者通过数组与链表构造出来。栈实际应用递归,计算机函数执行调用,数学问题如: 8皇后问题 , 汉诺塔, 阶乘问题, 迷宫问题等。队列实际应用消息中间件(如…...