基于STM32F103C8T6的同步电机驱动-电流环PI与力矩模式

基于STM32F103C8T6的同步电机驱动-电流环PI与力矩模式

本系列文章:

  1. 基于STM32F103C8T6的同步电机驱动-CubeMX配置与IQmath调用
  2. 基于STM32F103C8T6的同步电机驱动-PWM驱动代码以及SVPWM的实现
  3. 基于STM32F103C8T6的同步电机驱动-ADC采样与基于MT6701的角度获取
  4. 基于STM32F103C8T6的同步电机驱动-电流采样去直流偏置
  5. 基于STM32F103C8T6的同步电机驱动-电流环PI与力矩模式

一、电流环PI控制器代码编写

本文中选用的电流环控制器为抗积分饱和PI控制器。

/****file:my_pid.c*****/
#include "my_pid.h"

//电流环参数
uint16_t wb		= 200;			//电流环带宽
float Kp_c		= 0.168f;		//电流环P参数:Kp_c=wb*L
float Ki_c		= 0.0612f;		//电流环I参数:Ki_c=wb*R*Ts
float Ka_c		= 5.9523f;		//电流环A参数:Ka_c=1/Kp_c

/**********************************************************************
 * 函数名称: id_pi_contral
 * 功能描述: 电流环PI控制器
 * 输入参数: _iq id_now;_iq id_tar;
 * 输出参数: _iq id_out;
 * 返 回 值: 无
 * 修改日期        版本号     修改人        修改内容
 * -----------------------------------------------
 * 2024/07/05        V1.0     TONY0925       创建
 ***********************************************************************/
_iq id_pi_contral(_iq id_now,_iq id_tar)
{
    static _iq err_id = _IQ(0);
    static _iq u_id   = _IQ(0);
    static _iq out_id = _IQ(0);
    static _iq excess_id = _IQ(0);
    static _iq sum_id    = _IQ(0);
	//PI电流环1.预期值与测量值作差得到偏差值
	err_id = id_tar  - id_now;
	//PI电流环2.计算PI控制器预输出值
	u_id   = sum_id + _IQmpy( _IQ(Kp_c) , err_id );
	//PI电流环3.进行限幅输出,得到最终输出值
	if (u_id >_IQ(Umax))
	{
		out_id = _IQ(Umax);
	}
	else if (u_id <_IQ(-Umax))
	{
		out_id = _IQ(-Umax);
	}
	else
	{
		out_id = u_id;
	}	
	//PI电流环4.将预输出值与实际输出值作差得到饱和差值
	excess_id = u_id - out_id;
	//PI电流环5.将饱和差值代入积分累计项
	sum_id = sum_id + _IQmpy( _IQ(Ki_c) , err_id ) - _IQmpy( _IQ(Ka_c) , excess_id );
	return out_id;
}
/**********************************************************************
 * 函数名称: iq_pi_contral
 * 功能描述: 电流环PI控制器
 * 输入参数: _iq iq_now;_iq iq_tar;
 * 输出参数: _iq iq_out;
 * 返 回 值: 无
 * 修改日期        版本号     修改人        修改内容
 * -----------------------------------------------
 * 2024/07/05        V1.0     TONY0925       创建
 ***********************************************************************/
_iq iq_pi_contral(_iq iq_now,_iq iq_tar)
{
    static _iq err_iq = _IQ(0);
    static _iq u_iq   = _IQ(0);
    static _iq out_iq = _IQ(0);
    static _iq excess_iq = _IQ(0);
    static _iq sum_iq    = _IQ(0);
	//PI电流环1.预期值与测量值作差得到偏差值
	err_iq = iq_tar  - iq_now;
	//PI电流环2.计算PI控制器预输出值
	u_iq   = sum_iq + _IQmpy( _IQ(Kp_c) , err_iq );
	//PI电流环3.进行限幅输出,得到最终输出值
	if (u_iq >_IQ(Umax))
	{
		out_iq = _IQ(Umax);
	}
	else if (u_iq <_IQ(-Umax))
	{
		out_iq = _IQ(-Umax);
	}
	else
	{
		out_iq = u_iq;
	}	
	//PI电流环4.将预输出值与实际输出值作差得到饱和差值
	excess_iq = u_iq - out_iq;
	//PI电流环5.将饱和差值代入积分累计项
	sum_iq = sum_iq + _IQmpy( _IQ(Ki_c) , err_iq ) - _IQmpy( _IQ(Ka_c) , excess_iq );
	return out_iq;
}

