【CFD小工坊】尝试完成一个简单的溃坝流算例(1)

news2024/11/17 23:24:31

【CFD小工坊】尝试完成一个简单的溃坝流算例(1)

  • 前言
  • 算例简介
  • 网格生成
  • 数据的读入与输出
    • 模型参数的读入
    • 网格数据及结果数据的输出

前言

我们从一个简单的算例开始,从实际建模过程中学习和做代码。我选择的算例是一个矩形区域内的溃坝流,其效果如下图所示。图片来自于Caleffi等人1的数值模拟研究。
在这里插入图片描述
如图所示,模型的计算区域为一个正方形,中间处有一个有缺口的大坝。在初始时刻,大坝的一侧水位很高,另一侧水位很低(甚至可能没有水);之后水流从坝口泄出,形成了溃坝流(dam break)。这个算例边界简单,常被用于测试模型对急变流、干湿边界处理等方面的模拟性能,是一个典型的测试算例。

我们的第一任务是实现模型的输入和输出。让模型能够读入我们的网格数据,并能按照一定格式输出模型的结果数据。

算例简介

本算例的计算区域是一个边长为200m的正方形区域,区域内有厚度为10m的大坝(位于 95 m < x < 105 m 95m<x<105m 95m<x<105m处),如下图所示。计算域为平底,底高程为0m,其四周及大坝所对应的壁面处均采用壁面边界条件(closed boundary)。
在这里插入图片描述
在初始时刻, x < 100 m x<100m x<100m区域的水位为10.0m,其余模型参数见下表。

工况编号上游水深下游水深曼宁系数 n n n模拟时长
A10.0m0m(或Hwet0.036.0s
B10.0m5.0m0.07.2s

网格生成

计算网格可通过Triangle库(Triangle库的使用方式可见《学习使用Triangle库》)、SMS或DHI-MIKE等软件辅助生成。生成后,需要通过处理得到两类数据:

  1. 网格节点位置;包括节点总数、编号及坐标值。(用于制作points.txt文件)
  2. 网格几何关系;包括网格编号、构成各个三角形网格的节点序号。(用于制作cells.txt文件)

上述数据文件的格式详见《【CFD小工坊】模型网格(三角形网格)》。(该博文已于2024年3月更新)

我制作的网格如下图所示。网格节点总数3412,三角形网格总数6537。
在这里插入图片描述
之后编辑得到points.txt文件和cells.txt文件,它们的格式如下:

  1. points.txt(第一行是节点总数,之后依次是1~3412号节点的x、y坐标,一行数据对应一个节点)
3412
0	0
0	200
60	95
60	170
95	0
95	95
95	170
95	200
...
  1. cells.txt(第一行是三角形网格总数,之后依次是每个网格所对应三个节点的节点号;一行数据对应一个三角形网格)
6537
218	991	992
866	867	728
646	2171	2172
1810	1744	1809
143	2379	140
2306	2303	2304
3113	421	2481
4	1123	2775
2254	302	3334
...

数据的读入与输出

首先,我们确定一个主程序代码的框架:

int main() {
	int i, j;

	// Read the meshgrid
	ReadGrid();
	// Construct the grid
	ConstructGrid();
	// Read the parameters and other data
	ReadInput();
	// Initialization
	//Initial();

	// output at the initial time
	output_plt(0);
	
	// Main Loop
	// Main Loop (not finished...)
	//while (time <= time_end) {
	//
	//}
	
	system("pause");	// just for test...
}

首先,调用函数ReadGrid()和ConstructGrid()来读入和构建网格。这两个程序的详细代码可见《【CFD小工坊】模型网格(三角形网格)》。关于网格的读入,不在此赘述了。
随后是通过函数ReadInput()读入一些模型参数,例如糙率、模拟时间等。
之后边是初始化程序并启动主循环while (time <= time_end) ,这一部分我们之后再实现。
此外,在运行前,我们试着通过函数output_plt( )输出一下读入的网格和初始场数据。当然,这个函数在后面输出模型结果的时候都会用到。

模型参数的读入

在模型中,参数文件是input.txt,即程序将读取我们写在input.txt中的数字,作为各个参数。

/*
* Function: ReadInput
* Usage: Read the data from the input files and other data files
* Called by: main
*/
void ReadInput() {
	int i, j;
	int bot_type;				// bot_type= 0: zb=constant; bot_type=1: read zb at grid points; bot_type=2: read zb at cell centers.
	int ini_eta;				// ini_eta = 0: eta=constant (eta00) initially; ini_eta = 1: read eta from a file (eta0.txt).
	int ini_vel;				// ini_vel = 0: u,v=zero initially; ini_vel = 1: read u and v from a file (vel0.txt).
	double bot0, eta00;

	// 1. time parameters
	time_str = get_double_val("TimeStr");
	time_end = get_double_val("TimeEnd");
	CFL = get_double_val("CFL");
	dt_min = get_double_val("DtMin");
	dt_max = get_double_val("DtMax");

	// 2. bottom elevation
	zbp = (double *)malloc(sizeof(double) * Np);
	zbc = (double *)malloc(sizeof(double) * Nc);
	bot_type = get_int_val("BotType");
	if (bot_type == 0) {
		// zb = constant
		bot0 = get_double_val("Bot0");
		for (i = 0; i < Np; i++) {
			zbp[i] = bot0;
		}
		for (i = 0; i < Nc; i++) {
			zbc[i] = bot0;
		}
	}
	else if (bot_type == 1) {
		// read zbp from zb.txt
		// ...
		}
	}
	else if (bot_type == 2) {
		// read zbc from zb.txt
		// ...
	}
	else {
		fprintf(stderr, "Please verify the value of BotType in input.txt. \n");
		exit(EXIT_FAILURE);
	}

	// 3. initial fields
	eta = (double *)malloc(sizeof(double) * Nc);
	u = (double *)malloc(sizeof(double) * Nc);
	v = (double *)malloc(sizeof(double) * Nc);
	char* row2 = (char*)malloc(sizeof(char) * N_str);
	// 3.1 initial surface elevation
	ini_eta = get_int_val("IniEta");
	if (ini_eta == 0) {
		eta00 = get_double_val("Eta0");
		for (i = 0; i < Nc; i++) {
			eta[i] = eta00;
		}
	}
	else if (ini_eta == 1) {
		// read water elevation from eta0.txt
		// ...
	}
	else {
		fprintf(stderr, "Please verify the value of Initial_Eta in input.txt. \n");
		exit(EXIT_FAILURE);
	}

	// 3.2 initial velocities
	ini_vel = get_int_val("IniVel");
	if (ini_vel == 0) {
		for (i = 0; i < Nc; i++) {
			u[i] = 0.0;
			v[i] = 0.0;
		}
	}
	else if (ini_vel == 1) {
		// read velocities from vel0.txt
		// ...
	}
	else {
		fprintf(stderr, "Please verify the value of Initial_Vel in input.txt. \n");
		exit(EXIT_FAILURE);
	}
	free(row2);

	// 4. open boundary
	// ...
	
	// 5. physical parameters
	vis0 = get_double_val("Viscosity");
	n_fric = get_double_val("Manning");

	// 6. numerical parameters
	DepDry = get_double_val("DepDry");
	DepWet = get_double_val("DepWet");
	theta_fric = get_double_val("ThetaFric");

	// 7. output setups
	out_str = get_double_val("OutStr");
	out_int = get_double_val("OutInt");

	fprintf(stderr, "The input.txt has been read... \n");
}

