python光学仿真面向对象光学元件类的实现

光学元件类

平面反射镜是一种极为简单的模型,因为我们只需要考虑一个平面即可。但是除此之外的其他光学元件,可能会变得有些复杂:我们必须考虑光在入射面和出射面的行为。

这当然是一句废话,而且我们也有了一个初步的解决方案:将光学元件拆成前表面和后表面即可。如果光需要在光学元件中反射多次,那就将光学元件拆成需要反射次数的表面个数即可,完美而无脑。

这说明我们已经熟悉了程序员的思维,我们眼中的世界已经不再是一个所见即所得的世界,我们看到的是一个个抽象零部件的表现。但是也不要惊慌,程序员和正常人也未必有很大的区别,因为我们除了可以将这个世界拆解,也可以将拆解之后的部件重新构造回这个世界。

尝试着将问题想得复杂一些,光学系统中有许多光学元件,光会透过每个光学元件很多次,而且每次的入射点、出射点都会有一定的偏差。由于光学元件可能会对光的能量有所吸收,从而引起发热。而且每次的入射点、出射点不同,则发热位置也不一样。由于发热会导致光学元件发生形变,所以下一次光和光学元件的作用也会发生变化。

也就是说,对于每个光学元件来说,除了有固定的前表面、后表面,还有入射点、出射点、发热、形变等不断变化的参数。这样的一个过于实际的问题促使我们构造一种更加贴近现实的数据类型,换句话说,我们要创建一个对象,这个对象能够封装各种变量和功能,我们输入一个参量,这个对象的状态也会跟着发生变化。

这就是所谓的面向对象。

class Opti():
    def __init__(self,edge1,edge2):
        self.edge1 = edge1
        self.edge2 = edge2

在上例中,我们定义了一个光学元件类,这个光学元件有两个表面,这两个表面既可以是平面,也可以说弧面。这样,我们就建立了一个类。其中,__init__为初始化方法,self表示我们所创建的这个类本身。一般来说,如果类中的方法不加修饰符的话,就必须将self当作第一个参数。

self.edge1表示这个Opti类中,有一个成员的名字叫edge1。当这个类被初始化的时候,我们就可以对其进行赋值了。

有些元件可能只有一个表面,比如全反镜;有些可能有多个表面,比如偏振立方体。而且,我们在做实验的时候,也需要对不同的光学元件进行比较,从而得到最好的实验结果。所以,如果我们想改变已经建好的光学元件,应该怎么办呢?

其实很简单,只要增加一个方法,使得可以插入或者删除新的表面即可。

#文件Opti.py
class Opti():
    def __init__(self,edges=[[(0,-1),(0,1)],[(0,1),(0,-1),(1/2,0)]]):
        self.edges = [{'index':i,'dots':edges[i]}
                      for i in range(len(edges))]
    #edge格式为(dot1,dot2,...)
    def insertEdge(self,edge,albedo=0):
        self.edges.append(
            {'index':len(self.edges),'dots':edge})
    #可接受编号和点集
    def delEdge(self,edge):
        try:
            if isinstance(edge,list):   #如果edge的类型是list
                for edg in self.edges:
                    if edg['dots']==edge:
                        edge = edg['index']
            del self.edges[edge]
        except:
            print("no this edge")

在上面的代码中,可以看到初始化函数被预设了一些值,这点与普通函数并无二致。我们可以看到,默认插入的两个曲面分别是平面[(0,-1),(0,1)]和弧面[(0,1),(0,-1),(1/2,0)],可见默认生成一个平凸镜。

成员变量self.edges即光学表面的列表,每个光学表面有两个参数,分别是索引index和点集dots。由此前的光学抽象可知,当点对中有两个点的时候,代表平面;有三个点的时候,代表弧面。

方法insertEdge为插入一个光学表面,其中,编号为这个光学表面在self.edges中的索引号;delEdge顾名思义为删除某个光学表面。如果传入的edge为一个列表,则说明传入的是一个参数确定的曲面,此时通过遍历self.edges找到这个表面,并得到其索引。

如果传入的参数为一个单值,那么说明传入的是索引号,所以直接删除即可。

在这个方法中,使用了一种新的代码块try:...except...,这是一种异常机制,即尝试运行try:块中的代码,如果运行失败,则执行except。如果我们没能执行成功delEdge,则说明我们输入的表面并不在这个光学元件中,所以输出"no this edge"

这好像是第一次看到print这个命令呢,一般来说这应该是最先接触到的函数,毕竟对于大多数程序员来说,敲下的第一行代码就是

print("hello world")
print('hello world')

