网易首页 > 网易号 > 正文 申请入驻

OpenFOAM和Python的流场动态模态分解:从数据提取到POD-DMD分析

0
分享至

本文探讨了Python脚本与动态模态分解(DMD)的结合应用。我们将利用Python对从OpenFOAM模拟中提取的二维切片数据进行DMD计算。这种方法能够有效地提取隐藏的流动模式,深化对流体动力学现象的理解。

使用开源CFD软件OpenFOAM,有两种方法可以对CFD数据进行DMD计算。第一种方法是直接将OpenFOAM的场数据读入Python;第二种方法则是从OpenFOAM中提取二维切片,然后对这些数据进行DMD计算。

本文将重点介绍第二种方法,即利用Python的强大库直接分析从OpenFOAM提取的二维切片数据,执行DMD并可视化提取的模态。

OpenFOAM案例模态分解准备指南

本研究的起点是雷诺数为100的方形圆柱周围完全发展的、统计稳定的流动。在此基础上,我们将模拟时间延长至100个涡脱落周期。在每个脱落周期内,我们从数据中提取16次二维切片。二维切片的提取是通过OpenFOAM中的surfaces函数对象实现的,具体配置如下:

surfaces
{
type surfaces;
libs ("libsampling.so");
writeControl timeStep;
writeInterval 142;
surfaceFormat vtk;
fields (p U);
interpolationScheme cellPoint;
surfaces
{
zNormal
{
type cuttingPlane;
point (0 0 0.05);
normal (0 0 1);
interpolate true;
}
};
};
// ************************************************************************* //

模拟完成后,在案例的postProcessing目录中会生成一个名为surfaces的子目录,其中包含所有提取的表面数据。目录结构如下:

surfaces/
├── 4771.2000000577236
│ └── zNormal.vtp
├── 4772.6200000577546
│ └── zNormal.vtp
├── 4774.0400000577856
│ └── zNormal.vtp
├── 4775.4600000578166
│ └── zNormal.vtp
.
.
.

在进行后续分析之前,请确保案例模拟已完成且表面数据已成功提取。

表面数据提取

为了从OpenFOAM生成的VTK文件中提取数据,我们将使用PyVista库。PyVista是可视化工具包(VTK)的Python接口,通过NumPy包装VTK库,提供了直接访问数组的方法和类。它为VTK的强大可视化后端提供了一个文档完善的Pythonic接口,便于快速原型设计、分析和空间参考数据集的可视化集成。

PyVista在科学计算可视化中具有重要价值,尤其适用于演示和研究论文的图形生成。同时它也作为其他依赖3D网格渲染的Python模块的支持库。

导入必要的模块,包括PyVista:

import matplotlib.colors
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
import fluidfoam as fl
import scipy as sp
import os
import matplotlib.animation as animation
import pyvista as pv
import imageio
import io
%matplotlib inline
plt.rcParams.update({'font.size' : 18, 'font.family' : 'Times New Roman', "text.usetex": True})

接下来设置路径变量和常量:

### 常量
d = 0.1
Ub = 0.015
### 路径
Path = 'E:/deephub/Sq_Cyl_Surfaces/surfaces/'
save_path = 'E:/deephub/SquareCylinderData/'
Files = os.listdir(Path)

现在可以尝试读取第一个快照表面:

Data = pv.read(Path + Files[0] + '/zNormal.vtp')
grid = Data.points
x = grid[:,0]
y = grid[:,1]
z = grid[:,2]
rows, columns = np.shape(grid)
print('rows = ', rows, 'columns = ', columns)
print(Data.array_names)

输出:

['TimeValue', 'p', 'U']

从输出可以看出,我们的二维切片包含了时间值、压力场和速度场。利用PyVista,可以为每个快照提取涡量场,并将结果数据组织成一个大型矩阵,以便进行后续的POD计算。具体实现如下:

Data = pv.read(Path + Files[0] + '/zNormal.vtp')
grid = Data.points
x = grid[:,0]
y = grid[:,1]
z = grid[:,2]
rows, columns = np.shape(grid)
print('rows = ', rows, 'columns = ', columns)
### 对U场进行处理
Snaps = len(Files) # 快照数量
data_Vort = np.zeros((rows,Snaps-1))
for i in np.arange(0,Snaps-1):
data = pv.read(Path + Files[i] + '/zNormal.vtp')
gradData = data.compute_derivative('U', vorticity=True)
grad_pyvis = gradData.point_data['vorticity']
data_Vort[:,i:i+1] = np.reshape(grad_pyvis[:,2], (rows,1), order='F')
np.save(save_path + 'VortZ.npy', data_Vort)

