初学51单片机之PWM实例呼吸灯以及遇到的问题(已解答)

PWM全名Pulse Width Modulation中文称呼脉冲宽度调制 如图

这是一个周期10ms、频率是100HZ的波形,但是每个周期内,高低电平宽度各不相同,这就是PWM的本质。

占空比是指高电平占整个周期的比列,上图第一个波形的占空比是40%,第二个是60%,第三个是80%。

本案将以PWM控制来制作一个呼吸灯,以及一个大致模拟人体呼吸的呼吸灯。

上代码

# include<reg52.h>

sbit PWMOUT = P0^0;
sbit ADDR0 = P1^0;
sbit ADDR1 = P1^1;
sbit ADDR2 = P1^2;
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;

unsigned long PeriodCnt = 0; //PWM周期计数值
unsigned char HighRH = 0;    //高电平重载值的高字节
unsigned char HighRL = 0;   //高电平重载值的低字节
unsigned char LowRH = 0;    //低电平重载值的高字节
unsigned char LowRL = 0;    //低电平重载值的低字节
unsigned char T1RH = 0;     //T1重载值的高字节
unsigned char T1RL = 0;     //T1重载值的低字节

void ConfigTimer1(unsigned int ms);
void ConfigPWM(unsigned int fr,unsigned char dc);

void main()
{
  EA = 1;           //开启中断
	ENLED = 0;        //使能U3
	ADDR3 = 1;
	ADDR2 = 1;       //使能LED
	ADDR1 = 1;
	ADDR0 = 0;
	
	ConfigPWM(100,10);  //配置并启动PWM
	ConfigTimer1(50);  //用T1定时调整PWM
	while(1);
}

/* 配置并启动T1,ms为定时时间  */
void ConfigTimer1(unsigned int ms)
{
  unsigned long tmp;               //定义临时变量
	
	tmp = 11059200/12;             //定时器计数频率
	tmp = (tmp * ms)/1000;         //计算所需的计数值
	tmp = 65536 - tmp ;           //计数定时器重载值
	tmp = tmp +12 ;                    //补偿中断响应延时造成的误差
	T1RH = (unsigned char)(tmp >> 8);  //定时器重载值拆分高低字节
	T1RL = (unsigned char)tmp;
	TMOD &= 0x0F;                   //0000 1111 清零T1的控制位
	TMOD |= 0x10;                 //0001 0000 配置T1的模式为1
	TH1 = T1RH;                   //加载T1的重载值
	TL1 = T1RL;                  
	ET1 = 1;                      //使能T1中断
	TR1 = 1;                      //启动T1

}

/*配置并启动PWM,fr为频率,dc为占空比 */
void ConfigPWM(unsigned int fr , unsigned char dc)
{
  unsigned int high, low;
	
	PeriodCnt = (11059200/12) /fr;    //计算一个周期所需的计数值
	high = (PeriodCnt * dc) /100;     //计算高电平所学的计数值
	low = PeriodCnt - high;           //计算低电平所需的计数值
	high = 65536 - high +12;          //计算高电平的定时器重载值并补偿中断延时
	low = 65536 -low +12;             //计算低电平的定时器重载值并补偿中断延时
	HighRH = (unsigned char)(high >> 8); //高电平重载值拆分高低电平
	HighRL = (unsigned char)high;
	LowRH = (unsigned char)(low >> 8);  //低电平重载值拆分高低电平
	LowRL = (unsigned char)low;
	TMOD &= 0xF0;                   //清零T0的控制位
	TMOD |= 0x01;                   //配置T0为模式1
	TH0 = HighRH;                   //加载T0的重载值
	TL0 = HighRL;
	ET0 = 1;                        //使能T0中断
	TR0 = 1;                        //启动T0
	PWMOUT = 0;                     //输出高电平
	
}

/* 占空比调整函数,频率不变只调整占空比 */

