第 8 章 电机测速(自学二刷笔记)

embedded/2024/10/18 18:28:40/

重要参考:

课程链接:https://www.bilibili.com/video/BV1Ci4y1L7ZZ

讲义链接:Introduction · Autolabor-ROS机器人入门课程《ROS理论与实践》零基础教程

 

8.3.3 电机测速01_理论

测速实现是调速实现的前提,本节主要介绍AB相增量式编码器测速原理。

1.概念

百度百科关于编码器介绍如下:

编码器(encoder)是将信号(如比特流)或数据进行编制、转换为可用以通讯、传输和存储的信号形式的设备。编码器把角位移或直线位移转换成电信号,前者称为码盘,后者称为码尺。按照读出方式编码器可以分为接触式和非接触式两种;按照工作原理编码器可分为增量式和绝对式两类。增量式编码器是将位移转换成周期性的电信号,再把这个电信号转变成计数脉冲,用脉冲的个数表示位移的大小。绝对式编码器的每一个位置对应一个确定的数字码,因此它的示值只与测量的起始和终止位置有关,而与测量的中间过程无关。

2.测速原理

关于编码器相关概念简单了解即可,在此需要着重介绍的是 AB相增量式编码器测速原理:
AB相编码器主要构成为A相与B相,每一相每转过单位的角度就发出一个脉冲信号(一圈可以发出N个脉冲信号),A相、B相为相互延迟1/4周期的脉冲输出,根据延迟关系可以区别正反转,而且通过取A相、B相的上升和下降沿可以进行单频或2倍频或4倍频测速。

3.测速举例

假设编码器旋转1圈输出11个脉冲,减速比为 90。伪代码如下:

单频计数:

//设置一个计数器
int count = 0;
//当A为上升沿时
if(B为高电平){count++;
}else {count--;
}
//....
//速度=单位时间内统计的脉冲的个数 / (11*90) / 单位时间

2倍频计数:

//设置一个计数器
int count = 0;
//当A为上升沿时
if(B为高电平){count++;
}else {count--;
}
//当A为下降沿时
if(B为低电平){count++;
}else {count--;
}//....
//速度=单位时间内统计的脉冲的个数 / (11*2*90) / 单位时间

4倍频计数:

//设置一个计数器
int count = 0;
//当A为上升沿时
if(B为高电平){count++;
}else {count--;
}
//当A为下降沿时
if(B为低电平){count++;
}else {count--;
}
//当B为上升沿时
if(A为低电平){count++;
} else {count--;
}
//当B为下降沿时
if(A为高电平){count++;
} else {count--;
}
//....
//速度=单位时间内统计的脉冲的个数 / (11*4*90) / 单位时间

8.3.4 电机测速02_实现

需求:统计并输出电机转速。

思路:先统计单位时间内以单频或2倍频或4倍频的方式统计脉冲数,再除以一圈对应的脉冲数,最后再除以时间所得即为电机转速。

核心:计数时,需要在A相或B相的上升沿或下降沿触发时,实现计数,在此需要使用中断引脚与中断函数。

Arduino Mega 2560 的中断引脚:2 (interrupt 0), 3 (interrupt 1),18 (interrupt 5), 19 (interrupt 4), 20 (interrupt 3), 21 (interrupt 2)

实现流程:

  1. 编写Arduino程序先实现脉冲数统计;
  2. 编写Arduino程序再实现转速计算相关实现;
  3. 上传到Arduino并测试。
1.编码实现脉冲统计

核心知识点:attachInterrupt()函数(请参考 8.2.2 介绍)。

代码:

