51单片机输出频率可调脉宽可调的PWM


前言

前段时间有个师弟问我关于51单片机输出PWM的问题,我看了网上的资料,各有千秋,但有一些的思路对新手不是很友好;还有一些输出的脉宽和频率不是可调的。因此写下这篇关于51单片机如何输出频率可调脉宽可调的PWM的文章


一、PWM是什么?

关于PWM网上的介绍已经很多了,我这里就简单的介绍一下。
这里插入图片描述
这个是方波,Tp是脉冲宽度,简称脉宽,PWM就是这个脉宽可以调节大小的方波,所以叫脉宽调制(Pulse Width Modulation)
PWM频率就是在一秒内一个完整的PWM发出的次数,比如PWM的频率为1HZ,即一秒发送1次,那T=1000ms;若PWM的频率为50HZ,即一秒发送50次,那T=20ms。

二、输出PWM

如何输出PWM

其实要输出一个PWM很简单,我们可以假设PWM输出引脚为P2^0。

while(1)
{
P2^0 = 1;
delay(100);//这里相当于Tp
P2^0 = 0;
delay(200);
}

让引脚输出高电平,延迟一段时间后再输出低电平,通过改变延迟的时间就可以调节PWM的脉宽。
但是这样输出的频率是不可调的,也是不准确的;而且一旦有其它程序加入,波形就会发生变化。

使用定时器控制输出频率

PWM输出频率我们可以借助定时器来实现,我们可以让定时器固定一段时间输出高电平,一段时间输出低电平。输出高电平和低电平时间之和即为PWM的周期(周期为频率的倒数),我们只要通过改变输出高电平和低电平时间之和就可以改变PWM的频率了。


sbit led = P0 ^ 0; // 定义P20口是led
sbit Pwm = P2 ^ 0;

//定时器0初始化,每1ms进入一次中断
void Timer0Init()
{
    TMOD |= 0X01; // 选择为定时器0模式,工作方式1,仅用TR0打开启动。
    TH0 = 0XFC; // 给定时器赋初值,定时1ms
    TL0 = 0X18;
    ET0 = 1; // 打开定时器0中断允许
    EA = 1;  // 打开总中断
    TR0 = 1; // 打开定时器
}

void main()
{
    Timer0Init(); // 定时器0初始化
    Pwm = 1;// 先让PWM引脚输出高电平
    while (1)
    {
    }
}

//定时器0的中断服务函数,即每1ms就执行这段程序
void Timer0() interrupt 1
{
    static u16 i;
    TH0 = 0XFC; // 给定时器赋重新初值,定时1ms
    TL0 = 0X18;
    i++;
    if (i >= 20)
    {
    	i = 0;
        Pwm = ~PWM;
    }
    led = ~led;
}

我们让PMW输出引脚每20ms电平反转一次,led输出引脚每1ms电平反转一次进行比较。
PWM仿真图
这是PWM输出的仿真图,蓝色是led引脚输出波形,周期是2ms,红色是PWM引脚输出波形,周期是40ms。(因为每隔20ms反转一次电平,高电平和低电平持续时间之和才是PWM的一个周期的时间)

这样,我们就可以通过改变电平反转的时间来控制PWM的输出频率了。

使用定时器控制输出占空比

上面我们已经可以控制PWM的频率了,那我们如何控制高电平在整个周期中所占的比例(即占空比)呢?

我们可以在一个固定的时间内,改变高电平所占的比例,即可以改变占空比。

sbit led = P0 ^ 0; // 定义P20口是led
sbit Pwm = P2 ^ 0;

u8 Pwm_Duty = 0;
u16 Pwm_fre = 50;

/*******************************************************************************
 * 函 数 名         : Timer0Init
 * 函数功能		   : 定时器0初始化
 * 输    入         : 无
 * 输    出         : 无
 *******************************************************************************/
void Timer0Init()
{
    TMOD |= 0X01; // 选择为定时器0模式,工作方式1,仅用TR0打开启动。
    TH0 = 0XFC; // 给定时器赋初值,定时1ms
    TL0 = 0X18;
    ET0 = 1; // 打开定时器0中断允许
    EA = 1;  // 打开总中断
    TR0 = 1; // 打开定时器
}

void main()
{
    Timer0Init(); // 定时器0初始化
    Pwm = 1;
    led = 1;
    Pwm_Duty = 40;
    while (1)
    {

    }
}
/*******************************************************************************
 * 函 数 名         : void Timer0() interrupt 1
 * 函数功能		   : 定时器0中断函数
 * 输    入         : 无
 * 输    出         : 无
 *******************************************************************************/