void AdjustDutyCycle(unsigned char dc)
{
  unsigned int high,low;
	
	high = (PeriodCnt * dc) / 100;  //计算高电平所需的计数值
	low  = PeriodCnt - high;        //计算低电平所需的计数值
	high = 65536 - high +12;        //计算高电平的定时器重载值并补偿中断延时
	low = 65536 - low +12;          //计算低电平的定时器重载值并补偿中断延时
	HighRH = (unsigned char)(high >> 8); //高电平重载值拆分为高低字节
	HighRL = (unsigned char)high;
	LowRH = (unsigned char)(low >> 8); //低电平重载值拆分为高低字节
	LowRL = (unsigned char)low;
	

}

/* T0中断服务函数,产生PWM输出 */
void interruptTimer0() interrupt 1
{
  if(PWMOUT == 1)                  //当输出位高电平时,装载低电平值并输出低电平
	{
	  TH0 = LowRH;
		TL0 = LowRL;
		PWMOUT = 0;
	}
	else                       //当输出为低电平时,装载高电平值并输出高电平
	{
	  TH0 = HighRH;
		TL0 = HighRL;
		PWMOUT = 1;
	}

}

/* T1中断服务函数,定时动态调整占空比 */

void interruptTimer1() interrupt 3
{
  static bit dir = 0;
	static unsigned char index = 0;
	unsigned char code table[13] = {
	  5,18,30,41,51,60,68,75,81,86,90,93,95      //占空比调整
	};
	TH1 = T1RH;
	TL1 = T1RL;
	AdjustDutyCycle(table[index]);            //调整PWM的占空比
	if(dir == 0)                               //逐步增大占空比
	{
	  index++;
		if(index >= 12)   
		 {
		   dir = 1;
		 }
	 
	}
	else                   //减少占空比
	{
	  index--;
		if(index == 0)
		 {
		   dir = 0;
		 }
	}
}

看结果视频:这是一个频率比较快的呼吸灯,提供下逻辑导图。

呼吸灯_哔哩哔哩_bilibili

笔者用它的占空比数组做了一个曲线图,

竖轴是占空比,横轴是时间。

上篇博文数字秒表,笔者已经计算过类似的时间补偿。本案也演算一次

首先这个时间循环可以看着是这样如下图,无论占空比如何改变频率都是100HZ即10ms

上次博文笔者求解了数字秒表的误差,这次在求解呼吸灯的时候遇到了一些问题,而且这些问题目前不知如何解决;

第一个是函数赋值出错的问题:对于函数 ConfigPWM(unsigned int fr , unsigned char dc)

当运行到

HighRH = (unsigned char)(high >> 8);  变量high 与low的值应该都已经赋值完毕,而且赋值完的结果应该是是high =FC73 low=DFA5。但是笔者debug的结果是

可以看到high的值没有问题,高低电平拆分,从新赋值也没问题。

但是Low出问题了,Low的值竟然是0x20A5,拆分后的赋值又是正确的,那么这里到底是哪里出了问题?为什么High值没有问题,low值却有问题呢 ?它们两的计算过程都一样,这是一处问题。

第二个问题已经解决了:

刚才没考虑到PWMOUT是单片机P0^0端口,不是单纯的变量,因此在debug的时候是无法按照变量的方式是去考虑的,把PWMOUT改成变量就可以进入if函数了。刚吃完饭灵光一线。

第二:在debug时间误差的时候笔者发现,定时器0中断,程序指针一直无法进入if函数,直接进入了else函数,导致中断时间间隔一直都是1ms,看视频                      keil5Debug过程异常_哔哩哔哩_bilibili

可以看到debug过程中无法进入定时器0中断的if函数,但是如果你注释掉if函数了的关键语句PWMOUT = 0;它就不会工作,显然if函数是起作用的,事实上笔者后续按了好久的F5j都已经72了,时间都到85ms了,i还是0。85ms定时器1中断都进入一次了,很快就要进入第二次了,if函数还是没进入一次,这显然不符合程序逻辑 的。事实上主函数第一次执行 ConfigPWM()的时候,把PWMOUT赋值为1,是可以进入一次if函数的,然后会经过9ms的等待再次进入中断进入else()函数,从此后就再也无法进入if函数了,后续现象和视频一样。

