基于51单片机——蓝牙模块控制模拟舵机

 文章目录


前言

文章大部分转自微信公众号“牛元昌"————如果该实验对您有帮助的话,希望能大力支持,点个关注,如果能打赏就更好了;B站搜“牛元昌",有对应的实验视频演示及更多实验视频,您的支持是我继续创作的动力,如需要定制代码,请留言说明。


一、实验原理

1、该模块与单片机采用串口通信,手机连接蓝牙模块,通过无线信号发送数据给蓝牙模块,蓝牙模块收到数据后会把数据通过串口发送给单片机,单片机收到数据以后,就可以对数据进行处理,完成单片机对外设的控制。

        2、此实验需要把蓝牙模块配置为从机模式,若模块出厂默认就是从机模式,就不需要配置了,(如果不知道默认是什么模式,可以询问客服),主从一体的模块,可用AT指令配置成为从机模式就可以了,过程也比较简单,B站上面有很多教程。

二、实验模块

1.蓝牙模块

蓝牙模块的介绍:

         蓝牙模块可通过与单片机的串口相连,借助电脑或手机的蓝牙与单片机实现异步全双工通信。常见的蓝牙模块有HC-05主从一体蓝牙模块、HC-06从机蓝牙模块、低功耗BLE蓝牙模块(cc2540或cc2541)。不同的蓝牙模块的AT指令集不完全相同。 工作模式:AT模式,这时我们可以给模块发AT指令,从而可以查询模块的参数或者对模块进行设置。此时指示灯为慢闪状态。\n透传模式,就是当蓝牙模块连接上之后,我们可以直接发送数据给蓝牙模块,而蓝牙模块不管我们发的是什么,都直接传给对方。(这时发AT命令是无效的。)  蓝牙模块的使用:设置使用蓝牙模块的第一步为设置参数,如波特率等。方法为将USB转TTL模块与蓝牙模块连接,连接方法为\n\n并将USB转TTL模块插入电脑,检查设备管理器是否显示串口。然后使用串口调试助手写入AT指令进行查询或操作指令。蓝牙模块的响应将在串口调试助手的接收区显示。

2.sg90舵机模块

 SG90舵机简介

       SG90舵机是一种位置(角度)伺服的驱动器,适用于那些需要角度不断变化并可以保持的控制系统。在机器人机电控制系统中,舵机控制效果是性能的重要影响因素。舵机可以在微机电系统和航模中作为基本的输出执行机构,其简单的控制和输出使得单片机系统非常容易与之接口。

SG90舵机应用

      SG90舵机目前在高档遥控玩具,如航模、包括飞机模型、潜艇模型、遥控机器人中已经使用得比较普遍。

SG90舵机导线

      SG90舵机上有三根线,分别是GND(棕色线)、VCC(红色线)和SIG(黄色线),也就是地线、电源线和信号线

代码如下(示例):

/*
			模拟舵机控制原理:0~180°
				0.5ms-----0度
				1.0ms-----45度
				1.5ms-----90度
				2.0ms-----135度
				2.5ms-----180度
*/

#include <REGX52.H>
#include "timer0.h"
#include "Uart.h"
#include  "Delay.h"

sbit SG_PWM=P1^0;							//信号控制引脚
unsigned char count=0;					     //计次  赋初值为0
unsigned char PWM_count=0;         //0--0度,2--45度,4--90度,6--135度,8--180度
unsigned char i;
void main()
{
  Timer0_Init();
  Uart_Init();
  while(1)
  {
		
   }
}

void Timer0() interrupt 1      //特别注意此处,0--外部中断0,1--定时器中断0,2--外部中断1,3--定时器中断1,4--串行口中断1
{
  
  TL0 = 0x33;		
  TH0 = 0xFE;			 	//重新赋计数初值为0.5ms
  
  count++;					//计次,每500us  count+1
	count%=40;				//周期为20ms
	
  if(count<PWM_count)
  {									
    SG_PWM=1;
  }
  else
  {
    SG_PWM=0;
  }
}


