基于ekf的永磁同步电机电阻、电感、磁链的辨识,辨识算法基于S- function编写
永磁同步电机参数辨识这个坑,我去年踩得够呛。传统离线辨识方法动不动就要停机做实验,在线辨识里EKF(扩展卡尔曼滤波)算是比较优雅的方案。这次咱们聊聊怎么用S-function把电阻、电感、磁链这三个关键参数给揪出来。
先上点硬货,S-function的架子得搭起来。在Simulink里新建个Level-2 M文件S函数,核心逻辑在Update方法里折腾。状态变量选得讲究点,电机电流、转速这些观测值不用说,重点是把待辨识的R、L、ψ作为扩展状态塞进去:
function Update(block) x_hat = block.Dwork(1).Data; % 状态向量 [id; iq; R; L; psi] P = block.Dwork(2).Data; % 协方差矩阵 u = block.InputPort(1).Data; % 输入电压 y = block.InputPort(2).Data; % 实测电流 % 参数 Ts = 1e-4; % 别超过控制周期 Q = diag([1e-4, 1e-4, 1e-6, 1e-8, 1e-8]); R_noise = diag([0.1, 0.1]); % EKF预测步 [F, G] = jacobian(x_hat, u); x_pred = x_hat + Ts * state_eq(x_hat, u); P_pred = F * P * F' + G * Q * G'; % 更新步 H = observation_jacobian(x_hat); K = P_pred * H' / (H * P_pred * H' + R_noise); x_hat = x_pred + K * (y - observe(x_hat)); P = (eye(5) - K*H) * P_pred; % 参数约束 x_hat(3) = max(min(x_hat(3), 2), 0.1); % R限制在0.1~2欧 x_hat(4) = max(min(x_hat(4), 0.01), 0.001); % L限制在1~10mH block.Dwork(1).Data = x_hat; block.Dwork(2).Data = P; end这段代码里有几个魔鬼细节:首先是状态方程和观测方程的雅可比矩阵计算,这玩意儿相当于EKF的导航仪。比如状态方程的偏导数矩阵F,得按照电机模型来:
function [F, G] = jacobian(x, u) R = x(3); L = x(4); psi = x(5); F = [ -R/L, 0, -x(1)/L, (R*x(1)-u(1))/L^2, 0; 0, -R/L, -x(2)/L, (R*x(2)-u(2))/L^2, -x(5)/L; zeros(3,5) ]; G = eye(5); % 过程噪声耦合矩阵 end注意第三行那些zeros(3,5),这里隐含了一个重要设定——假设参数变化是随机游走过程。也就是说,我们认为R、L、ψ这些参数在短时间内变化缓慢,用随机过程来建模。这种处理虽然不严格,但实操中够用了。
调参老司机都知道,协方差矩阵Q和R的设置直接决定收敛性能。我的经验是先用仿真数据标定个大概:给参数真值加个10%的扰动作为初始值,Q对角元素从1e-4开始试,R根据传感器精度来定。遇到过参数震荡的情况?试着把Q(3:5)再调小一个量级,相当于降低参数变化的"灵敏度"。
最后在Simulink里搭测试模型时,记得把S-function的采样时间设为固定步长,最好跟控制周期对齐。遇到过不收敛的坑吗?检查下电机模型里的转速项有没有正确处理,磁链参数ψ的初始值别设成0——那会直接让雅可比矩阵出现除零错误。
参数辨识结果出来之后,别急着欢呼,拿负载突变的数据验证下。我碰到过电感辨识值在突加负载时飘移的情况,后来发现是状态方程里漏了转速耦合项。说到底,EKF就是个带着数学约束的滤波器,模型准确性才是它发挥作用的根基。