这个问题产生的缘由笔者不清楚,如果有读者小伙伴知道原因,请留下言。笔者多多感谢。如果有keil软件的话,可以复制过去,debug一下,看是否和笔者的结论一样。事实上笔者用keil4也试了,结果也一样。

然后笔者想既然已经实现一个呼吸灯功能了,那么用呼吸灯模拟下人正常呼吸的频率。

根据笔者自己的体验吸气要2s,呼气要3s。一个周期是5s。然后是呼吸曲线图。

本案的LED的是低电平使能,因此曲线图最高点占空比95,应是它的反向占空比即5%。这个数组是

   unsigned char code tableup[10] = {            //吸气数组2s
     95,80,67,56,45,36,27,18,10,5
    };

    unsigned char code tabledown[17] = {         //呼气数组3s多一点
      5,10,20,31,44,56,68,78,85,89,92,93,94,95,95,95,95
    };

上代码

# include<reg52.h>

sbit PWMOUT = P0^0;
sbit ADDR0 = P1^0;
sbit ADDR1 = P1^1;
sbit ADDR2 = P1^2;
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;

unsigned long PeriodCnt = 0; //PWM周期计数值
unsigned char HighRH = 0;    //高电平重载值的高字节
unsigned char HighRL = 0;   //高电平重载值的低字节
unsigned char LowRH = 0;    //低电平重载值的高字节
unsigned char LowRL = 0;    //低电平重载值的低字节
unsigned char T1RH = 0;     //T1重载值的高字节
unsigned char T1RL = 0;     //T1重载值的低字节

void ConfigTimer1(unsigned int ms);
void ConfigPWM(unsigned int fr,unsigned char dc);

void main()
{
  EA = 1;           //开启中断
	ENLED = 0;        //使能U3
	ADDR3 = 1;
	ADDR2 = 1;       //使能LED
	ADDR1 = 1;
	ADDR0 = 0;
	
	ConfigPWM(100,95);  //配置并启动PWM
	ConfigTimer1(50);  //用T1定时调整PWM
	while(1);
}

/* 配置并启动T1,ms为定时时间  */
void ConfigTimer1(unsigned int ms)
{
  unsigned long tmp;               //定义临时变量
	
	tmp = 11059200/12;             //定时器计数频率
	tmp = (tmp * ms)/1000;         //计算所需的计数值
	tmp = 65536 - tmp ;           //计数定时器重载值
	tmp = tmp +12 ;                    //补偿中断响应延时造成的误差
	T1RH = (unsigned char)(tmp >> 8);  //定时器重载值拆分高低字节
	T1RL = (unsigned char)tmp;
	TMOD &= 0x0F;                   //0000 1111 清零T1的控制位
	TMOD |= 0x10;                 //0001 0000 配置T1的模式为1
	TH1 = T1RH;                   //加载T1的重载值
	TL1 = T1RL;                  
	ET1 = 1;                      //使能T1中断
	TR1 = 1;                      //启动T1

}

/*配置并启动PWM,fr为频率,dc为占空比 */
void ConfigPWM(unsigned int fr , unsigned char dc)
{
  unsigned int high, low;
	
	PeriodCnt = (11059200/12) /fr;    //计算一个周期所需的计数值
	high = (PeriodCnt * dc) /100;     //计算高电平所学的计数值
	low = PeriodCnt - high;           //计算低电平所需的计数值
	high = 65536 - high +12;          //计算高电平的定时器重载值并补偿中断延时
	low = 65536 -low +12;             //计算低电平的定时器重载值并补偿中断延时
	HighRH = (unsigned char)(high >> 8); //高电平重载值拆分高低电平
	HighRL = (unsigned char)high;
	LowRH = (unsigned char)(low >> 8);  //低电平重载值拆分高低电平
	LowRL = (unsigned char)low;
	TMOD &= 0xF0;                   //清零T0的控制位
	TMOD |= 0x01;                   //配置T0为模式1
	TH0 = HighRH;                   //加载T0的重载值
	TL0 = HighRL;
	ET0 = 1;                        //使能T0中断
	TR0 = 1;                        //启动T0
	PWMOUT = 1;                     //输出高电平
	
}

