【manim】3b1b的"Almost" Fourier Transform复刻
最近在做Fourier Transform的内容,记录一下今天下午的成果。
本文代码全部自行编写,需要math and music项目完整工程可以在gayhub上获取。(现在还没弄完,就先不发了。)
概要
第一部分:
图像代码部分原理很直接,即极坐标参数方程的转化。
第二部分:
关于图像质点中心的问题,数学上需要使用复数与微积分的知识求出。
3b1b的原代码也是将质点的x-coordinate of center of mass图像直接用公式绘制。
但是本文使用的是暴力的方法,直接用Mobject的get_center通过帧updater描点的方法绘制离散的点集。如果需要,可以拟合为曲线。
原视频:https://www.youtube.com/watch?v=spUNpyF58BY&t=188s


bilibili:
https://www.bilibili.com/video/BV1pW411J7s8?share_source=copy_web
数学公式在geogebra中构建如下:

即将cos(mx)+1转化为参数方程[(1+cos(mt))·cos(nt),(1+cos(mt))·sin(nt)]
关于sin和cos的选择:
正弦和余弦差别在于初始相位。这里出于教学考虑用余弦而不是正弦,主要是因为复指数函数的实部是余弦。如果用正弦的话,会发现质心的纵坐标而不是横坐标在频率重合的时候达到峰值(而且是负的)。


