0

0

FMPy 中 FMU 单步仿真出现 NaN 的根本原因与专业解决方案

碧海醫心

碧海醫心

发布时间:2026-03-07 13:45:14

|

905人浏览过

|

来源于php中文网

原创

本文详解 FMPy 高频单步仿真(如 5e-7 s 步长)中 FMU 输出 NaN 的典型成因,指出 simulate_fmu() 封装逻辑引发的隐式状态干扰问题,并提供基于底层 API 的可控单步执行方案,附可复用代码与关键注意事项。

本文详解 fmpy 高频单步仿真(如 5e-7 s 步长)中 fmu 输出 nan 的典型成因,指出 `simulate_fmu()` 封装逻辑引发的隐式状态干扰问题,并提供基于底层 api 的可控单步执行方案,附可复用代码与关键注意事项。

在基于 FMPy 集成 Simcenter 导出 FMU 的实时协同仿真场景中(例如与 Simulink 构建反馈闭环),当采用极小步长(如 5×10⁻⁷ s)进行严格单步推进式仿真时,频繁观察到 FMU 输出在第 7–10 步后迅速发散并返回 NaN 值。该现象并非 FMU 模型本身数值不稳定所致——同一 FMU 在 Simulink 原生环境中运行完全正常,而一旦通过 FMPy 封装为 MATLAB 函数并部署至 Azure 容器,问题即刻复现。根本原因在于:simulate_fmu() 是一个面向“全周期批量仿真”设计的高层封装函数,其内部自动执行初始化、事件检测、步长自适应、状态重置与终止逻辑等操作。在高频、非连续、外部驱动的单步模式下,这些默认行为(尤其是重复调用 initialize() 或隐式修改 fmi2SetTime()/fmi2SetContinuousStates())极易破坏 FMU 内部连续状态的一致性,导致积分器失效或代数环求解崩溃,最终输出 NaN。

因此,正确路径是绕过 simulate_fmu(),直接使用 FMPy 的底层 FMU 实例 API 进行精细化控制。核心原则是:
仅执行必要动作:setReal() 输入 → doStep() 推进 → getReal() 读取输出;
严格管理仿真时间:手动维护 _current_simulation_time,避免 doStep() 参数与内部时钟冲突;
禁用所有自动生命周期干预:不调用 setupExperiment()、initialize()、terminate() 等,保持 FMU 实例长期驻留状态。

以下是一个生产就绪的单步执行函数示例(基于 custom_input.py 范式优化):

AskAI
AskAI

无代码AI模型构建器,可以快速微调GPT-3模型,创建聊天机器人

下载
def step_fmu(self, inputs_array_values, step_size=5e-7, interval=None):
    """
    执行 FMU 单步(或指定时间间隔)仿真,返回 (time, output1, output2, ...) 元组
    :param inputs_array_values: 当前输入值列表,顺序需与 self._vrs_inputs 严格一致
    :param step_size: 通信步长(单位:秒),必须与 FMU 声明的最小步长兼容
    :param interval: 实际推进的时间长度(默认等于 step_size),支持子步长对齐
    """
    if interval is None:
        interval = step_size

    stop_time = self._current_simulation_time + interval

    # 关键:在时间推进循环内精确控制每一步
    while self._current_simulation_time < stop_time:
        # 1. 设置输入变量(确保 value references 与输入值一一对应)
        self._fmu.setReal(list(self._vrs_inputs.values()), inputs_array_values)

        # 2. 执行单次通信步进(注意:currentCommunicationPoint 必须为当前时刻)
        status = self._fmu.doStep(
            currentCommunicationPoint=self._current_simulation_time,
            communicationStepSize=step_size
        )
        if status != 0:
            raise RuntimeError(f"FMI doStep failed with status {status}")

        # 3. 手动更新仿真时钟(非依赖 FMU 内部时钟)
        self._current_simulation_time += interval

    # 4. 读取当前时刻输出
    outputs = list(self._fmu.getReal(list(self._vrs_outputs.values())))
    # 将当前仿真时间作为首元素返回,便于 Simulink 同步
    result_row = [float(self._current_simulation_time)] + outputs
    self._results.append(str(tuple(result_row)))  # 可选:日志记录

    return tuple(result_row)

关键注意事项与最佳实践:

  • ? 输入/输出变量引用(VRs)必须预先解析并缓存:在 FMU 加载后一次性调用 model_description.modelVariables 获取 valueReference,避免每次 step_fmu() 中重复查找;
  • ? doStep() 的 communicationStepSize 应与 FMU 声明的 defaultStepSize 或 Simcenter 导出设置严格一致,不建议动态修改;
  • ? 禁止在单步循环中调用 reset() 或 freeInstance(),FMU 实例需全程存活以维持状态连续性;
  • ? Simulink 端务必使用 Unit Delay 模块打破代数环,确保 FMU 输出延迟一拍进入反馈回路——这与文中所述“1 step delay will be created in Simulink”设计完全吻合;
  • ? 调试建议:启用 FMPy 日志(logging.basicConfig(level=logging.DEBUG))并检查 fmi2DoStep 返回码,非零状态码(如 fmi2Error, fmi2Fatal)是定位底层 FMI 调用失败的直接依据。

通过该方案,用户不仅彻底消除了 NaN 问题,更获得了对仿真时序、状态流与错误处理的完全掌控力,为高精度、低延迟的跨工具链协同仿真奠定了坚实基础。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
fprintf在matlab中的用法
fprintf在matlab中的用法

fprintf是MATLAB中用于格式化输出的函数。fprintf的基本语法为“fprintf(fileID, format, A)”,其中,fileID是一个标识符,用于指定要写入的文件,如果要将数据写入到命令窗口中,则可以使用1作为fileID的值,format是一个字符串,用于指定输出的格式,A是要输出的数据。

498

2023.09.28

数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

1110

2023.10.12

Matlab中length函数的用法
Matlab中length函数的用法

在Matlab中,length函数用于返回向量、数组或字符串中的元素个数。想了解更多length函数的相关内容,可以阅读本专题下面的文章。

331

2023.11.22

Matlab中axis函数用法
Matlab中axis函数用法

在Matlab中,axis函数用于控制坐标轴的范围和比例。想了解更多axis函数的相关内容,可以阅读本专题下面的文章。

781

2023.11.23

subplot在matlab中的用法
subplot在matlab中的用法

subplot在matlab中用于在同一个图窗中创建多个子图。通过指定子图的行数、列数和当前绘图位置,可以在每个子图中绘制不同的图形。想了解更多subplot在matlab中的用法,可以访问下面的文章。

150

2023.11.27

scilab和matlab的区别
scilab和matlab的区别

scilab和matlab的区别:1、注释符号;2、预设变量的表示;3、操作符的用法;4、矩阵的定义与调用;5、程序的编辑与执行;6、数据类型;7、函数库;8、图形界面;9、社区支持与生态系统;10、跨平台兼容性;11、价格。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

193

2023.12.11

Matlab如何四舍五入
Matlab如何四舍五入

Matlab可以通过round函数和格式化输出函数来对数值来进行四舍五入操作。更多关于Matlab相关的问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

427

2023.12.12

Matlab中axis函数用法介绍
Matlab中axis函数用法介绍

在Matlab中,axis函数用于设置当前坐标轴的范围和刻度。想了解更多axis函数的相关内容,可以阅读本专题下面的文章。

163

2023.12.13

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号