找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 7123|回复: 7
打印 上一主题 下一主题
收起左侧

51单片机pwm频率和脉宽的独立调制源程序 产生15HZ~50KHZ的方波

  [复制链接]
跳转到指定楼层
楼主
ID:328361 发表于 2018-5-12 17:34 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
  1. //------------------------------------------------------------------------------------
  2. //程序功能简介:本程序产生15HZ~~~50KHZ的方波,并且实现频率和脉宽的独立调制,即可
  3. //在改变频率的同时不改变脉宽,再改变脉宽的同时不改变频率;同时设置
  4. //两个调节步长------在KEY键按下时,粗调,没有按下时,细调;
  5. //程序思路: 本程序用到两个定时器------定时器0和定时器1,其中定时器0工作在定时方式下,
  6. //决定方波的频率;定时器1,同样工作在定时方式下,用于设定脉宽;
  7. //------------------------------------------------------------------------------------
  8. #include <reg52.h>
  9. #include <math.h>
  10. #define  uchar unsigned char
  11. #define  uint unsigned int
  12. #define  ALL  65536             //定时器工作方式1时,最大基数长度 65536;
  13. #define  F_osc 12000000         //晶振频率12M;

  14. sbit KEY_F_UP=P0^2;             //频率上调按钮;
  15. sbit KEY_F_DOWN=P0^3;           //频率下调按钮;
  16. sbit KEY_W_UP=P0^4;             //脉宽上调按钮;
  17. sbit KEY_W_DOWN=P0^5;           //脉宽下调按钮;
  18. sbit KEY=P0^6;                  //粗细调节按钮-----按下为粗调,否则为细调;
  19. sbit OUTPUT=P1^0;               //波形输出;

  20. uchar TIMER0_H,TIMER0_L,TIMER1_H,TIMER1_L;//定时器0和定时器1的初值设置;
  21. uchar PERCENT=50;               //初始占空比;
  22. uchar FLAG_F=0,FLAG_W=0;//频率调节标志和脉宽调制标志;
  23. uint  FREQ=50000;               //初始频率;
  24. float temp;                     //临时全局变量,用于数据传递;
  25. void delay(uchar t);           //延时函数,用于按键去抖;
  26. void init();                      //初始化函数,用于定时器的初始化;
  27. void calculate_F();     
  28. //频率计算函数,当频率变化,计算出定时器0初值;
  29. void calculate_W();     
  30. //脉宽计算函数,脉宽变化时,计算出定时器1初值;
  31. void key_scan();             //按键扫描函数;
  32. void timer0();                //定时器0中断函数;
  33. void timer1();                //定时器1中断函数;
  34. void delay(uchar t)
  35. {
  36.     uchar i,j;
  37.     while(t--)             //每个脉冲为1us
  38.      {
  39.       for(i=0;i<100;i++)
  40.        for(j=0;j<100;j++);
  41.      }
  42.   }
  43. void calculate_F()
  44. {
  45.     TIMER0_H = (uint)temp/256;
  46.     TIMER0_L = (uint)temp%256;
  47. }
  48. void calculate_W()
  49. {
  50.     float TEMP;
  51.     TEMP = (1 - PERCENT/100.0)*ALL + temp*PERCENT/100.0;
  52.     TIMER1_H = (uint)TEMP/256;
  53.     TIMER1_L = (uint)TEMP%256;
  54.   }
  55.   void key_scan()
  56.   {
  57.     delay(4);
  58.     if(!KEY_F_UP)     //频率上调键按下;
  59.     {
  60.       FLAG_F=1;    //置标志位;
  61.       if(!KEY)
  62.         FREQ+=10;
  63.       else
  64.         FREQ++;
  65.       if(FREQ>50000)
  66.         FREQ=1;
  67.     }
  68.     else if(!KEY_F_DOWN)   //频率下调键按下;
  69.     {
  70.       FLAG_F=1;    //置标志位;
  71.       if(!KEY)
  72.         FREQ-=10;
  73.       else
  74.         FREQ--;
  75.       if(FREQ<1)
  76.         FREQ=50000;
  77.      }
  78.      else if(!KEY_W_UP)    //脉宽上调键按下;
  79.      {
  80.        FLAG_W=1;    //置标志位;
  81.        if(!KEY)
  82.          PERCENT+=5;
  83.        else
  84.          PERCENT++;
  85.        if(PERCENT>49)
  86.          PERCENT=1;
  87.       }
  88.      else if(!KEY_W_DOWN)   //脉宽下调键按下;
  89.      {
  90.        FLAG_W=1;    //置标志位;
  91.        if(!KEY)
  92.          PERCENT-=5;
  93.        else
  94.          PERCENT--;
  95.        if(PERCENT<1)
  96.          PERCENT=49;
  97.       }
  98.       else ;
  99.       }
  100.   void timer0() interrupt 1      //决定频率
  101.    {
  102.      TH0=TIMER0_H;
  103.      TL0=TIMER0_L;
  104.      TR1=1;           //开定时器1;
  105.      OUTPUT=1;
  106.    }
  107.    void timer1() interrupt 3     //决定脉宽
  108.    {
  109.      TH1=TIMER1_H;
  110.      TL1=TIMER1_L;
  111.      TR1=0; //关定时器1,一定要这一步,因为定时器1的定时短于定时器0;
  112.      OUTPUT=0;
  113.    }
  114.    void init()
  115.    {
  116.      TMOD=0x11;      //定时器0和定时器1都工作在方式1,16位计数器;
  117.      calculate_F();  //初始为1KHZ,占空比为50%;
  118.      calculate_W();
  119.      TH0=TIMER0_H;
  120.      TL0=TIMER0_L;
  121.      TH1=TIMER1_H;
  122.      TL1=TIMER1_L;
  123.      ET0=1;
  124.      ET1=1;
  125.      EA=1;
  126.      TR0=1;
  127.      TR1=1;

  128.    }
  129. main()
  130. {
  131.     init();
  132.     while(1)
  133.     {
  134.        key_scan();
  135.        if(FLAG_F)     //改变频率时要注意要进行脉宽的重新设置;
  136.        {
  137.          calculate_F();
  138.          calculate_W();
  139.          FLAG_F=0;
  140.        }
  141.        if(FLAG_W)     // 脉宽改变,频率不改变;
  142.        {
  143.          calculate_W();
  144.          FLAG_W=0;
  145.        }
  146.      }
  147. }