manim代码:
创建的Mobject有:axes axes_label polarPlane D_delta D_rou D_move
以及三个图像:axes_graph polar_graph para_graph
然后使用add_update()添加DecimalNumber作为变化数值到动画更新函数即可。
第一部分代码:
class FourierTrans_First(Scene):
def construct(self):
axes = Axes(
x_range=[-0, 2*PI, PI/4],
y_range=[-0.1, 2.5, 1],
x_length=14,
y_length=2
).to_edge(UP)
axes_label = axes.get_axis_labels('Time','Intensity')
polarPlane = PolarPlane(
radius_max=2.0,
azimuth_units="TAU radians",
size=4,
azimuth_label_font_size=24,
radius_config={"font_size": 24},
).add_coordinates().to_edge(DOWN+LEFT)
#显示坐标与坐标标签
self.add(axes,axes_label,polarPlane)
#一些全局参数-修改这三个参数就好
delta = 0#偏移量
rou = 20#theta的系数
move = 10#极坐标变换的变化程度系数
dn_kwargs = {'num_decimal_places':3,}
D_delta = DecimalNumber(delta,**dn_kwargs)
D_rou = DecimalNumber(rou,**dn_kwargs)
D_move = DecimalNumber(move,**dn_kwargs).shift(DOWN)
#解析式-公式修改这里就好
r = lambda t: np.sin(D_rou.get_value()*t)+1
par = lambda t: np.array([
(r(t))*np.cos(D_move.get_value()*t),
(r(t))*np.sin(D_move.get_value()*t),
0])
#更新函数
def Func_axes():
return axes.plot(r,color=PINK)
def Func_polar():
return polarPlane.plot_polar_graph(r, [0, 2 * PI], color=ORANGE)
def Func_para():
para = ParametricFunction(
par,
t_range = np.array([0, 3.14*2]),
fill_opacity=0)\
.set_color(color=[RED,YELLOW,BLUE,RED])\
.shift(DOWN*2+RIGHT*2)
self.add(Dot(para.get_center()))
return para
axes_graph = Func_axes()
polar_graph = Func_polar()
para_graph = Func_para()
axes_graph.add_updater(lambda mob: mob.become(Func_axes()))
polar_graph.add_updater(lambda mob: mob.become(Func_polar()))
para_graph.add_updater(lambda mob:mob.become(Func_para()))
self.add(D_rou,D_move)
self.play(Write(axes_graph),Write(polar_graph),Write(para_graph))
#self.play(ChangeDecimalToValue(D_rou,20),run_time=5,rate_func=linear)
self.play(ChangeDecimalToValue(D_move,20),run_time=5,rate_func=linear)
self.wait(1)
第二部分(含一)代码:
from manim import *
class FourierTrans_First(Scene):
def construct(self):
#########全局参数-修改这5个参数就好#########
delta = 0#偏移量
rou = 0#theta的系数
move = 1#极坐标变换的变化程度系数
target = 1#追踪质心y坐标
track_end = 18#扭扭的结尾
############################################
dn_kwargs = {'num_decimal_places':3,}
D_delta = DecimalNumber(delta,**dn_kwargs)
D_rou = DecimalNumber(rou,**dn_kwargs).shift(LEFT)
D_move = DecimalNumber(move,**dn_kwargs).next_to(D_rou,DOWN)
D_target = DecimalNumber(target,**dn_kwargs).next_to(D_move,DOWN)
#######解析式-修改这里 《公式r》 就好#######
r = lambda t: np.sin(D_rou.get_value()*t)+1
############################################
par = lambda t: np.array([
(r(t))*np.cos(D_move.get_value()*t),
(r(t))*np.sin(D_move.get_value()*t),
0])
#coordinate system & labels
axes = Axes(
x_range=[-0, 2*PI, PI/4],
y_range=[-0.1, 2.5, 1],
x_length=14,
y_length=2
).to_edge(UP)
axes_label = axes.get_axis_labels('Time','Intensity')
#polar plane
polarPlane = PolarPlane(
radius_max=2.0,
azimuth_units="TAU radians",
size=4,
azimuth_label_font_size=24,
radius_config={"font_size": 24},
).add_coordinates().to_edge(DOWN+LEFT)
#point track plane
pointAxes = Axes(
x_range=[0, track_end+track_end*0.3,1],
y_range=[-0.5, 1],
x_length=6,
y_length=4
).to_edge(UP).to_edge(DOWN+RIGHT)
#更新函数
def Func_axes():
return axes.plot(r,color=PINK)
def Func_polar():
return polarPlane.plot_polar_graph(r, [0, 2 * PI], color=ORANGE)
def Func_para():
para = ParametricFunction(
par,
t_range = np.array([0, 3.14*2]),
fill_opacity=0)\
.set_color(color=[RED,YELLOW,BLUE,RED])
return para
def Func_para_with_messPoint():
para = ParametricFunction(
par,
t_range = np.array([0, 3.14*2]),
fill_opacity=0)\
.set_color(color=[RED,YELLOW,BLUE,RED])
messPoint = Dot(pointAxes.coords_to_point(D_move.get_value(),para.get_y()),color=GREEN_B,radius=0.02)
self.add(messPoint)
return para
axes_graph = Func_axes()
polar_graph = Func_polar()
para_graph = Func_para()
#define messPoint
messPoint = Dot(para_graph.get_center())
#定义updater_func表达式
axU = lambda mob: mob.become(Func_axes())
poU = lambda mob: mob.become(Func_polar())
paU = lambda mob:mob.become(Func_para().shift(DOWN+RIGHT*3))
paU_with_messPoint = lambda mob:mob.become(Func_para_with_messPoint().move_to(para_graph))
#添加updater
axes_graph.add_updater(axU)
polar_graph.add_updater(poU)
para_graph.add_updater(paU)
################################################
####################开始动画####################
################################################
#显示坐标与坐标标签
group_axes = VGroup(axes,axes_label,axes_graph)
group_polar = VGroup(polarPlane,polar_graph)
self.add(D_rou,D_move)
self.play(Write(group_axes),Write(group_polar),Write(para_graph))
self.play(ChangeDecimalToValue(D_rou,track_end),run_time=14,rate_func=linear)
#左下角的图像消失,右下角的图像挪来左边
polar_graph.remove_updater(poU)
self.play(FadeOut(polar_graph))
para_graph.remove_updater(paU)
self.play(para_graph.animate.move_to(group_polar))
self.play(GrowFromCenter(pointAxes))
#开始扭扭~并且加点追踪
para_graph.add_updater(paU_with_messPoint)
D_move.set_value(0)
self.play(ChangeDecimalToValue(D_move,track_end),run_time=30,rate_func=linear)
self.wait(1)
【manim】3b1b的"Almost" Fourier Transform复刻的更多相关文章
- 傅里叶变换 - Fourier Transform
傅里叶级数 傅里叶在他的专著<热的解析理论>中提出,任何一个周期函数都可以表示为若干个正弦函数的和,即: \[f(t)=a_0+\sum_{n=1}^{\infty}(a_ncos(n\o ...
- 使用 scipy.fft 进行Fourier Transform:Python 信号处理
摘要:Fourier transform 是一个强大的概念,用于各种领域,从纯数学到音频工程甚至金融. 本文分享自华为云社区<使用 scipy.fft 进行Fourier Transform:P ...
- 从傅里叶级数(Fourier series)到离散傅里叶变换(Discrete Fourier transform)
从傅里叶级数(Fourier series)到离散傅里叶变换(Discrete Fourier transform) 一. 傅里叶级数(FS) 首先从最直观的开始,我们有一个信号\(x(t)\)(满足 ...
- YY表行推荐十块顶级复刻表,一比一开模复刻,外观堪比正品
随着国内制表工艺的逐渐提升,顶级复刻表的行列里成员越来越多,今天复刻表工厂就总结一下最值得入手的十款顶级复刻表来和大家分享. TOP 10:爱彼 AP15400购买指数★★★ AP15400采用顶级复 ...
- 复刻smartbits的国产网络测试工具minismb简介
复刻smartbits的国产网络性能测试工具minismb,是一款专门用于测试智能路由器,网络交换机的性能和稳定性的软硬件相结合的工具.可以通过此工具测试任何ip网络设备的端口吞吐率,带 ...
- 复刻smartbits的国产网络性能测试工具minismb-如何配置Ping报文
复刻smartbits的国产网络性能测试工具minismb,是一款专门用于测试智能路由器,网络交换机的性能和稳定性的软硬件相结合的工具.可以通过此以太网测试工具测试任何ip网络设备的端口吞吐率,带宽, ...
- 复刻smartbits的国产网络测试工具minismb-使用burst模式
复刻smartbits的国产网络性能测试工具minismb,是一款专门用于测试智能路由器,网络交换机的性能和稳定性的软硬件相结合的工具.可以通过此工具测试任何ip网络设备的端口吞吐率,带宽,并发连接数 ...
- 复刻smartbits的国产网络测试工具minismb-如何添加数据流
复刻smartbits的国产网络性能测试工具minismb,是一款专门用于测试智能路由器,网络交换机的性能和稳定性的软硬件相结合的工具.可以通过此工具测试任何ip网络设备的端口吞吐率,带宽,并发连接数 ...
- 复刻smartbits的国产网络测试工具minismb-如何测试路由器
复刻smartbits的国产网络性能测试工具minismb,是一款专门用于测试智能路由器,网络交换机的性能和稳定性的软硬件相结合的工具.可以通过此工具测试任何ip网络设备的端口吞吐率,带宽,并发连接数 ...
随机推荐
- Arraylist集合、对象数组
Arraylist集合 ArrayList是List接口的一个实现类,它是程序中最常见的一种集合. 他的特点:在增加或删除指定位置的元素时,会创建新的数组,效率比较低,因此不适合做大量的增删操作,Ar ...
- JAVA 异常 基本知识
异常 异常定义 异常是运行过程中出现的错误 人为错误:填写错误等 随机错误:网络中断.内存耗尽等 一个健壮的程序必须处理各种各样的错误 Java的异常是class Object Throwable E ...
- VR技术赋能五大领域,不止高级,更高效!
除了VR游戏.VR影视作品,究竟还有哪些产业领域会应用到VR技术并为生活带来改变呢?今天就帮大家好好梳理一下~ VR赋能交通,不只是高级 最近在网上看到了VR考驾照的新闻,网友都赞叹,现在学车都这么高 ...
- 密码学系列之:PKI的证书格式表示X.509
目录 简介 一个证书的例子 X.509证书的后缀 .pem .cer, .crt, .der .p7b, .p7c .p12 .pfx 证书的层级结构和交叉认证 x.509证书的使用范围 总结 简介 ...
- 运行时应用自我保护(RASP):应用安全的自我修养
应用程序已经成为网络黑客想要渗透到企业内部的绝佳目标. 因为他们知道如果能发现并利用应用程序的漏洞,他们就有超过三分之一的机会成功入侵. 更重要的是,发现应用程序漏洞的可能性也很大. Contrast ...
- 手把手教你实现一个图片压缩工具(Vue与Node的完美配合)
前言 图片压缩对于我们日常生活来讲,是非常实用的一项功能.有时我们会在在线图片压缩网站上进行压缩,有时会在电脑下软件进行压缩.那么我们能不能用前端的知识来自己实现一个图片压缩工具呢?答案是有的.效果展 ...
- Linux shell环境的配置
shell配置文件分类 按生效范围分类:全局和局部 按登录方式分类:交互式和非交互式 按功能分类:profile和bashrc shell配置文件按生效范围分类: 全局配置:针对有所用户有效 /etc ...
- Python: list列表的11个内置方法
先来逼逼两句: 在实际开发中,经常需要将一组(不只一个)数据存储起来,以便后边的代码使用.在VBA中有使用数组,可以把多个数据存储到一起,通过数组下标可以访问数组中的每个元素.Python 中没有数组 ...
- Netty 如何高效接收网络数据?一文聊透 ByteBuffer 动态自适应扩缩容机制
本系列Netty源码解析文章基于 4.1.56.Final版本,公众号:bin的技术小屋 前文回顾 在前边的系列文章中,我们从内核如何收发网络数据开始以一个C10K的问题作为主线详细从内核角度阐述了网 ...
- Spring Boot 知识点总结
现在仅总结重要和实用的知识点,更加全面的请见链接:1.:2.. 微服务:架构风格(服务微化):一个应用应该是一组小型服务:可以通过HTTP的方式进行互通:微服务:每一个功能元素终都是一个可独立替换和独 ...