/** 测速实现:*  阶段1:脉冲数统计*  阶段2:速度计算* * 阶段1:*  1.定义所使用的中断引脚,以及计数器(使用 volatile 修饰)*  2.setup 中设置波特率,将引脚设置为输入模式*  3.使用 attachInterupt() 函数为引脚添加中断出发时机以及中断函数*  4.中断函数编写计算算法,并打印*    A.单频统计只需要统计单相上升沿或下降沿*    B.2倍频统计需要统计单相的上升沿和下降沿*    C.4倍频统计需要统计两相的上升沿和下降沿*  5.上传并查看结果*  * */
int motor_A = 21;//中端口是2
int motor_B = 20;//中断口是3
volatile int count = 0;//如果是正转,那么每计数一次自增1,如果是反转,那么每计数一次自减1 void count_A(){//单频计数实现//手动旋转电机一圈,输出结果为 一圈脉冲数 * 减速比/*if(digitalRead(motor_A) == HIGH){if(digitalRead(motor_B) == LOW){//A 高 B 低count++;  } else {//A 高 B 高count--;  }}*///2倍频计数实现//手动旋转电机一圈,输出结果为 一圈脉冲数 * 减速比 * 2if(digitalRead(motor_A) == HIGH){if(digitalRead(motor_B) == HIGH){//A 高 B 高count++;  } else {//A 高 B 低count--;  }} else {if(digitalRead(motor_B) == LOW){//A 低 B 低count++;  } else {//A 低 B 高count--;  }  }}//与A实现类似
//4倍频计数实现
//手动旋转电机一圈,输出结果为 一圈脉冲数 * 减速比 * 4
void count_B(){if(digitalRead(motor_B) == HIGH){if(digitalRead(motor_A) == LOW){//B 高 A 低count++;} else {//B 高 A 高count--;}} else {if(digitalRead(motor_A) == HIGH){//B 低 A 高count++;} else {//B 低 A 低count--;}}}void setup() {Serial.begin(57600);//设置波特率  pinMode(motor_A,INPUT);pinMode(motor_B,INPUT);attachInterrupt(2,count_A,CHANGE);//当电平发生改变时触发中断函数//四倍频统计需要为B相也添加中断attachInterrupt(3,count_B,CHANGE);
}void loop() {//测试计数器输出delay(2000);Serial.println(count);}
2.转速计算

思路:需要定义一个开始时间(用于记录每个测速周期的开始时刻),还需要定义一个时间区间(比如50毫秒),时时获取当前时刻,当当前时刻 - 上传结束时刻 >= 时间区间时,就获取当前计数并根据测速公式计算时时速度,计算完毕,计数器归零,重置开始时间

核心知识点:当使用中断函数中的变量时,需要先禁止中断noInterrupts(),调用完毕,再重启中断interrupts()(关于noInterrupts与interrupts请参考 8.2.2 介绍)。

代码(核心):

2中代码除了 loop 实现,无需修改。

int reducation = 90;//减速比,根据电机参数设置,比如 15 | 30 | 60
int pulse = 11; //编码器旋转一圈产生的脉冲数该值需要参考商家电机参数
int per_round = pulse * reducation * 4;//车轮旋转一圈产生的脉冲数 
long start_time = millis();//一个计算周期的开始时刻,初始值为 millis();
long interval_time = 50;//一个计算周期 50ms
double current_vel;//获取当前转速的函数
void get_current_vel(){long right_now = millis();  long past_time = right_now - start_time;//计算逝去的时间if(past_time >= interval_time){//如果逝去时间大于等于一个计算周期//1.禁止中断noInterrupts();//2.计算转速 转速单位可以是秒,也可以是分钟... 自定义即可current_vel = (double)count / per_round / past_time * 1000 * 60;//3.重置计数器count = 0;//4.重置开始时间start_time = right_now;//5.重启中断interrupts();Serial.println(current_vel);}
}void loop() {delay(10);get_current_vel();}
3.测试

将代码上传至Arduino,打开出口监视器,手动旋转电机,可以查看到转速信息。

 


http://www.ppmy.cn/embedded/38155.html

相关文章

密码学《图解密码技术》 记录学习 第十一章

目录 十一章 11.1 本章学习的内容 11.2 什么是密钥 11.2.1 密钥就是一个巨大的数字 DES 的密钥 三重 DES 的密钥 AES 的密钥 11.2.2 密钥与明文是等价的 11.2.3 密码算法与密钥 11.3各种不同的密钥 11.3.1 对称密码的密钥与公钥密码的密钥 11.3.2 消息认证码的密钥与…

【C++】每日一题 114 二叉树展开为链表

给你二叉树的根结点 root &#xff0c;请你将它展开为一个单链表&#xff1a; 展开后的单链表应该同样使用 TreeNode &#xff0c;其中 right 子指针指向链表中下一个结点&#xff0c;而左子指针始终为 null 。 展开后的单链表应该与二叉树 先序遍历 顺序相同。 #include <…

【Qt】深入Qt信号与槽:事件驱动编程的艺术与实践

文章目录 前言&#xff1a;1. Linux 信号 与 Qt 信号1.1. Linux 信号1.1.1. 概念1.1.2. 信号处理 1.2. Qt 信号1.2.2. 概念1.2.2. 信号处理方式1.2.3. 注意事项 2. connect 函数2.1. connect函数概述2.2. Qt类的继承关系2.3. connect具体使用方式&#xff1a; 3. 自定义槽函数3…

使用远程NVIDIA GPU资源加速本地计算的策略与实现

摘要&#xff1a; 在计算密集型任务如深度学习和科学计算中&#xff0c;GPU提供了必不可少的加速。NVIDIA作为领先的GPU制造商&#xff0c;其GPU广泛应用于各类计算任务。本文详细介绍了几种方法&#xff0c;通过这些方法&#xff0c;本地应用程序可以有效利用远程NVIDIA GPU资…

MySQL调优-01反范式化表设计

MySQL调优-01反范式化表设计 数据库设计三范式 第一范式&#xff08;1NF&#xff09;。确保数据库表的每一列都是不可分割的原子数据项&#xff0c;即列中不可包含数组、记录等非原子数据项&#xff1b;确保表中没有重复的属性或列&#xff0c;每个属性只出现一次&#xff0c;…

稀疏数据在机器学习任务中的应用问题

什么是稀疏数据 在机器学习任务中&#xff0c;稀疏数据是指在大量数据中&#xff0c;只有少部分数据是有效或非零的情况。在稀疏数据集中&#xff0c;有大量的0值或者缺失值。 例如&#xff0c;在自然语言处理中&#xff0c;当我们使用"词袋"模型表示文本信息时&am…

react状态管理之state

第三章 - 状态管理 随着你的应用不断变大&#xff0c;更有意识的去关注应用状态如何组织&#xff0c;以及数据如何在组件之间流动会对你很有帮助。冗余或重复的状态往往是缺陷的根源。在本节中&#xff0c;你将学习如何组织好状态&#xff0c;如何保持状态更新逻辑的可维护性&…

DHCP协议:动态主机配置协议

目录 概述 DHCP的功能 DHCP系统组成 DHCP报文分析 DHCP报文格式 DHCP选项分析 DHCP运行机制 客户端与服务器交互以分配IP地址 客户端与服务器交互以重用原来分配的地址 DHCP租约更新 租约释放 客户端状态转换 构造和发送DHCP报文 中继代理 总结 概述 DHCP&#x…