找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机控制实现PWM输出经L298驱动芯片后驱动电机

[复制链接]
跳转到指定楼层
楼主
使用proteus进行仿真:运用单片机(AT89C52、STM32、MSP430等)控制实现PWM输出经驱动芯片后驱动电机,(驱动自选)
需实现以下功能:
1、pwm频率为1KHz,初始占空比50%
2、PWM输出占空比可调
3、可控制两个电机同时正转、反转、一正一反;
(可实物,提供所需器材)

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)



单片机源程序如下:
  1. #include<reg52.h>  // 引入STC89C52头文件
  2. #define uint unsigned int  // 定义无符号整型变量
  3. #define uchar unsigned char  // 定义无符号字符型变量

  4. uchar time, count = 50, flag1 = 1, flag2 = 1;  // 定义变量

  5. uchar seg[] = {0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39};  // 数码管显示字符

  6. sbit PWM1 = P3^0;  // PWM 输出引脚 1
  7. sbit PWM2 = P3^1;  // PWM 输出引脚 2
  8. sbit PWM3 = P3^2;  // PWM 输出引脚 3
  9. sbit PWM4 = P3^3;  // PWM 输出引脚 4

  10. sbit key_add = P2^3;  // 增加按键
  11. sbit key_dec = P2^4;  // 减少按键
  12. sbit key_turn1 = P2^5;  // 旋转按键 1
  13. sbit key_turn2 = P2^6;  // 旋转按键 2

  14. sbit RS = P3^5;  // LCD 控制引脚 RS
  15. sbit RW = P3^6;  // LCD 控制引脚 RW
  16. sbit E = P3^7;  // LCD 控制引脚 E

  17. // 延时函数
  18. void delayxms(uint z);
  19. // 旋转按键1的处理函数
  20. void Motor_turn1();
  21. // 旋转按键2的处理函数
  22. void Motor_turn2();
  23. // 增加按键的处理函数
  24. void Motor_add();
  25. // 减少按键的处理函数
  26. void Motor_dec();
  27. // 定时器0的初始化函数
  28. void timer0_init();
  29. // 定时器1的初始化函数
  30. void timer1_init();
  31. // 向LCD发送命令
  32. void w_com(uchar j);
  33. // 向LCD发送数据
  34. void w_dat(uchar j);
  35. // 初始化LCD
  36. void lcd_init();

  37. void main()
  38. {
  39.     uint mm, nn;
  40.     lcd_init();
  41.     timer0_init();
  42.     timer1_init();
  43.     while (1)
  44.     {
  45.         Motor_turn1();
  46.         Motor_turn2();
  47.         Motor_add();
  48.         Motor_dec();

  49.         w_com(0x80);
  50.         w_dat('d');
  51.         w_dat('u');
  52.         w_dat('t');
  53.         w_dat('y');
  54.         w_dat(' ');
  55.         w_dat('r');
  56.         w_dat('a');
  57.         w_dat('t');
  58.         w_dat('i');
  59.         w_dat('o');
  60.         w_dat(':');
  61.         w_com(0xc7);

  62.         mm = seg[count/10];
  63.         nn = seg[count%10];
  64.         w_dat(mm);
  65.         w_dat(nn);
  66.         w_dat('%');
  67.     }
  68. }

  69. // 延时函数,延时z毫秒
  70. void delayxms(uint z)
  71. {
  72.     uint x, y;
  73.     for (y = z; y > 0; y--)
  74.         for (x = 110; x > 0; x--);
  75. }

  76. // 处理旋转按键1
  77. void Motor_turn1()
  78. {
  79.     if (key_turn1 == 0)
  80.     {
  81.         delayxms(2);
  82.         if (key_turn1 == 0)
  83.         {
  84.             flag1 = ~flag1;
  85.         }
  86.         while (!key_turn1);
  87.     }
  88. }

  89. // 处理旋转按键2
  90. void Motor_turn2()
  91. {
  92.     if (key_turn2 == 0)
  93.     {
  94.         delayxms(2);
  95.         if (key_turn2 == 0)
  96.         {
  97.             flag2 = ~flag2;
  98.         }
  99.         while (!key_turn2);
  100.     }
  101. }

  102. // 处理增加按键
  103. void Motor_add()
  104. {
  105.     if (key_add == 0)
  106.     {
  107.         delayxms(2);
  108.         if (key_add == 0)
  109.         {
  110.             count += 5;
  111.             if (count >= 100)
  112.             {
  113.                 count = 0;
  114.             }
  115.         }
  116.         while (!key_add);
  117.     }
  118. }

  119. // 处理减少按键
  120. void Motor_dec()
  121. {
  122.     if (key_dec == 0)
  123.     {
  124.         delayxms(2);
  125.         if (key_dec == 0)
  126.         {
  127.             count -= 5;
  128.             if (count >= 100)
  129.             {
  130.                 count = 0;
  131.             }
  132.         }
  133.         while (!key_dec);
  134.     }
  135. }

  136. // 定时器0初始化
  137. void timer0_init()
  138. {
  139.     TMOD = 0x01;
  140.     TH0 = (65536 - 10) / 256;
  141.     TL0 = (65536 - 10) % 256;
  142.     TR0 = 1;
  143.     ET0 = 1;
  144.     EA = 1;
  145. }

  146. // 定时器0中断处理函数
  147. void timer0_int() interrupt 1
  148. {
  149.     TR0 = 0;
  150.     TH0 = (65536 - 10) / 256;
  151.     TL0 = (65536 - 10) % 256;
  152.     TR0 = 1;

  153.     if (flag1 == 1)
  154.     {
  155.         PWM1 = 0;
  156.         time++;
  157.         if (time < count)
  158.             PWM2 = 1;
  159.         else
  160.             PWM2 = 0;

  161.         if (time >= 100)
  162.             time = 0;
  163.     }
  164.     else
  165.     {
  166.         PWM2 = 0;
  167.         time++;
  168.         if (time < count)
  169.             PWM1 = 1;
  170.         else
  171.             PWM1 = 0;

  172.         if (time >= 100)
  173.             time = 0;
  174.     }
  175. }

  176. // 定时器1初始化
  177. void timer1_init()
  178. {
  179.     TMOD = 0x10;
  180.     TH1 = (65536 - 10) / 256;
  181.     TL1 = (65536 - 10) % 256;
  182.     TR1 = 1;
  183.     ET1 = 1;
  184.     EA = 1;
  185. }

  186. // 定时器1中断处理函数
  187. void timer1_int() interrupt 3
  188. {
  189.     TR1 = 0;
  190.     TH1 = (65536 - 10) / 256;
  191.     TL1 = (65536 - 10) % 256;
  192.     TR1 = 1;

  193.     if (flag2 == 1)
  194.     {
  195.         PWM3 = 0;
  196.         time++;
  197.         if (time < count)
  198.             PWM4 = 1;
  199.         else
  200.             PWM4 = 0;

  201.         if (time >= 100)
  202.             time = 0;
  203.     }
  204.     else
  205.     {
  206.         PWM4 = 0;
  207.         time++;
  208.         if (time < count)
  209.             PWM3 = 1;
  210.         else
  211.             PWM3 = 0;

  212.         if (time >= 100)
  213.             time = 0;
  214.     }
  215. }

  216. // 向LCD发送命令
  217. void w_com(uchar j) {
  218.     RS = 0;
  219.     RW = 0;
  220.     E = 1;
  221.     P1 = j;
  222.     E = 0;
  223.     delayxms(3);
  224. }

  225. // 向LCD发送数据
  226. void w_dat(uchar j) {
  227.     RS = 1;
  228.     RW = 0;
  229.     E = 1;
  230.     P1 = j;
  231.     E = 0;
  232.     delayxms(2);
  233. }

  234. // 初始化LCD
  235. void lcd_init() {
  236.     delayxms(10);
  237.     w_com(0x38);
  238.     delayxms(10);
  239.     w_com(0x0c);
  240.     delayxms(10);
  241.     w_com(0x06);
  242.     delayxms(10);
  243.     w_com(0x01);
  244.     delayxms(10);
  245.     w_com(0x38);
  246.     delayxms(10);
  247. }
复制代码

Keil代码与Proteus仿真下载: 仿真程序.7z (65.19 KB, 下载次数: 19)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

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

使用道具 举报

沙发
ID:1086975 发表于 2023-7-9 14:09 | 只看该作者
因为51单片机性能不好,所以在液晶屏上显示就是很不灵敏
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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