让我们检查一下生成的data_Vort数组的维度:

data_Vort.shape
### 输出
### (96624, 1600)

此外,我们可以可视化涡量场的一个快照:

这个可视化结果展示了方形圆柱周围的涡量分布,为我们提供了流场结构的直观认识。

正交分解(POD)

为了确定动态模态分解(DMD)的最佳近似秩,我们可以对涡量场数据进行正交分解(POD)分析。POD是一种强大的降维技术,能够捕捉流场中的主要能量结构。

以下是POD分析的Python实现:

### POD分析
# 构建数据矩阵
X = data_Vort
# 计算并去除平均场
X_mean = np.mean(X, axis = 1)
Y = X - X_mean[:,np.newaxis]
# 计算协方差矩阵
C = np.dot(Y.T, Y)/(Y.shape[1]-1)
# 对协方差矩阵进行奇异值分解
U, S, V = np.linalg.svd(C)
# 计算POD模态
Phi_POD = np.dot(Y, U)
# 计算时间系数
a = np.dot(Phi_POD.T, Y)

接下来可以分析POD特征值以评估各模态的能量贡献:

Energy = np.zeros((len(S),1))
for i in np.arange(0,len(S)):
Energy[i] = S[i]/np.sum(S)
X_Axis = np.arange(Energy.shape[0])
heights = Energy[:,0]
fig, axes = plt.subplots(1, 2, figsize = (12,4))
ax = axes[0]
ax.plot(Energy, marker = 'o', markerfacecolor = 'none', markeredgecolor = 'k', ls='-', color = 'k')
ax.set_xlim(0, 20)
ax.set_xlabel('Modes')
ax.set_ylabel('Energy Content')
ax = axes[1]
cumulative = np.cumsum(S)/np.sum(S)
ax.plot(cumulative, marker = 'o', markerfacecolor = 'none', markeredgecolor = 'k', ls='-', color = 'k')
ax.set_xlabel('Modes')
ax.set_ylabel('Cumulative Energy')
ax.set_xlim(0, 20)
plt.show()

分析结果显示,前21个POD模态捕捉了约99.9%的总能量。这一发现为我们后面选择DMD的近似秩提供了重要依据,表明使用21阶近似进行DMD分析是合理的。

以下是前几个POD模态的可视化结果,用于参考:

这些模态图展示了流场中的主要结构,为我们理解流动特性提供了直观的洞察。

动态模态分解(DMD)

动态模态分解是一种强大的技术,能够提取流场中的动态特征。以下是DMD算法的Python实现:

def DMD(X1, X2, r, dt):
# 对X1进行奇异值分解
U, s, Vh = np.linalg.svd(X1, full_matrices=False)
# 截断SVD矩阵
Ur = U[:, :r]
Sr = np.diag(s[:r])
Vr = Vh.conj().T[:, :r]
# 构建Atilde矩阵并计算其特征值和特征向量
Atilde = Ur.conj().T @ X2 @ Vr @ np.linalg.inv(Sr)
Lambda, W = np.linalg.eig(Atilde)
# 计算DMD模态
Phi = X2 @ Vr @ np.linalg.inv(Sr) @ W
# 计算连续时间特征值
omega = np.log(Lambda)/dt
# 计算DMD模态振幅
alpha1 = np.linalg.lstsq(Phi, X1[:, 0], rcond=None)[0]
b = np.linalg.lstsq(Phi, X2[:, 0], rcond=None)[0]
# DMD重构
time_dynamics = None
for i in range(X1.shape[1]):
v = np.array(alpha1)[:,0]*np.exp( np.array(omega)*(i+1)*dt)
if time_dynamics is None:
time_dynamics = v
else:
time_dynamics = np.vstack((time_dynamics, v))
X_dmd = np.dot(np.array(Phi), time_dynamics.T)
return Phi, omega, Lambda, alpha1, b, X_dmd

为了应用这个DMD函数,我们首先需要准备时间偏移的数据矩阵:

# 获取数据矩阵的两个时间步长偏移视图
X1 = np.matrix(X[:, 0:-1])
X2 = np.matrix(X[:, 1:])

然后,我们定义近似秩和时间步长:

r = 21 # 根据POD分析结果选择
dt = 0.01*142

接下来,我们执行DMD计算:

Phi, omega, Lambda, alpha1, b, X_dmd = DMD(X1, X2, r, dt)

在进行可视化之前,我们首先分析DMD特征值的分布。这有助于我们理解所识别的DMD模态的动态特性。我们将实部和虚部特征值绘制在单位圆上:

