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

news2025/1/16 6:50:52

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

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

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

相关文章

内容安全复习 2 - 网络信息内容的获取与表示

文章目录 信息内容的获取网络信息内容的类型网络媒体信息获取方法 信息内容的表示视觉信息视觉特征表达文本特征表达音频特征表达 信息内容的获取 网络信息内容的类型 网络媒体信息 传统意义上的互联网网站公开发布信息&#xff0c;网络用户通常可以基于网络浏览器获得。网络…

【Python机器学习实战】 | 基于线性回归以及支持向量机对汽车MPG与自重进行回归预测

&#x1f3a9; 欢迎来到技术探索的奇幻世界&#x1f468;‍&#x1f4bb; &#x1f4dc; 个人主页&#xff1a;一伦明悦-CSDN博客 ✍&#x1f3fb; 作者简介&#xff1a; C软件开发、Python机器学习爱好者 &#x1f5e3;️ 互动与支持&#xff1a;&#x1f4ac;评论 &…

excel如何实现按班级统计?

这个表有1-20个班(上表班级排名可以忽略不计)&#xff0c;需要计算每个班级的总分排名的各段人数&#xff0c;分段要求是0-60名&#xff0c;61-200名&#xff0c;201-600名。最后结果如下。 添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09; 如果年级有600…

【React】Axios请求头注入token

业务背景: Token作为用户的数据标识&#xff0c;在接口层面起到了接口权限控制的作用&#xff0c;也就是说后端有很多接口都需要通过查看当前请求头信息中是否含有token数据&#xff0c;来决定是否正常返回数据 // 添加请求拦截器 request.interceptors.request.use(config …

.NET C# 使用GDAL读取FileGDB要素类

.NET C# 使用GDAL读取FileGDB要素类 目录 .NET C# 使用GDAL读取FileGDB要素类1 环境2 Nuget3 Code 1 环境 VisualStudio2022 .NET6 GDAL 3.7.5 2 Nuget 3 Code using OSGeo.OGR; using OSGeo.OSR;namespace TestGDAL {internal class Program{static void Main(string[] a…

STM32单片机-PWR电源控制和WDG看门狗

STM32单片机-PWR电源控制和WDG看门狗 一、PWR简介二、低功耗模式三、修改主频&睡眠模式&停机模式&待机模式3.1 修改主频3.2 睡眠模式3.3 停机模式3.4 待机模式 四、WDG简介4.1 独立看门狗原理4.2 窗口看门狗原理4.3 IWDG和WWDG对比 五、独立看门狗&窗口看门狗5…

如何确保远程桌面安全

在数字化快速发展的今天&#xff0c;远程桌面技术广泛应用于企业办公、技术支持以及个人使用等领域。然而&#xff0c;随之而来的安全问题也不容忽视。白名单技术作为一种重要的安全防护手段&#xff0c;在确保远程桌面安全方面发挥着至关重要的作用。 一、白名单技术概述 白名…

ArcGIS批量设置多图层的三调地类符号

​​ 点击下方全系列课程学习 点击学习—>ArcGIS全系列实战视频教程——9个单一课程组合系列直播回放 01需求说明 这次我们要实现的是将多个地类图层批量符号化。比如将多个三调地类图斑批量符号化。 ​ 有什么好方法呢 &#xff1f; 我们可以将一个图层利用三调符号库进行…

一作分享|eDNA揭示水库梯级影响下乌江流域鱼类多样性的空间分布

在梯级开发背景下&#xff0c;乌江干流鱼类群落结构已发生显著改变&#xff0c;凌恩客户重庆师范大学生命科学学院通过eDNA宏条码技术对对乌江干流全面系统的鱼类资源现状进行了调查。本期邀请一作程如丽老师对文章进行了解读分享。 文章信息 标题&#xff1a;eDNA reveals spa…

录视频软件有哪些,5种软件分享(2024最新)

你是否也在电脑上游览着各种生动且有趣的视频&#xff1f;看到他人分享的视频时&#xff0c;是否也会思考自己如何才能录制出精美的视频&#xff1f; 随着数字化时代的到来&#xff0c;视频内容已经深入到我们生活的方方面面&#xff0c;无论是娱乐、学习还是工作&#xff0c;…