针对所需读入参数的类型,我将该函数段的代码分了七个部分。每个部分中用的读入方式有两种,第一种是通过get_val系列函数(这个系列函数会在之后的部分中讲解),它的功能就是从input.txt中读取所需名称的参数;第二种是读入一个初始场,如上述代码种的zb.txt,eta0.txt、vel0.txt。此外,我们还在这里设置了若干标识符,例如bot_type、ini_eta等,前者的意思是我们初始地形的设置方式,后者则是设置初始水位场的方式。
目前,我们暂不启用这些功能,它们有待我们的后续开发。目前,我们先定义我们的初始水位、流速都为0;其余参数暂时不读入。

网格数据及结果数据的输出

我们暂时采用tecplot作为后处理工具,故我们需要输出tecplot格式的结果数据文件和网格文件。代码如下:

/*
* Function: output_plt(count_p)
* Usage: output the results (water depth and velocities) as a tecplot file
* Called by: main
*/
void output_plt(int count_p) {
	int i;
	char filename[100];

	// 1. time
	FILE* fp1 = fopen(".\\result\\time.txt", "at");
	fprintf(fp1, "%d\t%lf \n", count_p, time);
	fclose(fp1);

	// 2. create a tecplot file (.plt)
	sprintf(filename, ".\\result\\%d.plt", count_p);
	FILE* fp2 = fopen(filename, "at");
	fprintf(fp2, "TITLE=\"Test\"\n");
	fprintf(fp2, "VARIABLES=\"X\",\"Y\",\"Zbot\",\"elev\",\"u\",\"v\"\n");
	fprintf(fp2, "ZONE T=\"ZONE1\",N=%d,E=%d,DATAPACKING=BLOCK,ZONETYPE=FETRIANGLE\n", Np, Nc);
	fprintf(fp2, "VARLOCATION=([3-6]=CELLCENTERED)\n");
	for (i = 0; i < Np; i++) {
		fprintf(fp2, "%f\n", xp[i]);
	}
	for (i = 0; i < Np; i++) {
		fprintf(fp2, "%f\n", yp[i]);
	}
	for (i = 0; i < Nc; i++) {
		fprintf(fp2, "%f\n", zbc[i]);
	}
	for (i = 0; i < Nc; i++) {
		fprintf(fp2, "%f\n", eta[i]);
	}
	for (i = 0; i < Nc; i++) {
		fprintf(fp2, "%f\n", u[i]);
	}
	for (i = 0; i < Nc; i++) {
		fprintf(fp2, "%f\n", v[i]);
	}
	for (i = 0; i < Nc; i++) {
		fprintf(fp2, "%d\t%d\t%d\n", node[0][i] + 1, node[1][i] + 1, node[2][i] + 1);
	}
	fclose(fp2);

	fprintf(stderr, "Output %d (Tecplot file) \n", count_p);
}

