51单片机项目(1)--直流电机速度控制与测量的仿真

文章介绍了如何使用51单片机通过定时器生成PWM来控制直流电机的五级速度,并通过LED显示速度等级。同时,利用电机的脉冲输出进行测速,结果显示在LCD1602屏幕上,实现了正反转及速度显示功能。
摘要由CSDN通过智能技术生成

       实现的功能如下:可以通过按键调节电机的速度,有5个速度等级,每个等级都有一个LED与之对应,LED指示当前属于什么速度等级。可以通过一个单刀开关控制电机的正反转。同时正反转也有LED指示灯与之对应。同时,电机的速度也可以测量,将测到的速度显示在LCD1602上。

        首先给出整体的电路图吧

        控制直流电机速度的思路是产生PWM波,用PWM控制其速度。PWM的占空比越大,电机的速度就越快。51单片机不像32单片机,有PWM输出模式。但是51单片机有定时器,我们可以利用这个定时器来人工产生PWM。

        51单片机定时器产生PWM的代码如下:

        

 

       

        首先配置定时器。使其0.1ms产生一次中断,然后每次中断对timer1这个变量++,加到100又让它清0,那么timer1这个量一直在0-100这个范围内变换。然后在pwmz()这个函数里面,将timer1和传入的参数a作比较,由于timer1一直在变化但是a一旦被选定了就是固定的,那么通过调整a,就能调整PWM输出的高电平时间,也就是控制PWM的占空比。 

        51单片机测速的代码如下:

        

 

        测速思路如下:我选用的电机每转一圈会输出24个脉冲,将这个脉冲输出脚接到单片机的外部中断管脚,并且在外部中断服务函数里面记录脉冲数。然后使用定时器0,产生1ms的中断。在定时器0的中断服务函数里面 记录时间。当时间过去1S时,读取外部中断所记录的脉冲数,然后再通过数学关系就可以得到转速了。

        需要完整工程的  请私信!!

        

        

  • 2
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
#include #define uchar unsigned char #define uint unsigned int #define ulong unsigned long #define LED_DAT P0 sbit LED_SEG0 = P2^7; sbit LED_SEG1 = P2^6; sbit LED_SEG2 = P2^5; sbit LED_SEG3 = P2^4; #define TIME_CYLC 100 //12M晶振,定时器10ms 中断一次 我们1秒计算一次转速 // 1000ms/10ms = 100 #define PLUS_PER 10 //码盘的齿数 ,这里假定码盘上有10个齿,即传感器检测到10个脉冲,认为1圈 #define K 1.65 //校准系数 unsigned char code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}; uchar data Disbuf[4];// 显示缓冲区 uint Tcounter = 0; //时间计数器 bit Flag_Fresh = 0; // 刷新标志 bit Flag_clac = 0; //计算转速标志 bit Flag_Err = 0; //超量程标志 void DisplayFresh();//在数码管上显示一个四位数 void ClacSpeed();//计算转速,并把结果放入数码管缓冲区 void init_timer();//初始化定时器T0\T1 void Delay(uint ms);//延时函数 void it_timer0() interrupt 1 /* interrupt address is 0x000b */ { TF0 = 0; //定时器 T0用于数码管的动态刷新 TH0 = 0xC0; TL0 = 0x00; Flag_Fresh = 1; Tcounter++; if(Tcounter>TIME_CYLC) { Flag_clac = 1;//周期到,该重新计算转速了 } } void it_timer1() interrupt 3 /* interrupt address is 0x001b */ { TF1 = 0; //定时器T1用于单位时间内收到的脉冲数 //要速度不是很快,T1永远不会益处 Flag_Err = 1; //如果速度很高,我们应考虑另外一种测速方法:T测速法 } void main(void) { Disbuf[0] = 0; //开时,初始化为0000 Disbuf[1] = 0; Disbuf[2] = 0; Disbuf[3] = 0; init_timer(); while(1) { if(Flag_Fresh) { Flag_Fresh = 0; DisplayFresh(); // 定时刷新数码管显示 } if(Flag_clac) { Flag_clac = 0; ClacSpeed(); //计算转速,并把结果放入数码管缓冲区 Tcounter = 0;//周期定时 清零 TH1=TL1 = 0x00;//脉冲计数清零 } if(Flag_Err) //超量程处理 { Disbuf[0] = 0x9e; //开时,初始化为0000 Disbuf[1] = 0x9e; Disbuf[2] = 0x9e; Disbuf[3] = 0x9e; while(1) { DisplayFresh();//不再测速 等待复位i } } } } //在数码管上显示一个四位数 void DisplayFresh() { P2 |= 0xF0; LED_SEG0 = 0; LED_DAT = table[Disbuf[0]]; Delay(1); P2 |= 0xF0; LED_SEG1 = 0; LED_DAT = table[Disbuf[1]]; Delay(1); P2 |= 0xF0; LED_SEG2 = 0; LED_DAT = table[Disbuf[2]]; Delay(1); P2 |= 0xF0; LED_SEG3 = 0; LED_DAT = table[Disbuf[3]]; Delay(1); P2 |= 0xF0; } //计算转速,并把结果放入数码管缓冲区 void ClacSpeed() { uint speed ; uint PlusCounter; PlusCounter = TH1*256 + TL1; speed =6*PlusCounter/K; //K是校准系数,如速度不准,调节K的大小 Disbuf[3] = (speed/1000)%10; Disbuf[2] = (speed/100)%10; Disbuf[1] = (speed/10)%10; Disbuf[0] = speed%10; } void init_timer()//初始化定时器T0\T1 { TMOD=0x51; //定时器0工作于定时方式1,定时器1工作于计数方式 TH0=(65536-10000)/256; TL0=(65536-10000)%256; //TO定时时间为10000个周期即10毫秒 TH1=0x00; TL1=0x00; ET0=1; /* enable timer0 interrupt */ EA=1; /* enable interrupts */ TR0=1; /* timer0 run */ ET1=1; /* enable timer1 interrupt */ EA=1; /* enable interrupts */ TR1=1; } //延时函数 void Delay(uint ms) { uchar i; while(ms--) for(i=0;i<100;i++); }

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入式小李

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值