二、电流环PI应用于力矩模式

/****file:main.c*****/
void  HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
	if(htim == &htim4)
	{
		my_Driver_Contral();
	}

}
/****file:my_foc.c*****/
/**********************************************************************
 * 函数名称: my_Driver_Contral
 * 功能描述: 驱动控制运行函数
 * 输入参数: 无
 * 输出参数: 无
 * 返 回 值: 无
 * 修改日期        版本号     修改人        修改内容
 * -----------------------------------------------
 * 2024/07/05        V1.0     TONY0925       创建
 ***********************************************************************/
void my_Driver_Contral(void)
{
	//中断进来先改变占空比,此时占空比为上一周期计算得到的值
	__HAL_TIM_SetCompare(&htim1, TIM_CHANNEL_1, PWM1_PULSE);
	__HAL_TIM_SetCompare(&htim1, TIM_CHANNEL_2, PWM2_PULSE);
	__HAL_TIM_SetCompare(&htim1, TIM_CHANNEL_3, PWM3_PULSE);		
	//电流电压ADC采样与计算
	get_ADCvalue();
	//计算相电流、母线电压、母线电流值
	g_current.i_a = _IQmpy(_IQmpy(_IQ(ADC_Value[2]), _IQ(K_1)) - _IQ(1.65), _IQ(K_2)) -i_a_offect;
	g_current.i_b = _IQmpy(_IQmpy(_IQ(ADC_Value[3]), _IQ(K_1)) - _IQ(1.65), _IQ(K_2)) -i_b_offect;
	g_current.i_c = -g_current.i_a - g_current.i_b;
	g_current.i_dc_now = _IQmpy(_IQmpy(_IQ(ADC_Value[0]), _IQ(K_1)) - _IQ(1.65), _IQ(K_2))-i_dc_offect;
	g_voltage.u_dc_now = _IQmpy( _IQ(ADC_Value[1]) , _IQ(0.00793)) ;
	//CLARK、PARK变换
	g_current.i_alpha = _IQmpy((g_current.i_a - _IQmpy(_IQ(0.5), g_current.i_b) - _IQmpy(_IQ(0.5), g_current.i_c)), _IQ(K_park));
	g_current.i_beta = _IQmpy((_IQmpy(_IQ(sqrt3_2), g_current.i_b) - _IQmpy(_IQ(sqrt3_2), g_current.i_c)), _IQ(K_park));		
	g_current.i_d     = _IQmpy( _IQcos(g_theta.now_radian) , g_current.i_alpha ) +_IQmpy( _IQsin(g_theta.now_radian) , g_current.i_beta );
	g_current.i_q     = _IQmpy( -_IQsin(g_theta.now_radian) , g_current.i_alpha ) +_IQmpy( _IQcos(g_theta.now_radian) , g_current.i_beta );		
	//编码器获取角度
	g_theta.now_angle = spi_get_angle();
	g_theta.now_theta_M = _IQmpy( _IQ(g_theta.now_angle) , _IQ(0.02197266) );
	g_theta.now_theta_E = _IQfrac(_IQmpy( g_theta.now_theta_M , _IQ(0.01944444) ));
	g_theta.now_theta_E = _IQmpy(  g_theta.now_theta_E, _IQ(360) );		
	g_theta.now_radian = _IQmpy( g_theta.now_theta_E , _IQ(pi_180) );	
	//编码器获取速度
	get_speed();
	//转速PI控制器
	
	//电流PI控制器
	g_current.i_d_tar = _IQ(g_current.i_d_tar_f);
	g_current.i_q_tar = _IQ(g_current.i_q_tar_f);
	g_voltage.u_d=id_pi_contral(g_current.i_d,g_current.i_d_tar);
	g_voltage.u_q=iq_pi_contral(g_current.i_q,g_current.i_q_tar);
	//SVPWM
//	g_voltage.u_d = _IQ(0);
//	g_voltage.u_q = _IQ(3);
	g_voltage.u_alpha = _IQmpy( g_voltage.u_d , _IQcos(g_theta.now_radian) ) - _IQmpy( g_voltage.u_q , _IQsin(g_theta.now_radian) ) ;
	g_voltage.u_beta = _IQmpy( g_voltage.u_q , _IQcos(g_theta.now_radian) ) + _IQmpy( g_voltage.u_d , _IQsin(g_theta.now_radian) ) ;		
	iq_svpwm(g_voltage.u_alpha ,g_voltage.u_beta);
	//计算占空比值
	PWM1_PULSE = 400*_IQtoF(g_duty_ratio1);
	PWM2_PULSE = 400*_IQtoF(g_duty_ratio2);
	PWM3_PULSE = 400*_IQtoF(g_duty_ratio3);		
	//串口信息打印
	Sendtwodata(_IQtoF(g_current.i_q)*100+500,_IQtoF(g_current.i_d)*100+500);
	//Sendtwodata(_IQtoF(g_speed.Mnow_rpm_iq)*1,_IQtoF(g_theta.now_theta_E)*10);	
}

