文章目录
实现电机最常使用的两个功能,转速控制和位置控制
使用PID闭环控制(控制线性系统最简单快捷的控制方法)
为了实现控制电机转动和闭环控制
需要:
- 电机(废话)
- 编码器(霍尔编码器或者光电编码器均可)
- 电机驱动(这里选的是l298n模块)

千万注意黑色的地线,单片机的地要与12V的地(L298n的地)连接
使用硬件PWM输出,定时器1,输出两路PWM分别代表PWM1和PWM2
设置频率为2.4KHz(约417us),最大占空比5000
使用通道1和2,其余均默认设置

定时器1初始化设置(生成的代码),里开启定时器与PWM输出
HAL_TIM_Base_Start_IT(&htim1);
HAL_TIM_Base_Start(&htim1);
HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1);
HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_2);
使用定时器的编码器模式,双边沿计数,默认设置就可


定时器2的初始化设置里加入,开启编码器模式和定时器
HAL_TIM_Encoder_Start(&htim2, TIM_CHANNEL_ALL); //开启编码器模式
HAL_TIM_Base_Start_IT(&htim2);
HAL_TIM_Base_Start(&htim2);
每10ms触发一次中断,用于计算PID


注意要打开中断
开启定时器中断
HAL_TIM_Base_Start_IT(&htim3);
HAL_TIM_Base_Start(&htim3);
用于调试,默认设置就可,使用printf重定向,无需开启中断

定时10ms读取一次编码器的计数值并清零,计算速度
电机是15线霍尔传感器,34:1减速比
详情看这个博客,传送门
float Get_Speed()
{
int16_t zj;
float Speed = 0;
zj = __HAL_TIM_GetCounter(&Encoder_TIM_Handle);
__HAL_TIM_SetCounter(&Encoder_TIM_Handle, 0);
Speed = (float)zj / (4 * 15 * 34) * 100 * 60;
return Speed;
}
间隔一段时间读取编码器的计数值(清零操作交由速度获取函数处理)
调用时需要将函数的输出值进行累加
float Get_Angle()
{
int16_t zj;
float angle = 0;
zj = __HAL_TIM_GetCounter(&Encoder_TIM_Handle);
angle = (float)zj / (4 * 15 * 34) * 360;
return angle;
}
通过更改PWM的占空比来控制电机转速
void motor(int16_t Speed)
{
if (Speed == 0)
{
__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel1, Motor_MAX_Duty + 1);
__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel2, Motor_MAX_Duty + 1);
}
else if (Speed > 0)
{
__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel1, Speed);
__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel2, 0);
}
else if (Speed < 0)
{
Speed *= -1;
__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, TIM_CHANNEL_1, 0);
__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, TIM_CHANNEL_2, Speed);
}
}
使用增量式PID
PID原理请看,传送门
typedef struct __PID_Increment_Struct
{
float Kp, Ki, Kd; //系数
float Error_Last1; //上次误差
float Error_Last2; //上次误差
float Out_Last; //上次输出
} PID_Increment_Struct;
float PID_Increment(PID_Increment_Struct *PID, float Current, float Target)
{
float err, //误差
out, //输出
proportion, //比例
differential; //微分
err = (float)Target - (float)Current; //计算误差
proportion = (float)err - (float)PID->Error_Last1; //计算比例项
differential = (float)err - 2 * (float)PID->Error_Last1 + (float)PID->Error_Last2; //计算微分项
out = (float)PID->Out_Last + (float)PID->Kp * proportion + (float)PID->Ki * err + (float)PID->Kd * differential; //计算PID
PID->Error_Last2 = PID->Error_Last1; //更新上上次误差
PID->Error_Last1 = err; //更新误差
PID->Out_Last = out; //更新上此输出
return out;
}
速度环就是让电机保持固定转速的PID控制系统
逻辑框图如下,
通过编码器获得转速送到输入作为反馈
输出通过控制PWM(正负和占空比)来控制电机转速
输入的是目标的转速
注意:PID的系数与间隔时间有关,PID需要间隔固定的时间进行调用

