电磁波时域有限差分方法(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

相关文章

SpringBoot使用Jasypt对YML文件配置内容加密的方法(数据库密码加密)

《SpringBoot使用Jasypt对YML文件配置内容加密的方法(数据库密码加密)》本文介绍了如何在SpringBoot项目中使用Jasypt对application.yml文件中的敏感信息(如数... 目录SpringBoot使用Jasypt对YML文件配置内容进行加密(例:数据库密码加密)前言一、J

Spring Boot 中正确地在异步线程中使用 HttpServletRequest的方法

《SpringBoot中正确地在异步线程中使用HttpServletRequest的方法》文章讨论了在SpringBoot中如何在异步线程中正确使用HttpServletRequest的问题,... 目录前言一、问题的来源:为什么异步线程中无法访问 HttpServletRequest?1. 请求上下文与线

解读为什么@Autowired在属性上被警告,在setter方法上不被警告问题

《解读为什么@Autowired在属性上被警告,在setter方法上不被警告问题》在Spring开发中,@Autowired注解常用于实现依赖注入,它可以应用于类的属性、构造器或setter方法上,然... 目录1. 为什么 @Autowired 在属性上被警告?1.1 隐式依赖注入1.2 IDE 的警告:

SpringBoot快速接入OpenAI大模型的方法(JDK8)

《SpringBoot快速接入OpenAI大模型的方法(JDK8)》本文介绍了如何使用AI4J快速接入OpenAI大模型,并展示了如何实现流式与非流式的输出,以及对函数调用的使用,AI4J支持JDK8... 目录使用AI4J快速接入OpenAI大模型介绍AI4J-github快速使用创建SpringBoot

Android开发中gradle下载缓慢的问题级解决方法

《Android开发中gradle下载缓慢的问题级解决方法》本文介绍了解决Android开发中Gradle下载缓慢问题的几种方法,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、网络环境优化二、Gradle版本与配置优化三、其他优化措施针对android开发中Gradle下载缓慢的问

python 3.8 的anaconda下载方法

《python3.8的anaconda下载方法》本文详细介绍了如何下载和安装带有Python3.8的Anaconda发行版,包括Anaconda简介、下载步骤、安装指南以及验证安装结果,此外,还介... 目录python3.8 版本的 Anaconda 下载与安装指南一、Anaconda 简介二、下载 An

Java中将异步调用转为同步的五种实现方法

《Java中将异步调用转为同步的五种实现方法》本文介绍了将异步调用转为同步阻塞模式的五种方法:wait/notify、ReentrantLock+Condition、Future、CountDownL... 目录异步与同步的核心区别方法一:使用wait/notify + synchronized代码示例关键

使用Go语言开发一个命令行文件管理工具

《使用Go语言开发一个命令行文件管理工具》这篇文章主要为大家详细介绍了如何使用Go语言开发一款命令行文件管理工具,支持批量重命名,删除,创建,移动文件,需要的小伙伴可以了解下... 目录一、工具功能一览二、核心代码解析1. 主程序结构2. 批量重命名3. 批量删除4. 创建文件/目录5. 批量移动三、如何安

Python使用Pandas对比两列数据取最大值的五种方法

《Python使用Pandas对比两列数据取最大值的五种方法》本文主要介绍使用Pandas对比两列数据取最大值的五种方法,包括使用max方法、apply方法结合lambda函数、函数、clip方法、w... 目录引言一、使用max方法二、使用apply方法结合lambda函数三、使用np.maximum函数

Qt 中集成mqtt协议的使用方法

《Qt中集成mqtt协议的使用方法》文章介绍了如何在工程中引入qmqtt库,并通过声明一个单例类来暴露订阅到的主题数据,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一,引入qmqtt 库二,使用一,引入qmqtt 库我是将整个头文件/源文件都添加到了工程中进行编译,这样 跨平台