theta = np.linspace(0, 2 * np.pi, 150)
radius = 1
a = radius * np.cos(theta)
b = radius * np.sin(theta)
fig, ax = plt.subplots()
ax.scatter(np.real(Lambda), np.imag(Lambda), color = 'r', marker = 'o', s = 100)
ax.plot(a, b, color = 'k', ls = '--')
ax.set_xlabel(r'$\Lambda_r$')
ax.set_ylabel(r'$\Lambda_i$')
ax.set_aspect('equal')
plt.show()

这个图显示所有特征值都位于单位圆上,表明DMD模态既不增长也不衰减,呈现稳定的特性。

为了可视化DMD模态,我们首先需要将DMD模态矩阵转换为数组:

A = np.squeeze(np.asarray(Phi))

然后可以使用Matplotlib绘制DMD模态:

Rect1 = plt.Rectangle((-0.5, -0.5), 1, 1, ec='k', color='white', zorder=2)
Mode = 11
fig, ax = plt.subplots(figsize=(11, 4))
p = ax.tricontourf(x/0.1, y/0.1, np.real(A[:,Mode]), levels = 1001, vmin=-0.005, vmax=0.005, cmap = cmap)
ax.add_patch(Rect1)
ax.xaxis.set_tick_params(direction='in', which='both')
ax.yaxis.set_tick_params(direction='in', which='both')
ax.xaxis.set_ticks_position('both')
ax.yaxis.set_ticks_position('both')
ax.set_xlim(-1, 20)
ax.set_ylim(-5, 5)
ax.set_aspect('equal')
ax.set_xlabel(r'$\bf x/d$')
ax.set_ylabel(r'$\bf y/d$')
ax.text(0, 4, r'$f_i =' + str(np.imag(Lambda[Mode])) + '$', fontsize = 25, color = 'black')
plt.show()

这个图展示了第11个DMD模态的空间结构。类似地,我们可以绘制前6个DMD模态:

这些DMD模态图揭示了流场中的关键动态结构,为我们深入理解方形圆柱周围的流动特性提供了重要依据。

通过结合POD和DMD分析,我们不仅捕捉了流场的主要能量结构,还揭示了这些结构随时间的演化特性。这种综合分析方法为复杂流动系统的研究提供了强大的工具,能够帮助我们更深入地理解流体动力学现象。

总结

本文详细介绍了一种基于OpenFOAM和Python的流场动态分析方法。我们从OpenFOAM模拟数据的提取和处理开始,利用PyVista库高效地处理二维切片数据。通过正交分解(POD)成功捕捉了流场的主要能量结构,为动态模态分解(DMD)的应用奠定了基础。DMD分析进一步揭示了流场的动态特征,使我们能够深入理解方形圆柱周围的复杂流动现象。

这种结合OpenFOAM、POD和DMD的综合分析方法,不仅提高了对复杂流体系统的认识,还为流体动力学研究提供了强大的工具。Python的灵活性和效率在整个分析过程中发挥了关键作用,展示了其在科学计算和数据可视化方面的优势。

https://avoid.overfit.cn/post/7d6faa4f21244df0ac7ed62f9833acd2

作者:Shubham Goswami

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
亲女儿去世才两周,泰王却急忙飞法国,下飞机头等大事竟是... 买私人飞机?!

亲女儿去世才两周,泰王却急忙飞法国,下飞机头等大事竟是... 买私人飞机?!

英国那些事儿
2026-07-01 02:10:13
韩国球迷怒骂滚出去!洪明甫被护送出境,入境大厅仅停留2分钟

韩国球迷怒骂滚出去!洪明甫被护送出境,入境大厅仅停留2分钟

史智文道
2026-07-01 17:20:21
“虚荣,就是这种面相”,家长用奶茶袋包大疆,初三女儿当场变脸

“虚荣,就是这种面相”,家长用奶茶袋包大疆,初三女儿当场变脸

妍妍教育日记
2026-07-01 20:35:03
周鸿祎:中国必须拥有自己的Mythos

周鸿祎:中国必须拥有自己的Mythos

看看新闻Knews
2026-07-01 10:48:03
韩国球迷抗议,让中国承担他们世界杯费用,是中国国足坑了他们

韩国球迷抗议,让中国承担他们世界杯费用,是中国国足坑了他们

流史岁月
2026-07-01 13:35:03
NBA重磅签约!火箭第二签!2年1300万拿下防守悍将,血赚!

NBA重磅签约!火箭第二签!2年1300万拿下防守悍将,血赚!