那编程的思路就很明显了,我们使用一个定时器中断,在固定的时间(10ms)调用计算一次PID
在这个定时器中断里,我们首先读取转速,之后压入PID进行计算,再将PWM给到电机就行
为了便于观察,这里加上了使用Printf通过串口发送给上位机显示的功能
这里的PID的参数是我调好的
PID_Increment_Struct PID_Speed = {3, 0.6, 0.6};
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
float Speed = 0;
int16_t set_speed = 0;
if (htim == &htim2)
{
}
else if (htim == &htim3)
{//10ms中断
Speed = Get_Speed();//获取转速
mb_speed = 3000;
set_speed = PID_Increment(&PID_Speed, Speed, mb_speed);//PID
if (set_speed > 5000)
set_speed = 5000;
else if (set_speed < -5000)
set_speed = -5000;//限幅
if (set_speed > 500 || set_speed < -500)//死区控制,改善电机异响
motor(set_speed);
printf("%f,%f\r\n", Speed,mb_speed);//打印当前和目标转速
}
}
这个是我用的电机,从某个车模上拆的,带有15线霍尔传感器,34:1减速比,额定电压12V,额定转速350 r/min

速度环是PID控制器,
我们的调整顺序是P->I->D
下面的图.横轴是时间,红线代表的是当前转速,绿线代表目标转速
比例部分是绝对的主力
如果P的极性错误,则电机会反相开到最大转速
我们从小向大调
Kp=1,Ki=0
我们可以看到,电机不转动,只有异响,说明Kp过小(至少一个数量级)

我们增大Kp,令Kp=10,Ki=0
可以看到,电机已经开始转动,但是距离需要的转速过远(Kp在同等数量级了)

我们继续增大Kp,令Kp=30,Ki=0
这时发现,转速已经达到了目标转速的2/3以上
这时我们继续增大Kp

Kp增大到80
发现并没有继续接近目标值很多了
这时再增加Kp也不会更接近目标值了
我们需要引入Ki了

这里放个Kp过大的现象,Kp=700
这种是电机来不及反应造成的

积分项是用于消除静态偏差(也就是Kp在合理范围内变大也无法继续接近目标值的现象)
我们让Kp=30开始调Ki
如果Ki的极性错误,则会出现如下图,即电机来回震荡运动
Kp=80,Ki=1
发现已经可以达到目标值了,回正速度比较慢,我们继续增大Ki(同一数量级)

Kp=80,Ki=5
这时就已经比较完美了,符合了我的要求了
如果自己的要求更高,可以减少步进值慢慢调一下

到了这里,速度环PID我们已经调完了
转速已经可以稳定了
这是调节位置环的前提
位置环是建立在速度环之上的
使用串级PID进行控制,内环是速度环,外环是位置环
可以加快收敛速度,提高抗干扰能力
我们的策略是当误差大于一圈(>360°或<-360°)时让电机自己以300r/min旋转,不引入PID控制
当误差在一圈内时,使用PID控制

加入位置环的代码如下
PID_Increment_Struct PID_Speed = {3, 0.6, 0.6};
PID_Increment_Struct PID_Angle = {3.1, 0, 0.06};
float angle;//角度
int aa = 0;//目标角度
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
float Speed = 0;
int16_t set_speed = 0;
float mb_speed;//目标速度
if (htim == &htim2)
{
}
else if (htim == &htim3)
{
angle += Get_Angle();
Speed = Get_Speed();
mb_speed = (int16_t)PID_Increment(&PID_Angle, angle, aa);
if (PID_Angle.Error_Last1 > 360)
mb_speed = 300;
else if (PID_Angle.Error_Last1 < -360)
mb_speed = -300;
// mb_speed = 300;
set_speed = PID_Increment(&PID_Speed, Speed, mb_speed);
if (set_speed > 5000)
set_speed = 5000;
else if (set_speed < -5000)
set_speed = -5000;
if (set_speed > 500 || set_speed < -500)//改善死区
motor(set_speed);
//printf("%f\r\n", Speed);
Speed = aa;
printf("%f,%f\r\n", angle, Speed);//输出当前和目标角度
}
}
只使用了P即可达到要求
注意,再位置环调整之前,要将速度环调整完毕
下图的横坐标是时间,红线是当前转动角度,绿线是目标角度
Kp=1时,设置的目标值是4000
前面的直线部分是误差大于一圈的,以固定转速旋转
最后的误差很小了


Kp=3
目标值每隔2s从-400到400