void Timer0() interrupt 1
{
    static u16 i;
    TH0 = 0XFC; // 给定时器赋初值,定时1ms
    TL0 = 0X18;
    i++;
    if (i >= (Pwm_Duty*20)/100)
    {
        Pwm = 0;
    }
    if(i>=10)
    {
        led = 0;
    }
    if (i >= 20)
    {
        i = 0;
        Pwm = 1;
        led = 1;
    }
}

Pwm_Duty 为高电平比例系数,比如Pwm_Duty =0.4,当i>=0.4*20=8时,PWM反转为低电平,当i>20时,电平再反转为高电平,这样高电平就占了整个周期的四成,所以PWM=40%。
在这里插入图片描述

不过应该注意的是,这里PWM在8ms已经反转一次,20ms时再反转一次,一个周期已经结束,即这里的PWM周期为20ms;即频率为50HZ。由于51单片机是8位整形单片机,没有浮点型数据,不能直接让Pwm_Duty =0.4,不然就可能取整为0,这里必须先让Pwm_Duty = 40*20=800再除以100才能得到8ms
在这里插入图片描述
这是PWM输出的仿真图,蓝色是led引脚输出波形,频率是50HZ,占空比是50%;红色是PWM引脚输出波形,频率是50HZ,占空比是40%。

这样,我们就可以控制Pwm_Duty来控制PWM的占空比了。

我们再把程序给优化一下,如下


#include "reg52.h" //此文件中定义了单片机的一些特殊功能寄存器

typedef unsigned int u16; // 对数据类型进行声明定义
typedef unsigned char u8;

sbit led = P0 ^ 0; // 定义P20口是led
sbit Pwm = P2 ^ 0;
sbit k1 = P1 ^ 1; // 定义P20口是led
sbit k2 = P1 ^ 2;

u8 Pwm_Duty = 40;//pwm的占空比(%)
u16 Pwm_fre = 50;//PWM的频率(HZ)

/*******************************************************************************
 * 函 数 名         : Timer0Init
 * 函数功能		   : 定时器0初始化
 * 输    入         : 无
 * 输    出         : 无
 *******************************************************************************/
void Timer0Init()
{
    TMOD |= 0X01; // 选择为定时器0模式,工作方式1,仅用TR0打开启动。
    TH0 = 0XFC;   // 给定时器赋初值,定时1ms
    TL0 = 0X18;
    ET0 = 1; // 打开定时器0中断允许
    EA = 1;  // 打开总中断
    TR0 = 1; // 打开定时器
}

/*******************************************************************************
 * 函 数 名       : main
 * 函数功能		 : 主函数
 * 输    入       : 无
 * 输    出    	 : 无
 *******************************************************************************/
void main()
{
    Timer0Init(); // 定时器0初始化
    Pwm = 1;
    led = 1;
    while (1)
    {
    }
}

/*******************************************************************************
 * 函 数 名         : void Timer0() interrupt 1
 * 函数功能		   : 定时器0中断函数
 * 输    入         : 无
 * 输    出         : 无
 *******************************************************************************/
void Timer0() interrupt 1
{
    static u16 i;
    TH0 = 0XFC; // 给定时器赋初值,定时1ms
    TL0 = 0X18;
    i++;
    if (i >= (Pwm_Duty * (1000 / Pwm_fre)) / 100)
    {
        Pwm = 0;
    }
    if (i >= (50 * (1000 / Pwm_fre)) / 100)
    {
        led = 0;
    }
    if (i >= 1000 / Pwm_fre)
    {
        i = 0;
        led = 1;
        Pwm = 1;
    }
}

把频率和占空比作为变量,即可以在程序运行中,通过改变Pwm_fre 改变PWM信号的频率,通过改变Pwm_Duty 改变PWM信号的占空比。

补充 : (Pwm_Duty * (1000 / Pwm_fre)) / 100这段可能有人看不懂,我解释一下:Pwm_fre是PWM的频率,频率和周期互为倒数,所以PWM周期 = 1/ Pwm_fre。由于程序是每1ms进入一次中断,频率是一秒内一个完整信号发送的次数,所以这里要乘1000,又因为51单片机不支持浮点型数据,所以需要写成(1000 / Pwm_fre);若信号频率为50hz,则这里周期为20ms。

  • 29
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值