找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3667|回复: 7
收起左侧

单片机步进电机调速控制 LCD1602显示圈数程序与Proteus仿真操作流

[复制链接]
ID:481843 发表于 2022-5-1 17:56 | 显示全部楼层 |阅读模式
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
1.png

Proteus仿真操作流如下:
步骤一:打开仿真.DSN文件(用Proteus7.8打开)
步骤二:左键双击U1单片机,打开加载HEX文件界面,左键单击文件夹图标

步骤三:选择给的1号源程序文件夹,找到HEX文件。
步骤四:左键双击打开HEX文件,跳到步骤二界面,点击OK

步骤五:左键点击左下角运行


单片机源程序如下:
  1. #include<reg52.h>
  2. #include<stdio.h>
  3. #include<intrins.h>

  4. #define uchar unsigned char
  5. #define uint unsigned int

  6. sbit RS = P2^4;//1602数据/命令选择端(H:数据寄存器L:指令寄存器)
  7. sbit RW = P2^5;//1602读/写选择端
  8. sbit E  = P2^6;//1602使能信号端

  9. sbit key1=P3^0;//用户按键
  10. sbit key2=P3^1;
  11. sbit key3=P3^2;
  12. sbit key4=P3^3;
  13. sbit key5=P3^4;

  14. uchar code B_Rotation[8]={0x7f,0x3f,0xbf,0x9f,0xdf,0xcf,0xef,0x6f}; //反转表格
  15. uchar code F_Rotation[8]={0xef,0xcf,0xdf,0x9f,0xbf,0x3f,0x7f,0x6f}; //正转表格


  16. uchar table1[]="dir: +  spe:005";
  17. uchar table2[]="cir:000 ang:000";

  18. uchar num;
  19. uint key,flag,speed=5,zflag,znum,fflag,fnum,select,pp,qq;
  20. //1键值2键值标志3速度4正转标志5正转圈数6反转标志7反转圈数8正转角度9反转角度10选择
  21. float zang,fang;
  22. /******************************************************************/
  23. /*                    延时函数 Xms                                   */
  24. /******************************************************************/
  25. void delay(uint z)
  26. {
  27.         uint x,y;
  28.         for(x=z;x>0;x--)
  29.                 for(y=110;y>0;y--);
  30. }
  31. /******************************************************************/
  32. /*                   LCD1602写命令操作                            */
  33. /******************************************************************/
  34. void WriteCommand(uchar com)
  35. {
  36.         delay(5);//操作前短暂延时,保证信号稳定
  37.         E=0;
  38.         RS=0;
  39.         RW=0;
  40.         P0=com;
  41.         E=1;
  42.         delay(5);
  43.         E=0;
  44. }
  45. /******************************************************************/
  46. /*                   LCD1602写数据操作                            */
  47. /******************************************************************/
  48. void WriteData(uchar dat)
  49. {
  50.         delay(5);  //操作前短暂延时,保证信号稳定
  51.         E=0;
  52.         RS=1;
  53.         RW=0;
  54.         P0=dat;
  55.         E=1;
  56.         delay(5);
  57.         E=0;
  58. }
  59. /******************************************************************/
  60. /*                           1602初始化程序                       */
  61. /******************************************************************/
  62. void InitLcd()//1602初始化程序
  63. {
  64.         delay(15);
  65.         WriteCommand(0x38); //display mode
  66.         WriteCommand(0x38); //display mode
  67.         WriteCommand(0x38); //display mode
  68.         WriteCommand(0x06); //当读或写一个字符后地址指针加一,且光标加一
  69.         WriteCommand(0x0c); //显示开及光标不显示
  70.         WriteCommand(0x01); //清屏

  71.         WriteCommand(0x80);
  72.         for(num=0;num<15;num++)//第一行的显示
  73.         {
  74.                 WriteData(table1[num]);
  75.                 delay(15);
  76.         }
  77.         WriteCommand(0x80+0x40);//第二行的显示
  78.         for(num=0;num<15;num++)
  79.         {
  80.                 WriteData(table2[num]);
  81.                 delay(15);
  82.         }

  83. }
  84. /******************************************************************/
  85. /*              独立键盘扫描函数                                  */
  86. /******************************************************************/
  87. void keyscan()  
  88. {        
  89.         
  90.         P3=0xff;//拉高P3口,以读取P3口的值
  91.         if(key1==0)//键1被按下
  92.         {
  93.                 delay(5);//延时消抖
  94.                 if(key1==0)
  95.                 {
  96.                         flag=1;
  97.                 }
  98.                 while(!key1);
  99.                 delay(5);//延时消抖
  100.                 while(!key1);//松手检测
  101.         }
  102.         if(key2==0)
  103.         {
  104.                 delay(5);
  105.                 if(key2==0)
  106.                 {
  107.                         flag=2;
  108.                 }
  109.                 while(!key2);
  110.                 delay(5);
  111.                 while(!key2);
  112.         }
  113.         if(key3==0)//键1被按下
  114.         {
  115.                 delay(5);//延时消抖
  116.                 if(key3==0)
  117.                 {
  118.                         flag=3;
  119.                 }
  120.                 while(!key3);
  121.                 delay(5);//延时消抖
  122.                 while(!key3);//松手检测
  123.         }
  124.         if(key4==0)
  125.         {
  126.                 delay(5);
  127.                 if(key4==0)
  128.                 {
  129.                         flag=4;
  130.                 }
  131.                 while(!key4);
  132.                 delay(5);
  133.                 while(!key4);
  134.         }
  135.         if(key5==0)
  136.         {
  137.                 delay(5);
  138.                 if(key5==0)
  139.                 {
  140.                         flag=5;
  141.                 }
  142.                 while(!key5);
  143.                 delay(5);
  144.                 while(!key5);
  145.         }
  146. }
  147. /******************************************************************/
  148. /*                           1602显示3位数                       */
  149. /******************************************************************/
  150. void write_sfm(uchar add,uint date)//1602显示
  151. {
  152.         uchar bai,shi,ge;
  153.         bai=date/100;
  154.         shi=date%100/10;
  155.         ge=date%10;
  156.         WriteCommand(0x80+add);//设置数据地址指针
  157.         WriteData(0x30+bai);
  158.         WriteData(0x30+shi);
  159.         WriteData(0x30+ge);
  160.         WriteData(0x20);
  161. }
  162. /******************************************************************/
  163. /*                           键值处理                             */
  164. /******************************************************************/
  165. void handle_flag()
  166. {        

  167.         uchar select1;
  168.         if(TR0==0&&TR1==0)
  169.         {        
  170.                 if(flag==1) //正转
  171.                 {
  172.                         WriteCommand(0x80+0x05);
  173.                         WriteData('+');
  174.                         delay(20);
  175.                         select=0;
  176.                 }
  177.                 if(flag==2) //反转
  178.                 {
  179.                         WriteCommand(0x80+0x05);
  180.                         WriteData('-');
  181.                         delay(20);
  182.                         select=1;
  183.                 }
  184.                 if(flag==3)//转速加
  185.                 {
  186.                     speed++;
  187.                         if(speed>=12)
  188.                                 speed=12;
  189.                         write_sfm(0x0c,speed);//显示速度
  190.                         delay(20);
  191.             }
  192.                 if(flag==4)        //转速减
  193.                 {
  194.                         speed--;
  195.                         if(speed<=1)
  196.                                 speed=1;
  197.                         write_sfm(0x0c,speed);//显示速度
  198.                         delay(20);
  199.                 }
  200. }
  201.         if(flag==5)////关闭中断,停止电机
  202.         {
  203.                 select1++;
  204.                 select1%=2;
  205.                 if( select1==0)
  206.                 {
  207.                         TR0=0;
  208.                         TR1=0;
  209.                         if(select==0)//显示转动的角度
  210.                                 write_sfm(0x40+0x0c,zang);
  211.                         else
  212.                                 write_sfm(0x40+0x0c,fang);
  213.              }
  214.                 else
  215.                 {if(select==0)
  216.                 {
  217.                         TR0=1;//开定时器0,产生正转相序
  218.                         TR1=0;//关定时器1
  219.                 }
  220.                 else
  221.                 {        
  222.                         TR1=1;//开定时器1,产生反转相序
  223.                         TR0=0;//关定时器0        
  224.                 }
  225.                 }                        
  226.         }
  227.         flag=0;
  228. }


  229. void main()
  230. {
  231.         
  232.         pp=0;
  233.         qq=0;
  234.    
  235.         InitLcd();
  236.         TMOD=0x11;
  237.     EA=1;
  238.         ET0=1;
  239.         ET1=1;

  240.         TH0=(65536-(14648/speed))/256;
  241.         TL0=(65536-(14648/speed))%256;
  242.         TH1=(65536-(14648/speed))/256;
  243.         TL1=(65536-(14648/speed))%256;

  244.         TR0=0;
  245.         TR1=0;
  246.         
  247.         while(1)
  248.         {        
  249.                 keyscan();
  250.                 handle_flag();        
  251.         }

  252. }

  253. void time0() interrupt 1
  254. {        
  255.         uint zz;
  256.         TH0=(65536-(14648/speed))/256; // 360/(5.625/64)=4096   
  257.                                    //假设60秒转一圈,即60秒中断4096次,
  258.                                                                    //那么一次中断需要60000000/4096=14648个时钟
  259.                                                                    //60000000/(4096*speed)   zflag=4096时一圈
  260.         TL0=(65536-(14648/speed))%256;
  261.         zz%=8;
  262.         P1=F_Rotation[zz];  //输出对应的相
  263.         zz++;
  264.         zflag++;
  265.         if(zflag==4096) //正转了一圈
  266.         {        
  267.                 zflag=0;
  268.                 znum++;
  269.                 write_sfm(0x40+0x04,znum);
  270.         }
  271.         if(zflag%64==0)//计算正转动的角度
  272.         {
  273.                 pp++;
  274.                 zang=(uint)(pp*5.625);
  275.                 if(zang==360)
  276.                         pp=0;
  277.                 write_sfm(0x40+0x0c,zang);
  278.         }
  279.         //zang=0.088*zflag;
  280.         //write_sfm(0x40+0x0c,zang);
  281. }
  282. void time1() interrupt 3
  283. {        
  284.         
  285.         uint ff;
  286.         TH1=(65536-(14648/speed))/256;
  287.         TL1=(65536-(14648/speed))%256;
  288.     ff%=8;
  289.         P1=B_Rotation[ff]; //输出对应的相
  290.         ff++;
  291.         fflag++;
  292.         if(fflag==4096) //反转了一圈
  293.         {        
  294.                 fflag=0;
  295.                 fnum++;
  296.                 write_sfm(0x40+0x04,fnum);
  297.         }
  298.         if(fflag%64==0)//计算反转动的角度
  299.         {
  300.                 qq++;
  301.                 fang=(uint)(qq*5.625);
  302.                 if(fang==360)
  303.                 qq=0;
  304.                 write_sfm(0x40+0x0c,fang);
  305.         }
  306. }