函数output_plt(count_p)自带参数count_p表示输出结果的次数,若是第一次输出结果,则count_p = 1;若是输出初始场,count_p = 0。之后我们需要在程序.exe同一目录下新建一个result文件夹,用于存放所有的结果文件。
第一类结果文件是时间序列,它记录了每一次输出结果是,对应的模拟时刻。结果存储在result文件夹下的time.txt里。第二类数据是tecplot数据.plt,它需要按照一定的格式,依次输入节点坐标、网格中心的水位、流速等等,最后还要附上网格的几何数据。

通过运行整个模型,我们得到的输出结果如下。我们在初始化时,预设了0的水位场和流速场,因此云图显示全局数据为0。
在这里插入图片描述
至此,我们初步实现了网格数据、模型参数读取和结果数据输出的功能!


  1. Valerio Caleffi , Alessandro Valiani & Andrea Zanni (2003) Finite volume method for simulating extreme flood events in natural channels, Journal of Hydraulic Research, 41:2, 167-177, DOI: 10.1080/00221680309499959 ↩︎

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ES入门六:Suggesters Api实践

都是负担在很多app上&#xff0c;当我们输入某些内容时候&#xff0c;它会立即做一些补全操作&#xff0c;如果我想实现上述的需求&#xff0c;我们就可以使用ES提供的Suggesters Api。那Suggesters是如何做到的那&#xff1f;简单来说&#xff0c;Suggesters会将输入的文本拆分…

每日一练:LeeCode-707. 设计链表 【链表+虚拟头结点+设计】

每日一练&#xff1a;LeeCode-707. 设计链表 【链表虚拟头结点设计】 思路设置虚拟头节点 本文是力扣 每日一练&#xff1a;LeeCode-707. 设计链表 【链表虚拟头结点设计】 学习与理解过程&#xff0c;本文仅做学习之用&#xff0c;对本题感兴趣的小伙伴可以出门左拐LeeCode-70…

Rabbitmq消息丢失-消费者消息丢失(二)

说明&#xff1a;消费端在处理消息的过程中出现异常&#xff0c;例如&#xff1a;业务逻辑异常&#xff0c;或者消费者被停机&#xff0c;或者网络断开连接等&#xff0c;以上等情况使消息没有得到正确恰当的处理&#xff0c;也会使消息丢失。 分析&#xff1a;分析就是说明中…

中科数安|防止电脑文件资料外泄

#防止电脑文件资料泄漏# 中科数安提供了一系列解决方案来防止电脑文件资料外泄。 www.weaem.com 这些解决方案包括以下几个方面&#xff1a; 访问控制&#xff1a;实施严格的文件访问控制&#xff0c;确保只有授权的人员能够访问和编辑核心文件。使用身份验证和权限管理系统&a…

Android APK包反编译为java文件教程

方法 流程&#xff1a; test.apk -> smali文件 -> dex文件 -> jar文件 ->java 文件 将APK包解压为 smail文件 下载 apktool工具 apktool.jar 将 test.apk 和 apktool.jar放同一目录下&#xff0c;并执行以下命令 java -jar apktool.jar d -f xxx.apk -o xxx(解…

30、类和接口

文章目录 接口概念接口和类之间有何关系&#xff1f; 可以使用接口来约束类接口继承接口接口还可以继承类接口为什么可以继承类内层原因&#xff1a;接口为什么可以继承类 用得出的结论解释最初的demo接口继承类的一些限制 接口概念 接口&#xff08;Interfaces&#xff09;可…

SAP PP学习笔记 - 豆知识07 - 如何查看BOM一览

SAP标准提供了CS03&#xff0c;只能查询单个的BOM&#xff0c;如果想查看一览&#xff0c;只能自己写SQVI 查询。 有其他高招的童鞋&#xff0c;请赐教啊。 1&#xff0c;SQVI 工具 SAP MM学习笔记18- SQVI 工具_sap sqvi-CSDN博客 输入查询名&#xff0c;然后点击 登录 2&a…