/* 占空比调整函数,频率不变只调整占空比 */

void AdjustDutyCycle(unsigned char dc)
{
  unsigned int high,low;
	
	high = (PeriodCnt * dc) / 100;  //计算高电平所需的计数值
	low  = PeriodCnt - high;        //计算低电平所需的计数值
	high = 65536 - high +12;        //计算高电平的定时器重载值并补偿中断延时
	low = 65536 - low +12;          //计算低电平的定时器重载值并补偿中断延时
	HighRH = (unsigned char)(high >> 8); //高电平重载值拆分为高低字节
	HighRL = (unsigned char)high;
	LowRH = (unsigned char)(low >> 8); //低电平重载值拆分为高低字节
	LowRL = (unsigned char)low;
	

}

/* T0中断服务函数,产生PWM输出 */
void interruptTimer0() interrupt 1
{
	
	
  if(PWMOUT == 1)                  //当输出位高电平时,装载低电平值并输出低电平
	{
	  TH0 = LowRH;
		TL0 = LowRL;
		PWMOUT = 0;
	}
	else                       //当输出为低电平时,装载高电平值并输出高电平
	{
	  TH0 = HighRH;
		TL0 = HighRL;
		PWMOUT = 1;
	}
 
}

/* T1中断服务函数,定时动态调整占空比 */

void interruptTimer1() interrupt 3
{
  static bit dir = 0;
	static unsigned char index = 0;
	static unsigned char index2 = 0;
	static char cnt = 0;
	//unsigned char code tableup[10] = {
	//  5,20,33,44,55,64,73,82,90,95      //占空比调整
	//};
	unsigned char code tableup[10] = {            //吸气数组
	 95,80,67,56,45,36,27,18,10,5
	};
	//unsigned char code tabledown[15] = {
	 // 95,90,80,69,56,44,32,22,15,11,8,7,6,5,5
	//};
	unsigned char code tabledown[17] = {         //呼气数组
	  5,10,20,31,44,56,68,78,85,89,92,93,94,95,95,95,95
	};
	
	TH1 = T1RH;
	TL1 = T1RL;
	cnt++;
	if(cnt >= 4)
	{
	   cnt = 0;          
	  if(dir == 0)                              
	  {
		 AdjustDutyCycle(tableup[index]);   //吸气2s
	   index++;
		 if(index >= 10)   
		  {
		    dir = 1;
				index = 0;
		  }
	 
	   }
	   else                   
	  {
			AdjustDutyCycle(tabledown[index2]);   //呼气3s
	    index2++;
		if(index2 >= 17)
		 {
		   dir = 0;
			 index2 = 0;
		  }
	  }
	}
	
}

看下结果:模拟人体呼吸灯2_哔哩哔哩_bilibili  

总结:不知道是keil5软件本身有问题,还是笔者的keil5软件有问题,还是笔者哪里没有考虑到,路过的小伙伴如果知道缘由,请多多留言,指点下笔者,笔者在此多多感谢。

再和初学的小伙伴分享一个关于定时器运行的相关问题。定时器计时相关_哔哩哔哩_bilibili

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ROS话题通信流程自定义数据格式