电机位置环,串级pid
电机速度环和位置环PID调参教程
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
这里是Ruby新手。完成一些练习后碰壁了。练习:计算一系列成绩的字母等级创建一个方法get_grade来接受测试分数数组。数组中的每个分数应介于0和100之间,其中100是最大分数。计算平均分并将字母等级作为字符串返回,即“A”、“B”、“C”、“D”、“E”或“F”。我一直返回错误:avg.rb:1:syntaxerror,unexpectedtLBRACK,expecting')'defget_grade([100,90,80])^avg.rb:1:syntaxerror,unexpected')',expecting$end这是我目前所拥有的。我想坚持使用下面的方法或.join,
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt
文章目录1.开发板选择*用到的资源2.串口通信(个人理解)3.代码分析(注释比较详细)1.主函数2.串口1配置3.串口2配置以及中断函数4.注意问题5.源码链接1.开发板选择我用的是STM32F103RCT6的板子,不过代码大概在F103系列的板子上都可以运行,我试过在野火103的霸道板上也可以,主要看一下串口对应的引脚一不一样就行了,不一样的就更改一下。*用到的资源keil5软件这里用到了两个串口资源,采集数据一个,串口通信一个,板子对应引脚如下:串口1,TX:PA9,RX:PA10串口2,TX:PA2,RX:PA32.串口通信(个人理解)我就从串口采集传感器数据这个过程说一下我自己的理解,
基础版云数据库RDS的产品系列包括基础版、高可用版、集群版、三节点企业版,本文介绍基础版实例的相关信息。RDS基础版实例也称为单机版实例,只有单个数据库节点,计算与存储分离,性价比超高。说明RDS基础版实例只有一个数据库节点,没有备节点作为热备份,因此当该节点意外宕机或者执行重启实例、变更配置、版本升级等任务时,会出现较长时间的不可用。如果业务对数据库的可用性要求较高,不建议使用基础版实例,可选择其他系列(如高可用版),部分基础版实例也支持升级为高可用版。基础版与高可用版的对比拓扑图如下所示。优势 性能由于不提供备节点,主节点不会因为实时的数据库复制而产生额外的性能开销,因此基础版的性能相对于
我使用irb。下面是我写的代码。“斧头”..“bc”我期待"ax""ay""az""ba"bb""bc"但结果只是“斧头”..“bc”我该如何纠正?谢谢。 最佳答案 >puts("ax".."bc").to_aaxayazbabbbc 关于ruby-从结束值创建一系列字符串,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/7617092/
LL库和HAL库简介LL:Low-Layer,底层库HAL:HardwareAbstractionLayer,硬件抽象层库LL库和hal库对比,很精简,这实际上是一个精简的库。LL库的配置选择如下:在STM32CUBEMX中,点击菜单的“ProjectManager”–>“AdvancedSettings”,在下面的界面中选择“AdvancedSettings”,然后在每个模块后面选择使用的库总结:1、如果使用的MCU是小容量的,那么STM32CubeLL将是最佳选择;2、如果结合可移植性和优化,使用STM32CubeHAL并使用特定的优化实现替换一些调用,可保持最大的可移植性。另外HAL和L
使用RubyonRails,我使用给定的增量(例如每30分钟)用时间填充“选择”。目前我正在YAML文件中写出所有的可能性,但我觉得有一种更巧妙的方法。我想我想提供一个开始时间、一个结束时间、一个增量,并且目前只提供一个名为“关闭”的选项(想想“business_hours”)。所以,我的选择可能会显示:'Closed'5:00am5:30am6:00am...[allthewayto]...11:30pm谁能想出更好的方法,或者只是将它们全部“拼写”出来的最佳方法? 最佳答案 此答案基于@emh的答案。defcreate_hour
目录一、ESP32简单介绍二、ESP32Wi-Fi模块介绍三、ESP32Wi-Fi编程模型四、ESP32Wi-Fi事件处理流程 五、ESP32Wi-Fi开发环境六、ESP32Wi-Fi具体代码七、ESP32Wi-Fi代码解读6.1主程序app_main7.2自定义代码wifi_init_sta()八、ESP32Wi-Fi连接验证8.1测试方法8.2服务器模拟工具sscom58.3测试代码8.4测试结果前言为了开发一款亚马逊物联网产品,开始入手ESP32模块。为了能够记录自己的学习过程,特记录如下操作过程。一、ESP32简单介绍ESP32是一套Wi-Fi(2.4GHz)和蓝牙(4.2)双模解决方
有道无术,术尚可求,有术无道,止于术。本系列SpringBoot版本3.0.4本系列SpringSecurity版本6.0.2本系列SpringAuthorizationServer版本1.0.2源码地址:https://gitee.com/pearl-organization/study-spring-security-demo文章目录前言1.OAuth2AuthorizationServerMetadataEndpointFilter2.OAuth2AuthorizationEndpointFilter3.OidcProviderConfigurationEndpointFilter4.N