复制代码
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏10 分享淘帖 顶3 踩
回复

使用道具 举报

沙发
ID:372402 发表于 2018-9-5 21:13 | 只看该作者
试过了,频率调不了啊老铁~
回复

使用道具 举报

板凳
ID:586915 发表于 2021-12-4 19:45 | 只看该作者
频率无法调 有会的修改下吗
回复

使用道具 举报

地板
ID:98926 发表于 2021-12-5 11:20 | 只看该作者
temp = ALL - F_osc/12.0/FREQ;//不知出于什么目的,把这句给删掉了
   
回复

使用道具 举报

5#
ID:98926 发表于 2021-12-5 11:26 | 只看该作者
我来助人为乐

temp = ALL - F_osc/12.0/FREQ;//不知出于什么目的,把这句删掉
void calculate_F()
{  temp = ALL - F_osc/12.0/FREQ;
    TIMER0_H = (uint)temp/256;
    TIMER0_L = (uint)temp%256;
}

评分

参与人数 1黑币 +50 收起 理由
admin + 50 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

6#
ID:98926 发表于 2021-12-5 11:28 | 只看该作者
lsx116 发表于 2021-12-4 19:45
频率无法调 有会的修改下吗

temp = ALL - F_osc/12.0/FREQ;
加到频率函数里
回复

使用道具 举报

7#
ID:917430 发表于 2024-4-16 23:17 | 只看该作者
xingzhe8 发表于 2021-12-5 11:26
我来助人为乐

temp = ALL - F_osc/12.0/FREQ;//不知出于什么目的,把这句删掉

谢谢大佬
回复

使用道具 举报

8#
ID:253767 发表于 2024-4-17 07:10 | 只看该作者
xingzhe8 发表于 2021-12-5 11:28
temp = ALL - F_osc/12.0/FREQ;
加到频率函数里

谢谢您,助人为乐!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表