ROS话题通信流程自定义数据格式 需求流程实现步骤定义msg文件编辑配置文件编译 在 ROS 通信协议中&#xff0c;数据载体是一个较为重要组成部分&#xff0c;ROS 中通过 std_msgs 封装了一些原生的数据类型,比如:String、Int32、Int64、Char、Bool、Empty… 但是&#xff0c;这些…

CST电磁仿真软件的参数类型和含义【电磁仿真入门教程】

如果你是一位工程师或设计师&#xff0c;那你对电磁仿真软件CST Studio Suite一定不会感到陌生。CST软件可以帮助你模拟电磁场和电路行为&#xff0c;从而优化产品设计。本文将带你了解CST电磁仿真软件的一些关键参数&#xff0c;并解释其含义。CST电磁仿真软件的参数是指在使用…

2024年文化传播与公共艺术国际会议(CCPA 2024)

2024年文化传播与公共艺术国际会议&#xff08;CCPA 2024&#xff09; 2024 International Conference on Cultural Communication and Public Arts 【重要信息】 大会地点&#xff1a;桂林 大会官网&#xff1a;http://www.icccpa.com 投稿邮箱&#xff1a;icccpasub-conf.co…

神经网络参数-----学习率(Learning Rate)

学习率 学习率是训练神经网络的重要超参数之一&#xff0c;它代表在每一次迭代中梯度向损失函数最优解移动的步长。它的大小决定网络学习速度的快慢。在网络训练过程中&#xff0c;模型通过样本数据给出预测值&#xff0c;计算代价函数并通过反向传播来调整参数。重复上述过程…

linux系统指令查漏补缺

目录 一.磁盘操作 二.lvm 三.top 4.nohup 一.磁盘操作 1. lsblk -f 显示磁盘和它的相关内容 2.tuen2fs -c -1 /dev/sdx 关闭某个磁盘的自检 3.修改配置&#xff0c;使文件系统不要开机自检 cat /etc/fstab 全0表示开机不自检 全1表示开机自检 同时在这个文件中可添加…

欧洲杯赛况@20240623

估计点击标题下「蓝色微信名」可快速关注 老牌劲旅捷克队面对格鲁吉亚&#xff0c;这是两队的首次交锋&#xff0c;格鲁吉亚是很放松的状态&#xff0c;每场比赛对他们都很新鲜&#xff0c;而捷克则谨慎多&#xff0c;至今为止&#xff0c;最倒霉的球员&#xff0c;可能就是捷克…

TensorFlow高阶API使用与PyTorch的安装

欢迎来到 Papicatch的博客 文章目录 &#x1f349;TensorFlow高阶API使用 &#x1f348;示例1&#xff1a;使用tf.keras构建模型 &#x1f34d;通过“序贯式”方法构建模型 &#x1f34d;通过“函数式”方法构建模型 &#x1f348;示例2&#xff1a;编译模型关键代码 &am…

B端列表:筛选器设计的十大要点,都是干货。

一、列表页的筛选器有什么作用 在B端电商平台或者企业内部管理系统中&#xff0c;列表页的筛选器是非常重要的功能之一。它能够帮助用户快速准确地找到所需的信息&#xff0c;提高工作效率&#xff0c;为企业的运营和决策提供有力支持。 首先&#xff0c;列表页的筛选器可以帮…

操作系统实训复习笔记(1)

目录 Linux vi/vim编辑器&#xff08;简单&#xff09; &#xff08;1&#xff09;vi/vim基本用法。 &#xff08;2&#xff09;vi/vim基础操作。 进程基础操作&#xff08;简单&#xff09; &#xff08;1&#xff09;fork()函数。 写文件系统函数&#xff08;中等&…

【BES2500x系列 -- RTX5操作系统】深入探索CMSIS-RTOS RTX -- 任务管理篇 -- 线程管理 --(二)

&#x1f48c; 所属专栏&#xff1a;【BES2500x系列】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &#x1f49…

华为HCIA综合实验(结合前几期所有内容)