复制代码

Keil4代码与Proteus7.5,8.8仿真下载:
资料.7z (388.36 KB, 下载次数: 166)
回复

使用道具 举报

ID:1032999 发表于 2022-6-15 08:59 | 显示全部楼层
您好,请问一下最后一个按键的作用是什么
回复

使用道具 举报

ID:95059 发表于 2022-7-9 20:59 | 显示全部楼层
转真电机不能转,啥情况啊??
回复

使用道具 举报

ID:228452 发表于 2022-7-9 22:17 | 显示全部楼层

"Hello, what is the function of the last button?"
Snap1.jpg
回复

使用道具 举报

ID:228452 发表于 2022-7-9 22:21 | 显示全部楼层
"The real motor can't turn, what's the situation? ?"

It is motor wiring (driver sequence)
also motor voltage should be 5V
回复

使用道具 举报

ID:95059 发表于 2022-7-23 17:21 | 显示全部楼层
确实不能转动,可惜了
回复

使用道具 举报

ID:445725 发表于 2024-12-23 18:31 | 显示全部楼层
非常不错的,可以转动,不过电机很抖,最大只能到12圈就加不上去了
回复

使用道具 举报

ID:445725 发表于 2024-12-23 18:39 | 显示全部楼层
可以转动,感谢老师的辛苦
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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