C++学习笔记:set和map

set和map set什么是setset的使用 关联式容器键值对 map什么是mapmap的使用map的插入方式常用功能map[] 的灵活使用 set 什么是set set是STL中一个底层为二叉搜索树来实现的容器 若要使用set需要包含头文件 #include<set>set中的元素具有唯一性(因此可以用set去重)若用…

Linux高级编程:进程间的通信(二)、IPC

回顾 共7种方式&#xff1a; 古老的进程间通信方式&#xff1a; 管道&#xff1a; 无名管道 有名管道 信号 系统V IPC进程对象 共享内存 消息队列 信号量集 socket通信 //网络 ------------------------- 无名管道 pipe&#xff08;&#xff09; 特点&#xff1a; 用于…

CSS3笔记

1.相同优先级的样式以写在后面的为主。 2.交集选择器&#xff0c;并且 条件挨在一起 p.rich{...} /*p元素class有rich的元素*/ 3.并集选择器&#xff0c;或者 逗号隔开 .class1,class2{...}/*满足其中一个类名都会使用该样式*/ 4.后代选择器 空格 隔开 所有符合的包括孙子及…

揭秘App访问量背后的秘密:数据统计与分析

在移动互联网时代&#xff0c;App已成为人们日常生活的重要组成部分。对于App运营者来说&#xff0c;了解用户的访问量、行为习惯等数据至关重要。本文将深入探讨如何精准统计App访问量&#xff0c;为运营者提供有价值的数据支持。 一、App访问量统计的重要性 访问量是衡量A…

2024年【焊工(初级)】找解析及焊工(初级)考试技巧

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 焊工&#xff08;初级&#xff09;找解析是安全生产模拟考试一点通生成的&#xff0c;焊工&#xff08;初级&#xff09;证模拟考试题库是根据焊工&#xff08;初级&#xff09;最新版教材汇编出焊工&#xff08;初级…

【EAI 027】Learning Interactive Real-World Simulators

Paper Card 论文标题&#xff1a;Learning Interactive Real-World Simulators 论文作者&#xff1a;Mengjiao Yang, Yilun Du, Kamyar Ghasemipour, Jonathan Tompson, Leslie Kaelbling, Dale Schuurmans, Pieter Abbeel 作者单位&#xff1a;UC Berkeley, Google DeepMind, …

leetcode 移除链表元素

本题中&#xff0c;我们是要移除链表的某一个节点&#xff0c;为了确保统一操作&#xff0c;我们需要使用虚拟头节点&#xff0c;这样我们删除节点的时候&#xff0c;就是把这个要删除的节点&#xff08;当前节点cur&#xff09;的前一个节点pre&#xff0c;使得pre.next指向要…

jsp阿帕奇安装教程

1.将压缩包解压&#xff0c;存放在自己所知道的位置 2.将软件文件夹打开 使用winr &#xff0c;输入cmd运行打开 输入Java或者Javac&#xff0c;出现一大串之后表明成功 接着在所解压的软件中点开bin这个文件夹&#xff0c;找到startup.bat点击 点击之后会出现黑框&#xff0c…

Linux中断实验:定时器实现按键消抖处理

一. 简介 前面文章学习了Linux驱动按键中断实验,文章地址如下: Linux驱动按键中断实验:按键中断功能的实现-CSDN博客 本文在Linux驱动按键中断实现的基础上,使用定时器实现按键消抖处理。 二. Linux中断实验:定时器实现按键消抖处理 1. 定时器处理按键消抖的原理 按…

java使用poi、ftl将html导出word,设置视图为 页面视图

1、修改html标签&#xff0c; 加入如下内容 <html xmlns:v"urn:schemas-microsoft-com:vml" xmlns:o"urn:schemas-microsoft-com:office:office" xmlns:w"urn:schemas-microsoft-com:office:word" xmlns:m"http://schemas.microsoft.c…

【NCRE Python】

基本输入输出函数 input 函数 input 函数用于从标准输入&#xff08;如键盘&#xff09;接收用户输入的字符串。当 input 函数被调用时&#xff0c;程序会暂停执行&#xff0c;等待用户输入文本并按回车键。用户输入的文本会作为字符串返回给程序。input 函数还可以接收一个字…

【笔记版】docker常用指令---systemctl类、docker状态

systemctl [options] docker 启动&#xff1a;system start docker查看状态&#xff1a;systemctl status docker停止&#xff1a;systemctl stop docker有警告&#xff1a;service关闭了&#xff0c;但是docker.socket仍响应解决方法&#xff1a;systemctl stop docker.socket…