第一章 实验目的 &#xff08;1&#xff09;配置Telnet&#xff0c;要求所有网络设备支持远程管理&#xff0c;密码为admin&#xff08;2&#xff09;配置Trunk&#xff0c;交换机之间的链路均为Trunk模式&#xff08;3&#xff09;配置VLAN&#xff0c;在SW2和SW3上创建相关…

QT实现人脸识别

QT实现人脸识别 Face.pro文件&#xff1a; QT core guigreaterThan(QT_MAJOR_VERSION, 4): QT widgetsCONFIG c11# The following define makes your compiler emit warnings if you use # any Qt feature that has been marked deprecated (the exact warnings # d…

力扣141A

文章目录 1. 题目链接2. 题目代码3. 题目总结4. 代码分析 1. 题目链接 Amusing Joke 2. 题目代码 #include<iostream> #include<string> using namespace std;int letterOfInt[30]; int letterAtDoorOfInt[30];int main(){string guestName;string hostName;strin…

Redis-在springboot环境下执行lua脚本

文章目录 1、什么lua2、创建SpringBoot工程3、引入相关依赖4、创建LUA脚本5、创建配置类6、创建启动类7、创建测试类 1、什么lua “Lua”的英文全称是“Lightweight Userdata Abstraction Layer”&#xff0c;意思是“轻量级用户数据抽象层”。 2、创建SpringBoot工程 3、引入相…

OpenCompass:大模型测评工具

大模型相关目录 大模型&#xff0c;包括部署微调prompt/Agent应用开发、知识库增强、数据库增强、知识图谱增强、自然语言处理、多模态等大模型应用开发内容 从0起步&#xff0c;扬帆起航。 大模型应用向开发路径&#xff1a;AI代理工作流大模型应用开发实用开源项目汇总大模…

FW Activity跳转动画源码解析(一)

文章目录 跳转动画实际操作的是什么?窗口怎么知道应该执行什么动画,是透明,还是平移,还是缩放,旋转? 跳转动画实际操作的是什么? startActivity调用之后进行页面跳转,会有一系列的涉及到ActivitStar,ActivityTask,ActivityManager等类的操作,最终在执行动画会调用到Surface…

数字化营销与传统营销的完美协奏曲!

在这个数字化的时代&#xff0c;营销的世界正在发生着巨大的变革&#xff01;数字化营销如火箭般崛起&#xff0c;但传统营销也并未过时。那么&#xff0c;如何让它们携手共进&#xff0c;创造出无与伦比的营销效果呢&#xff1f;今天&#xff0c;就让我们讲述一下蚓链数字化营…

已经被驳回的商标名称还可以申请不!

看到有网友在问&#xff0c;已经驳回的商标名称还可以申请不&#xff0c;普推商标知产老杨觉得要分析看情况&#xff0c;可以适当分析下看可不可以能申请&#xff0c;当然最终还是为了下证 &#xff0c;下证概率低的不建议申请。 先看驳回理由&#xff0c;如果商标驳回是绝对理…

【U8+】修改客户端自动清退时间

【需求描述】 用友U8软件中&#xff0c; 客户端自动清退时间目前最少只能设置为20分钟无操作自动清退&#xff0c; 不能再比20分钟少&#xff0c;例如10分钟无操作自动清退。 【解决方法】 打开注册表&#xff0c;找到下述路径&#xff0c; 【计算机\HKEY_LOCAL_MACHINE\SOFT…

漂亮!身体恢复正常水准!一个家庭幸不幸福,看能量流动的方向——早读(逆天打工人爬取热门微信文章解读)

美洲杯这个时间也太绝了&#xff0c;早上9点比赛&#xff0c;乌拉圭VS巴拿马 引言Python 代码第一篇 洞见 一个家庭幸不幸福&#xff0c;看能量流动的方向第二篇结尾 引言 今天起床 有种神奇的感觉 就是精神很不错 明明昨天晚上还是12点多才睡觉 早上6点20有意识 在头脑里面演…