g_current.i_d_tar_f设定始终为0,通过改变g_current.i_q_tar_f即可控制力矩模式下的力矩/电流大小。其中g_current.i_q_tar_f为正,则电机正转;为负,则电机反转。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/780362.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

**kwargs 字典解包传参的方式

字典解包传参 在Python中&#xff0c;****kwargs**是一种通过字典解包 (dictionary unpacking) 的方式进行参数传递的方式。它将一个字典的键值对解包并传递给函数的命名参数。 示例代码 kwargs实参: {name: "jordan", age: 18, score: [80, 85, 85]} get_info形…

U盘非安全退出后的格式化危机与高效恢复策略

在数字化时代&#xff0c;U盘作为数据存储与传输的重要工具&#xff0c;其数据安全备受关注。然而&#xff0c;一个常见的操作失误——U盘没有安全退出便直接拔出&#xff0c;随后再插入时却遭遇“需要格式化”的提示&#xff0c;这不仅让用户措手不及&#xff0c;更可能意味着…

windows内置的hyper-v虚拟机的屏幕分辨率很低,怎么办?

# windows内置的hyper-v虚拟机的屏幕分辨率很低&#xff0c;怎么办&#xff1f; 只能这么大了&#xff0c;全屏也只是把字体拉伸而已。 不得不说&#xff0c;这个hyper-v做的很烂。 直接复制粘贴也做不到。 但有一个办法可以破解。 远程桌面。 我们可以在外面的windows系统&…

科普文:构建可扩展的微服务架构设计方案

前言 微服务架构是一种新兴的软件架构风格&#xff0c;它将单个应用程序拆分成多个小的服务&#xff0c;每个服务都运行在自己的进程中&#xff0c;这些服务通过网络进行通信。这种架构的优势在于它可以提高应用程序的可扩展性、可维护性和可靠性。 在传统的应用程序架构中&…

minist数据集分类模型的训练

minist数据集训练 训练方法&#xff1a;利用pytorch来实现minist数据集的分类模型训练 训练模型如下图所示 模型代码&#xff1a; import torch from torch import nn from torch.nn import Flattenclass Net(nn.Module):def __init__(self):super().__init__()self.module …

grid布局下的展开/收缩过渡效果【vue/已验证可正常运行】

代码来自GPT4o&#xff1a;国内官方直连GPT4o <template><div class"container"><button class"butns" click"toggleShowMore">{{ showAll ? 收回 : 显示更多 }}</button><transition-group name"slide-fade&…

KDP数据分析实战:从0到1完成数据实时采集处理到可视化

智领云自主研发的开源轻量级Kubernetes数据平台&#xff0c;即Kubernetes Data Platform (简称KDP)&#xff0c;能够为用户提供在Kubernetes上的一站式云原生数据集成与开发平台。在最新的v1.1.0版本中&#xff0c;用户可借助 KDP 平台上开箱即用的 Airflow、AirByte、Flink、K…

14-35 剑和诗人9 - 普及 Agentic RAG

好吧&#xff0c;让我们直接进入正题——了解 Agentic RAG&#xff08;检索增强生成&#xff09;方法以及它如何彻底改变我们处理信息的方式。系好安全带&#xff0c;因为这将变得疯狂&#xff01; Agentic RAG 的核心在于为 RAG 框架注入智能和自主性。这就像对常规 RAG 系统…

阶段三:项目开发---搭建项目前后端系统基础架构:任务10:SpringBoot框架的原理和使用

任务描述 1、熟悉SpringBoot框架的原理及使用 2、使用IDEA创建基于SpringBoot、MyBatis、MySQL的Java项目 3、当前任务请在client节点上进行 任务指导 1、SpringBoot框架的选择和原理 2、MyBatis-Plus的选择和原理 3、使用IDEA创建基于SpringBootMyBatis-PlusMySQL的Jav…