顺静自然
2026-07-02 00:26:44
1968年,上万名中国官兵秘密潜入巴基斯坦,沿路堆放着一口口棺材

1968年,上万名中国官兵秘密潜入巴基斯坦,沿路堆放着一口口棺材

文史达观
2026-07-01 22:20:17
网友用X Money给马斯克转25美元:获本人回应后全网跟风打钱

网友用X Money给马斯克转25美元:获本人回应后全网跟风打钱

快科技
2026-06-29 11:37:24
苏州观前街大火:一场浓烟,烧醒了多少“安全装睡”的人?

苏州观前街大火:一场浓烟,烧醒了多少“安全装睡”的人?

娱乐圈见解说
2026-07-01 20:45:11
法国队内讧?9000万新星连续4场替补,赛后对德尚甩脸,拒绝握手

法国队内讧?9000万新星连续4场替补,赛后对德尚甩脸,拒绝握手

小火箭爱体育
2026-07-01 14:19:41
央视罕见公开西太对峙细节!日舰模拟攻击辽宁舰,结果沉默?

央视罕见公开西太对峙细节!日舰模拟攻击辽宁舰,结果沉默?

青青衫书生
2026-06-30 13:24:21
油桃被点名!调查发现:糖尿病患者常吃油桃,很快或迎来6个后果

油桃被点名!调查发现:糖尿病患者常吃油桃,很快或迎来6个后果

健康之光
2026-07-01 13:04:58
如果人到了老年还好色,足以证明这人有极强的生命力

如果人到了老年还好色,足以证明这人有极强的生命力

喵咪文化
2026-06-21 20:15:06
南部战区公布驱离菲律宾飞机过程,空军飞行员对讲记录公布:在自家院子转, 紧张个啥?!

南部战区公布驱离菲律宾飞机过程,空军飞行员对讲记录公布:在自家院子转, 紧张个啥?!

每日经济新闻
2026-07-01 08:29:36
一个人到了三四十岁,如果10万块存款都没有,说明已经沦落到底层

一个人到了三四十岁,如果10万块存款都没有,说明已经沦落到底层

舒山有鹿
2026-05-20 00:00:19
始料未及!针对哈里梅根入住皇家庄园,查尔斯方回应“从未邀请”

始料未及!针对哈里梅根入住皇家庄园,查尔斯方回应“从未邀请”

聪明的橙子hj
2026-06-30 16:49:18
赵嘉仁离队后!广厦超市开张,朱俊龙布朗被疯抢,广东要胡金秋?

赵嘉仁离队后!广厦超市开张,朱俊龙布朗被疯抢,广东要胡金秋?

绯雨儿
2026-07-01 14:34:24
“高考估分715查分299,女孩称试卷不是自己的”,绵阳市教体局:纯属谣言,查无此人

“高考估分715查分299,女孩称试卷不是自己的”,绵阳市教体局:纯属谣言,查无此人

台州交通广播
2026-07-01 16:04:04
蒸发6000亿,马斯克神话破灭,华尔街撤资,孙正义发声炮轰AI骗局

蒸发6000亿,马斯克神话破灭,华尔街撤资,孙正义发声炮轰AI骗局

古史青云啊
2026-07-01 14:50:27
前沿物理学重大验证:爱因斯坦又说对!引力传播速度几乎等于光速

前沿物理学重大验证:爱因斯坦又说对!引力传播速度几乎等于光速

聪先生
2026-06-30 13:42:32
2026-07-02 01:04:49
deephub incentive-icons
deephub
CV NLP和数据挖掘知识
2023文章数 1465关注度
往期回顾 全部

科技要闻

Claude Code被曝“植入木马”识别中国用户

头条要闻

六旬父亲背8个土鸡蛋接考后续:儿子报考大学已确定

头条要闻

六旬父亲背8个土鸡蛋接考后续:儿子报考大学已确定

体育要闻

卖球衣救子的门将,把德国扑出了世界杯

娱乐要闻

77岁牛群公证裸捐全部财产,清贫独居坚持月捐

财经要闻

新氧贷款:宣传年化15%,实际顶格24%

汽车要闻

同比暴涨188.4% 方程豹6月热销35607台

态度原创

旅游
艺术
本地
数码
公开课

旅游要闻

距离迪士尼不到2公里,藏着浦东的百年水乡,很多人却不知道

艺术要闻

西安美术学院,2026届油画系硕士研究生毕业作品选(二)

本地新闻

强烈建议,全国高校都向这所大学看齐!

数码要闻

华硕a豆高速固态U盘上架:280-959元

公开课

李玫瑾:为什么性格比能力更重要?

无障碍浏览 进入关怀版