void UART_Routine()  interrupt 4	//串口中断服务函数
{	
	if(RI==1)						//如果接收中断标志位为1
	{	
	  P1=SBUF;					   //发送的数据SBUF,赋给P1
	 if(P1==0X01)				   //0x01这个值可以通过蓝牙调试器编辑模式设置
	 {
	   PWM_count=1;
	 }
	 if(P1==0X02)				  
	 {
	   PWM_count=2;
	 }
	 if(P1==0X03)				 
	 {
	   PWM_count=3;
	 }
	 if(P1==0X04)				  
	 {
	   PWM_count=4;
	 }
	 if(P1==0X05)				  
	 {
	   PWM_count=5;
	 }
	 
 for(i = 1; i <= 3; i++)
	 {
		if(P1==0X06)				  
	 {
	   PWM_count=3;
		 PWM_count=PWM_count-i;

	 }
	 
	 if(P1==0X07)				  
	 {
	   PWM_count=3;
		 PWM_count = PWM_count+i;
		 
	 }
 }
	 
		
  RI=0;						//接收中断标志位软件清零
 }
}

总结(演示)

百度网盘:

链接:https://pan.baidu.com/s/1wOJ43Gf3_c2PyhR1tMzdpw

提取码:6666

蓝牙软件:手机应用市场搜索“蓝牙调试器”,即可下载;其他的蓝牙软件也同样适用。

  • 1
    点赞
  • 36
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
淘宝上卖的16路PWM舵机驱动模块的51单片机程序 部分程序如下 #include #include #include #include typedef unsigned char uchar; typedef unsigned int uint; sbit scl=P1^3; //时钟输入线 sbit sda=P1^4; //数据输入/输出端 sbit KEY1=P2^0; sbit KEY2=P2^1; #define PCA9685_adrr 0x80// 1+A5+A4+A3+A2+A1+A0+w/r //片选地址,将焊接点置1可改变地址, // 当IIC总 呱嫌 多片PCA9685或相同地址时才需焊接 // #define PCA9685_SUBADR1 0x2 // #define PCA9685_SUBADR2 0x3 // #define PCA9685_SUBADR3 0x4 #define PCA9685_MODE1 0x0 #define PCA9685_PRESCALE 0xFE #define LED0_ON_L 0x6 #define LED0_ON_H 0x7 #define LED0_OFF_L 0x8 #define LED0_OFF_H 0x9 // #define ALLLED_ON_L 0xFA // #define ALLLED_ON_H 0xFB // #define ALLLED_OFF_L 0xFC // #define ALLLED_OFF_H 0xFD #define SERVOMIN 115 // this is the 'minimum' pulse length count (out of 4096) #define SERVOMAX 590 // this is the 'maximum' pulse length count (out of 4096) #define SERVO000 130 //0度对应4096的脉宽计数值 #define SERVO180 520 //180度对应4096的脉宽计算值,四个值可根据不同舵机修改 /**********************函数的声明*********************************/ /*--------------------------------------------------------------- 毫秒延时函数 ----------------------------------------------------------------*/ void delayms(uint z) { uint x,y; for(x=z;x>0;x--) for(y=148;y>0;y--); } /*--------------------------------------------------------------- IIC总线所需的通用函数 ----------------------------------------------------------------*/ /*--------------------------------------------------------------- 微妙级别延时函数 大于4.7us ----------------------------------------------------------------*/ void delayus() { _nop_(); //在intrins.h文件里 _nop_(); _nop_(); _nop_(); _nop_(); } /*--------------------------------------------------------------- IIC总线初始化函数 ----------------------------------------------------------------*/ void init() { sda=1; //sda scl使用前总是被拉高 delayus(); scl=1; delayus(); } /*--------------------------------------------------------------- IIC总线启动信号函数 ----------------------------------------------------------------*/ void start() { sda=1; delayus(); scl=1; //scl拉高时 sda突然来个低电平 就启动了IIC总线 delayus(); sda=0; delayus(); scl=0; delayus(); } /*--------------------------------------------------------------- IIC总线停止信号函数 ----------------------------------------------------------------*/ void stop() { sda=0; delayus(); scl=1; //scl拉高时 sda突然来个高电平 就停止了IIC总线 delayus(); sda=1; delayus(); } /*--------------------------------------------------------------- IIC总线应答信号函数 ----------------------------------------------------------------*/ void ACK() { uchar i; scl=1; delayus(); while((sda=1)&&(i<255)) i++; scl=0; delayus(); } /*--------------------------------------------------------------- 写一个字节,无返回值,需输入一个字节值 ----------------------------------------------------------------*/ void write_byte(uchar byte) { uchar i,temp; temp=byte; for(i=0;i<8;i++) { temp=temp<<1; scl=0; delayus(); sda=CY; delayus(); scl=1; delayus(); } scl=0; delayus(); sda=1; delayus(); } /*--------------------------------------------------------------- 读一个字节函数,有返回值 ----------------------------------------------------------------*/ uchar read_byte() { uchar i,j,k; scl=0; delayus(); sda=1; delayus(); for(i=0;i<8;i++) { delayus(); scl=1; delayus(); if(sda==1) { j=1; } else j=0; k=(k<< 1)|j; scl=0; } delayus(); return k; } /*--------------------------------------------------------------- 有关PCA9685模块的函数 ----------------------------------------------------------------*/ /*--------------------------------------------------------------- 向PCA9685里写地址,数据 ----------------------------------------------------------------*/ void PCA9685_write(uchar address,uchar date) { start(); write_byte(PCA9685_adrr); //PCA9685的片选地址 ACK(); write_byte(address); //写地址控制字节 ACK(); write_byte(date); //写数据 ACK(); stop(); } /*--------------------------------------------------------------- 从PCA9685里的地址值中读数据(有返回值) ----------------------------------------------------------------*/ uchar PCA9685_read(uchar address) { uchar date; start(); write_byte(PCA9685_adrr); //PCA9685的片选地址 ACK(); write_byte(address); ACK(); start(); write_byte(PCA9685_adrr|0x01); //地址的第八位控制数据流方向,就是写或读 ACK(); date=read_byte(); stop(); return date; }
要实现51单片机通过蓝牙控制舵机,需要以下步骤: 1. 准备硬件设备:51单片机蓝牙模块舵机、电源等。 2. 将蓝牙模块连接到51单片机的串口,并将舵机接到单片机的PWM输出引脚上。 3. 编写51单片机的程序,通过蓝牙模块接收手机发送的指令,并将指令转换为对舵机控制信号。 4. 根据控制信号,控制舵机的转动角度和方向。 下面是一个简单的51单片机蓝牙控制舵机的程序示例: ```c #include <reg51.h> #define uchar unsigned char #define uint unsigned int sbit IN1 = P2^0; // 舵机控制引脚 void delay(uint xms) // 延时函数 { uint i, j; for (i = xms; i > 0; i--) { for (j = 110; j > 0; j--); } } void main() { uchar data; TMOD = 0x20; // 定时器1工作在方式2 TH1 = 0xFD; // 波特率设置为9600 TL1 = 0xFD; SCON = 0x50; // 串口工作在模式1 TR1 = 1; // 启动定时器1 while (1) { if (RI) { // 当接收到数据时 data = SBUF; // 读取串口数据 if (data == '0') { // 控制舵机转到0度 IN1 = 0; delay(10); IN1 = 1; delay(20); } else if (data == '1') { // 控制舵机转到90度 IN1 = 0; delay(15); IN1 = 1; delay(20); } else if (data == '2') { // 控制舵机转到180度 IN1 = 0; delay(20); IN1 = 1; delay(20); } RI = 0; // 清除接收中断标志 } } } ``` 在上面的程序中,我们通过串口接收手机发送的指令,根据指令控制舵机的转动。当收到字符'0'时,舵机转到0度;当收到字符'1'时,舵机转到90度;当收到字符'2'时,舵机转到180度。具体的控制信号需要根据舵机的型号和数据手册来确定。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

小邓分享官

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

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

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

打赏作者

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

抵扣说明:

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

余额充值