PCIe驱动开发(1)— 开发环境搭建

PCIe驱动开发&#xff08;1&#xff09;— 开发环境搭建 一、前言 二、Ubuntu安装 参考: VMware下Ubuntu18.04虚拟机的安装 三、QEMU安装 参考文章&#xff1a;QEMU搭建X86_64 Ubuntu虚拟系统环境 四、安装Ubuntu 下载地址&#xff1a;https://old-releases.ubuntu.com…

QWidget窗口抗锯齿圆角的一个实现方案(支持子控件)2

QWidget窗口抗锯齿圆角的一个实现方案&#xff08;支持子控件&#xff09;2 本方案使用了QGraphicsEffect&#xff0c;由于QGraphicsEffect对一些控件会有渲染问题&#xff0c;比如列表、表格等&#xff0c;所以暂时仅作为研究&#xff0c;优先其他方案 在之前的文章中&#…

k8s_集群搭建_在主节点中加入node节点_k8s集群自恢复能力演示_token过期重新生成令牌---分布式云原生部署架构搭建016

然后安装好了master节点以后,我们再来看如何把node节点加入进来,可以看到 只需要执行,命令行中提示的命令就可以了 比如上面的 Your Kubernetes control-plane has initialized successfully!To start using your cluster, you need to run the following as a regular user:…

人脸识别课堂签到系统【PyQt5实现】

人脸识别签到系统 1、运用场景 课堂签到,上班打卡,进出门身份验证。 2、功能类别 人脸录入,打卡签到,声音提醒,打卡信息导出,打包成exe可执行文件 3、技术栈 python3.8,sqlite3,opencv,face_recognition,PyQt5,csv 4、流程图 1、导入库 2、编写UI界面 3、打…

商家店铺电商小程序模板源码

橙色通用的商家入驻&#xff0c;商户商家&#xff0c;商家店铺&#xff0c;购物商城&#xff0c;商家购物平台app小程序网页模板。包含&#xff1a;商家主页、优先商家、商品详情、购物车、结算订单、个人中心、优惠券、会员卡、地址管理等功能页面。 商家店铺电商小程序模板源…

100359.统计X和Y频数相等的子矩阵数量

1.题目描述 给你一个二维字符矩阵 grid&#xff0c;其中 grid[i][j] 可能是 X、Y 或 .&#xff0c;返回满足以下条件的子矩阵数量&#xff1a; 包含 grid[0][0]X 和 Y 的频数相等。至少包含一个 X。 示例 1&#xff1a; 输入&#xff1a; grid [["X","Y",…

算法刷题笔记 滑动窗口(C++实现,非常详细)

文章目录 题目描述基本思路实现代码 题目描述 给定一个大小为n ≤ 10^6的数组。有一个大小为k的滑动窗口&#xff0c;它从数组的最左边移动到最右边。你只能在窗口中看到k个数字。每次滑动窗口向右移动一个位置。以下是一个例子&#xff1a; 该数组为 [1 3 -1 -3 5 3 6 7]&…

leetcode 66. 加一

leetcode 66. 加一 题解 刚开始只是以为在最后一位上加一就可以了 &#xff0c; 没想到还有进位呢&#xff0c; 比如说9的话&#xff0c; 加上1就是10&#xff0c; 返回的数组就是[1. 0],把进位的情况考虑进去就可以了。 class Solution { public:vector<int> plusOne(…

Vue3+.NET6前后端分离式管理后台实战(二十八)

1&#xff0c;Vue3.NET6前后端分离式管理后台实战(二十八)

Raw Socket(一)实现TCP三次握手

实验环境&#xff1a; Windows物理机&#xff1a;192.168.1.4 WSL Ubuntu 20.04.6 LTS&#xff1a;172.19.32.196 Windows下的一个http服务器&#xff1a;HFS&#xff0c;大概长这个样子&#xff1a; 客户端就是Ubuntu&#xff0c;服务端就是这个…

[图解]SysML和EA建模住宅安全系统-12-内部块图

1 00:00:00,580 --> 00:00:02,770 接下来我们来画流了 2 00:00:03,100 --> 00:00:05,050 首先第一个是站点状态 3 00:00:05,140 --> 00:00:08,130 从这里到这里&#xff0c;我们画一个过来 4 00:00:10,290 --> 00:00:11,890 这里流到这里 5 00:00:11,900 -->…