【CS.DS】数据结构 —— 图: 图的相关概念大全

文章目录 1 图的类型2 图的基本术语References 1 图的类型 图是一种数据结构&#xff0c;由节点&#xff08;顶点&#xff09;和边组成。图可以用来表示各种网络结构&#xff0c;如社交网络、交通网络、计算机网络等。根据边的性质&#xff0c;图可以分为以下几种类型&#xf…

Nvidia Isaac Sim搭建仿真环境 入门教程 2024(4)

Nvidia Isaac Sim 入门教程 2024 版权信息 Copyright 2023-2024 Herman YeAuromix. All rights reserved.This course and all of its associated content, including but not limited to text, images, videos, and any other materials, are protected by copyright law. …

【b站-湖科大教书匠】1 计算机网络概述-计算机网络微课堂

课程地址&#xff1a;【计算机网络微课堂&#xff08;有字幕无背景音乐版&#xff09;】 https://www.bilibili.com/video/BV1c4411d7jb/?share_sourcecopy_web&vd_sourceb1cb921b73fe3808550eaf2224d1c155 目录 1 概述 1.1 计算机网络在信息时代的作用 1.2 因特网概述…

PPT可以转换成Word吗?归纳了三种转换方式

PPT可以转换成Word吗&#xff1f;在当今快节奏的工作和学习环境中&#xff0c;不同格式文件之间的转换变得日益重要。PPT作为演示文稿制作的首选工具&#xff0c;广泛应用于会议演讲、教育培训等多个场景&#xff0c;而Word则是文档编辑与编排的基石。为了便于进一步编辑、分享…

ssh远程连接vps

打开ssh服务 sudo vi /etc/ssh/sshd_config查看是否好了 systemctl status sshd生成ssh私钥 生成在C:\Users\baozhongqi\.ssh 然后可以用自己密码登录或者用私钥登录 密码登录 ssh私钥登录 我使用的是tabby tabby下载 setup是Windows使用的版本 如果这样子不能ssh链接只能…

Redis通用命令详解

文章目录 一、Redis概述1.1 KEYS&#xff1a;查看符合模板的所有 key1.2 DEL&#xff1a;删除一个指定的 key1.3 EXISTS&#xff1a;判断 key 是否存在1.4 EXPIRE&#xff1a;给一个 key 设置有效期&#xff0c;有效期到期时该 key 会被自动删除1.5 TTL&#xff1a;查看一个 ke…

基于FPGA的Cordic向量模式原理及设计

目录 一、向量模式 1、向量模式原理 1.1 1.2 2、向量模式的MATLAB仿真 3、向量模式的FPGA实现 3.1 预处理 3.2 迭代 3.3 结果计算 一、向量模式 1、向量模式原理 已知直角坐标下一点&#xff08;x&#xff0c;y&#xff09;&#xff0c;如何求该点在极坐标系的坐标&a…

【2024最新华为OD-C/D卷试题汇总】[支持在线评测] API集群访问频次统计(100分) - 三语言AC题解(Python/Java/Cpp)

&#x1f36d; 大家好这里是清隆学长 &#xff0c;一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 &#x1f4bb; ACM银牌&#x1f948;| 多次AK大厂笔试 &#xff5c; 编程一对一辅导 &#x1f44f; 感谢大家的订阅➕ 和 喜欢&#x1f497; &#x1f…

多路h265监控录放开发-(4)完成摄像机管理的数据的增删改模型层代码(单例模式)

xcamera_config.h #pragma once #include <vector> #include <mutex> struct XCameraData {char name[1024] { 0 };char url[4096] { 0 }; //摄像机主码流char sub_url[4096] { 0 }; //摄像机辅码流char save_path[4096] { 0 }; //视频录制存放目…

RX8025/INS5T8025实时时钟-国产兼容RS4TC8025

该模块是一个符合I2C总线接口的实时时钟&#xff0c;包括一个32.768 kHz的DTCXO。 除了提供日历&#xff08;年、月、日、日、时、分、秒&#xff09;功能和时钟计数器功能外&#xff0c;该模块还提供了大量其他功能&#xff0c;包括报警功能、唤醒定时器功能、时间更新中断功能…