同时,我们除了数值类型之外,又认识了另一种数据类型,即字符。在python中,可以通过双引号或者单引号来表示单个字符或者字符串。即上述的hello world代码中,两行均正确,而且没什么区别。

现在,我们已经写了一个类,于是可以创建一个对象,在命令行中输入:

>>> from Opti import Opti
>>> Opti.__name__   #这是什么鬼
'Opti'
>>> x = Opti()      #创建对象,由于未输入参数,故皆为默认值
>>> x.edges         #现实类成员
[{'index': 0, 'dots': [(0, -1), (0, 1)]}, {'index': 1, 'dots': [(0, 1), (0, -1), (0.5, 0)]}]
>>> x.delEdge(1)    #调用类方法
>>> x.edges         #果然少了一个边
[{'index': 0, 'dots': [(0, -1), (0, 1)]}]
>>> x.delEdge(1)    #删除不存在的边是不可能的
no this edge
>>>

首先,from Opti import Opti的这两个Opti并不相同,前者代表包`Opti.py',后者代表Opti.py中的类'Opti',import之后便可以调用了。

然后出现了一个比较吊诡的事情,我们在类中并没有定义__name__,然而调用之后却有值产生。

请勿惊慌,其实是老熟人了。可以将__name__理解为python内部的内置属性,当我们直接执行某一.py文件时,这个__name__的值为__main__,否则的话就是类的名字。所以,到这个时候,我们似乎应该能明白入口函数的真正意义了吧。

继续向下,几乎所有的事情就都不出所料了。

以上就是python光学仿真面向对象光学元件类的实现的详细内容,更多关于python光学元件类的实现的资料请关注我们其它相关文章!

(0)

相关推荐

  • Python光学仿真教程实现光线追踪

    目录 光线追迹 几何抽象 光线 线段与圆弧 光线追迹 得益于计算机的计算的能力,通过追踪具有代表性的光线的传播轨迹,可以更加精确地描述光学系统的性能,光线追迹方法也因此大展其能,诸如Zemax.tracepro等软件便都提供了相应的功能. 而建立在折射定律基础之上的光线追迹方法,对数学功底要求较低,所以比较适合作为python初学者的入门项目.在接下来的这一章,希望通过对光线追迹的实现,掌握python中的列表.元组.字典.集合等数据类型的基本概念,并且对面向对象与函数式编程有一个基本的了解.

  • Python面向对象之类和对象实例详解

    本文实例讲述了Python面向对象之类和对象.分享给大家供大家参考,具体如下: 类和对象(1) 对象是什么? 对象=属性(静态)+方法(动态): 属性一般是一个个变量:方法是一个个函数: #类的属性 就是 类变量 #实例变量:定义在方法中的变量,只作用于当前实例的类. 例子: class Turtle:#python 中类名约定以大写字母开头 '''关于类的简单例子...''' #属性 == 类变量 color ="green" weight="10kg" legs

  • python光学仿真实现光线追迹折射与反射的实现

    目录 折射与反射 平面反射 平面折射 python实现 弧面问题 折射与反射 光线与光学元件相互作用,无非只有两件事,反射和透射.而就目前看来,我们所常用的光学元件,也无非有两种表面,即平面和球面,二维化之后就简化成了射线与线段,射线与劣弧的关系. 平面反射 无论从哪个角度来看,平面的反射折射都要比球面更简单,而反射问题要比折射问题更简单,所以,我们首先处理平面的反射问题. 反射定律即入射角等于反射角,心念及此,最为循规蹈矩的思路必然是先找到入射光线和平面的夹角,然后用这个夹角和平面(在二维空间

  • python光学仿真实现光线追迹之空间关系

    目录 空间关系 相交判定 射线排序 线弧关系 点弧关系 空间关系 变化始于相遇,所以交点是一切的核心. 相交判定 首先考察一束光线能否打在某个平面镜上.光线被抽象成了一个列表[a,b,c],平面镜则被抽象成为由两个点构成的线段[(x1,y1),(x2,y2)].两条直线的交点问题属于初等数学范畴,需要先将线段转换成直线的形式,然后再求交点.但是两条直线的交点可能落在线段的外面,从而不具有判定的意义. 如果我们的光学系统中有大量的光学元件,那么如果有一种方法可以快速判断光线是否与光学元件有交点,将

  • Python面向对象类编写细节分析【类,方法,继承,超类,接口等】

    本文实例讲述了Python面向对象类编写技术细节.分享给大家供大家参考,具体如下: 类代码编写细节 继续学习类.方法和继承. class语句 以下是class语句的一般形式: class <name>(superclass,...): data = value def method(self,...): self.member = value 在class语句内,任何赋值语句都会产生类属性,而且还有特殊名称方法重载运算符.例如,名为__init__的函数会在实例对象构造时调用(如果定义过的话)

  • python光学仿真面向对象光学元件类的实现

    光学元件类 平面反射镜是一种极为简单的模型,因为我们只需要考虑一个平面即可.但是除此之外的其他光学元件,可能会变得有些复杂:我们必须考虑光在入射面和出射面的行为. 这当然是一句废话,而且我们也有了一个初步的解决方案:将光学元件拆成前表面和后表面即可.如果光需要在光学元件中反射多次,那就将光学元件拆成需要反射次数的表面个数即可,完美而无脑. 这说明我们已经熟悉了程序员的思维,我们眼中的世界已经不再是一个所见即所得的世界,我们看到的是一个个抽象零部件的表现.但是也不要惊慌,程序员和正常人也未必有很大

  • Python光学仿真wxpython透镜演示系统初始化与参数调节

    初始化与参数调节面板 这一节将绘制出如下图所示的参数调节面板 对于上图来说,BoxSizer布局十分傻瓜,所以这里主要有两个方面需要注意,其一是opti和source这两个选项卡的实现,其二则是如何同时创建多个滚动条. 对于前者比较容易,无非是多用一个控件而已,即wx.NoteBook,使用方法乏善可陈,看代码即可学会. 对于后者当然也可以很容易,只要无脑罗列即可,只不过对于五个不同的参数就意味着要新建五组滚动条,要就要新建五个控制函数,而这五个控制函数的功能几乎是完全一样的.显然,这很愚蠢,所

  • Python光学仿真wxpython透镜演示系统框架

    透镜演示系统 框架 现在,我们可以做一个具备友好界面的透镜演示系统了.我们需要两个圆弧来表示透镜,一条线段表示主光轴,多条线段表示光线的传播路径.此外,还需要对光源和透镜的参数进行调节. 然而值得注意的一点是,我们在进行计算和画图过程中所用到的几何图形,在表达形式以及操作流程上可能并不相同.例如,对于光源发出的一条射线,它与透镜的作用流程为 寻找与透镜前表面的交点A 获取反射和透射直线 寻找透射直线与透镜后表面的交点B 计算透过透镜的直线 然而对于画图程序来说,光源S和A之间有一条线段,A和B之

  • Python光学仿真wxpython之DC绘图

    一般来说,系统与绘图程序之间的信息交换是由图形设备接口(Graphics Device Interface,GDI)实现的,在wxpython中,通过device context(DC)对象来实现GDI的功能. DC对象的创建非常简单,只需引用wx.PaintDC即可,而后则可通过dc来设置画笔dc.SetPen,有了画笔,就可以进行图形绘制了.于是,我们再考虑到图形的属性,包括形状.颜色与边框等,更细致地说,是图形形状.填充颜色.边框类型.边框颜色. 我们可以通过一个矩形的例子来说明: 上面的

  • python光学仿真PyQt5基础框架教程

    前几天为了自己搞一个光学仿真集成GUI界面,于是去研究了一下PyQt5,不得不说这个模块的使用性远远超过了tkinter,强烈推荐,于是准备出一个专栏,记录一下PyQt5学习中遇到的小问题. 这篇先来说说PyQt5创建时候的基础框架.代码如下: # -*- coding:utf-8 -*- import sys from PyQt5.QtWidgets import QMainWindow, QApplication class MainWindow(QMainWindow): def __in

  • 浅谈python中的面向对象和类的基本语法

    当我发现要写python的面向对象的时候,我是踌躇满面,坐立不安呀.我一直在想:这个坑应该怎么爬?因为python中关于面向对象的内容很多,如果要讲透,最好是用面向对象的思想重新学一遍前面的内容.这个坑是如此之大,犹豫再三,还是只捡一下重要的内容来讲吧,不足的内容只能靠大家自己去补充了. 惯例声明一下,我使用的版本是 python2.7,版本之间可能存在差异. 好,在开讲之前,我们先思考一个问题,看代码: 为什么我只创建是为 a 赋值,就可以使用一些我没写过的方法? 可能会有小伙伴说:因为 a

  • python光学仿真通过菲涅耳公式实现波动模型

    从物理学的机制出发,波动模型相对于光线模型,显然更加接近光的本质:但是从物理学的发展来说,波动光学旨在解决几何光学无法解决的问题,可谓光线模型的一种升级.从编程的角度来说,波动光学在某些情况下可以简单地理解为在光线模型的基础上,引入一个相位项. 波动模型 一般来说,三个特征可以确定空间中的波场:频率.振幅和相位,故光波场可表示为: import numpy as np import matplotlib.pyplot as plt from mpl_toolkits.mplot3d import

随机推荐