电磁波时域有限差分方法(FDTD)-C语言-FDTD和激励源和MUR边界

2023-11-07 21:30

本文主要是介绍电磁波时域有限差分方法(FDTD)-C语言-FDTD和激励源和MUR边界,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

参考:

葛德彪, 闫玉波. 电磁波时域有限差分方法[M]. 西安电子科技大学出版社, 2011.

Schneider J B. Understanding the Finite-Difference Time-Domain Method[J]. 2013.


最近要用FDTD做仿真,所以找了以上两本书,FDTD确实是有点难的。

想了想流程:

1:FDTD

2:激励源

3:吸收边界

4:散射


现在完成了FDTD和激励源,的TM部分,虽然都是抄书的,但对于一个不懂电磁波电磁场,数学极差的我来说,光是看懂代码就已经竭尽全力了。

何况还遇到了C语言宏定义的坑。

然后做了一部分的MUR吸收,效果不咋的,不知道就是这样的还是我代码有问题。


上代码,基本就是抄书的,但是简洁了一点。。

就不先做注释了,等到最后全部写完再做注释。

// Boundary.cpp: 定义控制台应用程序的入口点。
//#include "stdafx.h"
#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#define _CRT_SECURE_NO_WARNINGS#define ALLOC_2D(PNTR, NUMX, NUMY, TYPE)					\PNTR = (TYPE *)calloc((NUMX) * (NUMY), sizeof(TYPE));	\if (!PNTR) {												\perror("ALLOC_2D");									\fprintf(stderr, "Allocation failed for" #PNTR ". Terminating.\n");	\exit(-1);											\}
#define SizeX g->sizeX
#define SizeY g->sizeY
#define CDTDS g->cdtds
#define Time g->time
#define MaxTime g->maxTime#define Ez(M, N) g->ez[M * SizeY + N]
#define CA(M, N) g->ceze[M * SizeY + N]
#define CB(M, N) g->cezh[M * SizeY + N]#define Hx(M, N) g->hx[M * (SizeY) + N]
#define CP(M, N) g->chxh[M * (SizeY) + N]
#define CQ(M, N) g->chxe[M * (SizeY) + N]#define Hy(M, N) g->hy[M * SizeY + N]
//#define Chyh(M, N) g->chyh[M * SizeY + N]
//#define Chye(M, N) g->chye[M * SizeY + N]struct Grid {double *hx, *chxh, *chxe;double *hy, *chyh, *chye;double *ez, *ceze, *cezh;int sizeX, sizeY;int time, maxTime;int type;double cdtds;	//库朗数
};
typedef struct Grid Grid;void gridInit(Grid *g);
double driveSource(Grid *g, double time, double locX, double locY);
void writeFile(Grid *g);
void updateE(Grid *g);
void updateH(Grid *g);
void updateBoundary(Grid *g);
void EHInit(Grid *g);float c = 3e8;
float PI = 3.1415926535;
float MU0 = PI * 4e-7;
float EPS0 = (1 / (36 * PI)) * 1e-9;
float SIGMA0 = 0;
float SIGMAm0 = 0;
float DELTA = 5e-3;
float DELTAT = DELTA / (sqrt(2.0) * c);int main()
{int m, n;double source;Grid *g;ALLOC_2D(g, 100, 100, Grid);EHInit(g);gridInit(g);for (Time = 0; Time < 150; Time++){updateH(g);updateBoundary(g);updateE(g);source = driveSource(g, Time, 0, 0);Ez(SizeX / 2, SizeY / 2) = source;}writeFile(g);getchar();return 0;
}
void EHInit(Grid *g)
{int n, m;for (m = 0; m < SizeX; m++){for (n = 0; n < SizeY; n++){Hy(m, n) = 0;Hx(m, n) = 0;Ez(m, n) = 0;}}
}void gridInit(Grid *g)
{int m, n;SizeX = 100;SizeY = 100;MaxTime = 300;CDTDS = 1.0 / sqrt(2.0);ALLOC_2D(g->hx, SizeX, SizeY, double);ALLOC_2D(g->chxh, SizeX, SizeY, double);ALLOC_2D(g->chxe, SizeX, SizeY, double);ALLOC_2D(g->hy, SizeX, SizeY, double);ALLOC_2D(g->chyh, SizeX, SizeY, double);ALLOC_2D(g->chye, SizeX, SizeY, double);ALLOC_2D(g->ez, SizeX, SizeY, double);ALLOC_2D(g->ceze, SizeX, SizeY, double);ALLOC_2D(g->cezh, SizeX, SizeY, double);for (m = 0; m < SizeX; m++){for (n = 0; n < SizeY; n++){//Ez(m, n) = 0;CA(m, n) = (1 - 0.5 * SIGMA0 * DELTAT / MU0) / (1 + 0.5 * SIGMA0 * DELTAT / MU0);CB(m, n) = (DELTAT / EPS0) / (1 + (SIGMA0 * DELTAT / (2 * EPS0)));}}for (m = 0; m < SizeX; m++){for (n = 0; n < SizeY; n++){//Hx(m, n) = 0;CP(m, n) = (1 - (SIGMAm0 * DELTAT) / (2 * MU0)) / (1 + (SIGMAm0 * DELTAT) / (2 * MU0));CQ(m, n) = (DELTAT / MU0) / (1 + (SIGMAm0 * DELTAT / (2 * MU0)));}}printf("The CA is :%f\n", CA(50, 50));printf("The CB is :%f\n", CB(50, 50));printf("The CP is :%f\n", CP(50, 50));printf("The CQ is :%f\n", CQ(50, 50));
}double driveSource(Grid *g, double time, double locX, double locY)
{double result;double ricker_result;double ppw = 20;double cdtds = CDTDS;double w0 = 1000000000.0; //天线的中心频率double arg;result = time * time * sin(2 * 3.1415 * w0 * time) * exp(-0.93 * w0 * time);arg = 3.14159 * ((cdtds * time - 0.0) / ppw - 1.0);arg = arg * arg;ricker_result = (1.0 - 2.0 * arg) * exp(-arg);return ricker_result;
}void updateE(Grid *g)
{int m, n;for (m = 1; m < (SizeX - 1); m++){for (n = 1; n < (SizeY - 1); n++){Ez(m, n) = CA(m, n) * Ez(m, n) + CB(m, n) * ((Hy(m, n) - Hy((m - 1), n)) / DELTA - (Hx(m, n) - Hx(m, (n - 1))) / DELTA);}}
}void updateH(Grid *g)
{int m, n;for (m = 0; m < (SizeX); m++){for (n = 0; n < (SizeY - 1); n++){Hx(m, n) = CP(m, n) * Hx(m, n) - CQ(m, n) * (Ez(m, (n + 1)) - Ez(m, n)) / DELTA;}}for (m = 0; m < (SizeX - 1); m++){for (n = 0; n < (SizeY); n++){Hy(m, n) = CP(m, n) * Hy(m, n) + CQ(m, n) * (Ez((m + 1), n) - Ez(m, n)) / DELTA;}}
}void updateBoundary(Grid *g)
{int m, n;int firstX = 0, firstY = 0, lastX = 99, lastY = 99;Grid *g1;ALLOC_2D(g1, 100, 100, Grid);memcpy(g1, g, sizeof(Grid));gridInit(g1);//updateH(g1);updateE(g1);//修正Hy左边界m = firstX;for (n = 1; n < lastY; n++){Ez(m, n) = Ez((m + 1), n) + ((c * DELTAT - DELTA) / (c * DELTAT + DELTA)) * (g1->ez[(m+1) * SizeY + n] - Ez(m, n))//-((c * c * DELTAT * MU0) / (2 * (c * DELTAT + DELTA))) * (g1->hx[m*SizeY+(n+1)] - g1->hx[m*SizeY + (n - 1)] + g1->hx[(m+1)*SizeY + (n + 1)] - g1->hx[(m + 1)*SizeY + (n - 1)]);-((c * c * DELTAT * MU0) / (2 * (c * DELTAT + DELTA))) * (Hx(m, (n + 1)) - Hx(m, (n - 1)) + Hx((m + 1), (n + 1)) - Hx((m + 1), (n - 1)));}//修正Hy右边界m = lastX;for (n = 1; n < lastY; n++){Ez(m, n) = Ez((m - 1), n) + ((c * DELTAT - DELTA) / (c * DELTAT + DELTA)) * (g1->ez[(m - 1) * SizeY + n] - Ez(m, n))- ((c * c * DELTAT * MU0) / (2 * (c * DELTAT + DELTA))) * (Hx(m, (n + 1)) - Hx(m, (n - 1)) + Hx((m - 1), (n + 1)) - Hx((m - 1), (n - 1)));}/*//修正Hx下边界//修正Hx上边界*/
}void writeFile(Grid *g)
{int m, n;FILE *f = fopen("C:\\Users\\Administrator\\Desktop\\data.txt", "a");if (f == NULL) printf("open file error");for (n = SizeY - 1; n >= 0; n--){for (m = 0; m < SizeX; m++){fprintf(f, "%f,", Ez(m, n));}fprintf(f, "\n");}fclose(f);
}


然后用matlab对生成的东西成像:

imshow(data50,[]);

时间步为50:


时间步为100:


时间步为150:


会发现150的有反射波了,这是因为没加上吸收边界。


如果加上MUR吸收边界是这样的:

可以看到反射波确实有变好一点,但是边框上出现了黑边,还得对边上的Ez进行修正才行

这篇关于电磁波时域有限差分方法(FDTD)-C语言-FDTD和激励源和MUR边界的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/366381

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符

Python版本与package版本兼容性检查方法总结

《Python版本与package版本兼容性检查方法总结》:本文主要介绍Python版本与package版本兼容性检查方法的相关资料,文中提供四种检查方法,分别是pip查询、conda管理、PyP... 目录引言为什么会出现兼容性问题方法一:用 pip 官方命令查询可用版本方法二:conda 管理包环境方法

Linux云服务器手动配置DNS的方法步骤

《Linux云服务器手动配置DNS的方法步骤》在Linux云服务器上手动配置DNS(域名系统)是确保服务器能够正常解析域名的重要步骤,以下是详细的配置方法,包括系统文件的修改和常见问题的解决方案,需要... 目录1. 为什么需要手动配置 DNS?2. 手动配置 DNS 的方法方法 1:修改 /etc/res

JavaScript对象转数组的三种方法实现

《JavaScript对象转数组的三种方法实现》本文介绍了在JavaScript中将对象转换为数组的三种实用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录方法1:使用Object.keys()和Array.map()方法2:使用Object.entr

SpringBoot中ResponseEntity的使用方法举例详解

《SpringBoot中ResponseEntity的使用方法举例详解》ResponseEntity是Spring的一个用于表示HTTP响应的全功能对象,它可以包含响应的状态码、头信息及响应体内容,下... 目录一、ResponseEntity概述基本特点:二、ResponseEntity的基本用法1. 创

java中判断json key是否存在的几种方法

《java中判断jsonkey是否存在的几种方法》在使用Java处理JSON数据时,如何判断某一个key是否存在?本文就来介绍三种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的... 目http://www.chinasem.cn录第一种方法是使用 jsONObject 的 has 方法