找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 1832|回復(fù): 1
打印 上一主題 下一主題
收起左側(cè)

單片機(jī)控制實(shí)現(xiàn)PWM輸出經(jīng)L298驅(qū)動(dòng)芯片后驅(qū)動(dòng)電機(jī)

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
使用proteus進(jìn)行仿真:運(yùn)用單片機(jī)(AT89C52、STM32、MSP430等)控制實(shí)現(xiàn)PWM輸出經(jīng)驅(qū)動(dòng)芯片后驅(qū)動(dòng)電機(jī),(驅(qū)動(dòng)自選)
需實(shí)現(xiàn)以下功能:
1、pwm頻率為1KHz,初始占空比50%
2、PWM輸出占空比可調(diào)
3、可控制兩個(gè)電機(jī)同時(shí)正轉(zhuǎn)、反轉(zhuǎn)、一正一反;
(可實(shí)物,提供所需器材)

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



單片機(jī)源程序如下:
  1. #include<reg52.h>  // 引入STC89C52頭文件
  2. #define uint unsigned int  // 定義無(wú)符號(hào)整型變量
  3. #define uchar unsigned char  // 定義無(wú)符號(hào)字符型變量

  4. uchar time, count = 50, flag1 = 1, flag2 = 1;  // 定義變量

  5. uchar seg[] = {0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39};  // 數(shù)碼管顯示字符

  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;  // 旋轉(zhuǎn)按鍵 1
  13. sbit key_turn2 = P2^6;  // 旋轉(zhuǎn)按鍵 2

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

  17. // 延時(shí)函數(shù)
  18. void delayxms(uint z);
  19. // 旋轉(zhuǎn)按鍵1的處理函數(shù)
  20. void Motor_turn1();
  21. // 旋轉(zhuǎn)按鍵2的處理函數(shù)
  22. void Motor_turn2();
  23. // 增加按鍵的處理函數(shù)
  24. void Motor_add();
  25. // 減少按鍵的處理函數(shù)
  26. void Motor_dec();
  27. // 定時(shí)器0的初始化函數(shù)
  28. void timer0_init();
  29. // 定時(shí)器1的初始化函數(shù)
  30. void timer1_init();
  31. // 向LCD發(fā)送命令
  32. void w_com(uchar j);
  33. // 向LCD發(fā)送數(shù)據(jù)
  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. // 延時(shí)函數(shù),延時(shí)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. // 處理旋轉(zhuǎn)按鍵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. // 處理旋轉(zhuǎn)按鍵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. // 定時(shí)器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. // 定時(shí)器0中斷處理函數(shù)
  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. // 定時(shí)器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. // 定時(shí)器1中斷處理函數(shù)
  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發(fā)送命令
  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發(fā)送數(shù)據(jù)
  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. }
復(fù)制代碼

Keil代碼與Proteus仿真下載: 仿真程序.7z (65.19 KB, 下載次數(shù): 19)

評(píng)分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎(jiǎng)勵(lì)!

查看全部評(píng)分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏2 分享淘帖 頂 踩
回復(fù)

使用道具 舉報(bào)

沙發(fā)
ID:1086975 發(fā)表于 2023-7-9 14:09 | 只看該作者
因?yàn)?1單片機(jī)性能不好,所以在液晶屏上顯示就是很不靈敏
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表