基于51单片机的PWM控制直流电机设计( proteus仿真+程序+报告+讲解视频)

基于51单片机的PWM控制直流电机设计( proteus仿真+程序+报告+讲解视频)

仿真图proteus 7.8及以上

程序编译器:keil 4/keil 5

编程语言:C语言

设计编号:S0031

视频

基于51单片机的PWM控制直流电机设计

主要功能:

1.设计要求

采用51单片机作为核心控制器,控制直流电机的正转,反转,制动,停止。

2.功能要求

1)数码管显示当前转动方向和当前的PWM占空比0~100%。

2)电机(L298n)转速可以通过按键调整,也可以开始暂停,正转和反转。

3)按键可实现加速、减速、正转、反转、停止等功能。

1键:加速键,短按,占空比加1;

2键:减速键,短按,占空比减1;

3键:正反转切换键,按下后电机正反转;

4键:按一下停止,PWM脉宽清零。

仿真

开始仿真后,用数码管第一位显示0表示正转后面三位标志PWM脉宽(0-100), 示波器实时检测显示脉宽黄线是正转脉宽,蓝线是反转脉宽。

img

反转脉宽如下所示

img

程序:

img

部分代码:

#include<reg52.h>    
#include<intrins.h>                        
#define uchar unsigned char	
#define uint unsigned int  
		
/**********************************************************************
							L298n接口定义
**********************************************************************/  
sbit MOTOR_A_1=P3^6;
sbit MOTOR_A_2=P3^7;																   
sbit k1=P1^0;		//定义k1为p1.0口
sbit k2=P1^1;   //定义k2为p1.1口
sbit k3=P1^2;   //定义k3为p1.2口
sbit k4=P1^3;   //定义k4为p1.3口
uchar T=0;	   //定时标记
uchar W=0;	   //脉宽值	 0~100
uchar A=0;	   //方向标记 0,1
uchar k=0;	   //按键标记
uchar i=0;    	 //计数变量

uchar code table1[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};

uchar code table2[]={0xfe,0xfb,0xfd,0xf7};
		 
void delayms(uint t);					 			 
/**********************************************************************
						数码管显示
**********************************************************************/ 
void disp(void)
{
	P2=table2[3];					
	P0=table1[W%10];			//显示占空比个位
	delayms(1); 					//延时1ms
	P2=0xff;							//P0清1

	P2=table2[2];
	P0=table1[W/100]; 		//显示占空比百位
	delayms(1); 					//延时1ms
	P2=0xff;							//P0清1
  
	P2=table2[1];
	P0=table1[W/10%10];  //显示占空比十位
	delayms(1); 				 //延时1ms
	P2=0xff;   					 //P0清1

	P2=table2[0];
	P0=table1[A];  			//显示方向
	delayms(1); 				//延时1ms
	P2=0xff;						//P0清1
}

/**********************************************************************
							定时器变量定义
**********************************************************************/ 


void init(void)
{
	//启动中断
	TMOD=0x01;	
	EA=1;						
	ET0=1;
	TR0=1;
	//设置定时时间
	TH0=0xff;
	TL0=0xf6;
}

		
/**********************************************************************
							延时1ms
**********************************************************************/ 
void delayms(uint t)
{
	uchar j;
	while(t--)
	{
		for(j=0;j<250;j++)			//循环250次
		{
		   _nop_();             //系统延时          
	       _nop_();							//系统延时
	       _nop_();							//系统延时
	       _nop_();							//系统延时
		}
	}
}
/**********************************************************************
							独立按键检测
**********************************************************************/ 
void key(void)           //按键判断程序
{
	if(k1==0)							//按键1按下
	{
		while(k1==0);				//按键1抬起
		if(W==100)					//如果脉宽为100
			W=0;							//脉宽置0
		else
			W+=1;							//否则加1
	}
	else if(k2==0)				//按键2按下
	{				 
		while(k2==0);				//按键2抬起
		if(W==0)						//如果脉宽为0
			W=100;						//脉宽设置成100
		else
			W-=1;							//否则减1
	}
	else if(k3==0)				//按键3按下
	{			   
		while(k3==0);				//按键3抬起
		A=!A;								//方向标记取反
		k=!k;								//按键标记取反
	} 
	else if(k4==0)				//按键4按下
	{			   
		while(k4==0);				//按键4抬起
		W=0;								//脉宽清0									
	}
}

void main(void)
{

	init();			/系统初始化 
	while(1)	 
	{
	if(k==0)
	    MOTOR_A_2=0;
	else
		MOTOR_A_1=0;	 		
		key();			查询按键
	}
}

设计报告

img

资料清单:

下载方式见开头的视频
或下方百度云分享链接
https://docs.qq.com/doc/DS1RGSmxKQUZOa0dT
img

#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++); }
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

BT-BOX

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

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

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

打赏作者

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

抵扣说明:

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

余额充值