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

目录
  • 空间关系
    • 相交判定
    • 射线排序
    • 线弧关系
    • 点弧关系

空间关系

变化始于相遇,所以交点是一切的核心。

相交判定

首先考察一束光线能否打在某个平面镜上。光线被抽象成了一个列表[a,b,c],平面镜则被抽象成为由两个点构成的线段[(x1,y1),(x2,y2)]。两条直线的交点问题属于初等数学范畴,需要先将线段转换成直线的形式,然后再求交点。但是两条直线的交点可能落在线段的外面,从而不具有判定的意义。

如果我们的光学系统中有大量的光学元件,那么如果有一种方法可以快速判断光线是否与光学元件有交点,将会显得更加快捷。那么,如果一个直线穿过某个线段,就意味着这条线段的两个端点必然在直线的两侧。

import numpy as np
def isCross(abc,dots):
    abc = np.array(abc)                 #将abc的格式转成数组
    flag1 = abc.dot(list(dots[0])+[1])    #数组的点积
    flag2 = abc.dot(list(dots[1])+[1])
    return flag1*flag2

我们非常熟悉运算符"+",不过目前来说,只有数值和数组是支持加法运算的。所以,对于list(dots[0])+[1]这种表示着实让人有些摸不到头脑。

这个含义其实是符合人类直觉的。列表内的元素个数是可变的,两个列表相加可以理解为两个列表衔接在一起。当然,元组并不支持这种运算。
例如

>>> [1,2,3]+[4]
[1, 2, 3, 4]
>>> (1,2,3)+(4)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: can only concatenate tuple (not "int") to tuple

通过加法运算符来衔接两个列表,实际上相当于新建了一个变量,需要开辟新的内存空间。好在对于初学者来说这样不容易出错。

numpy中,+、-、*、/这几个运算符表示对应位置元素的运算。如果想使用点乘等其他运算,需要调用numpy中的其他函数。

>>> np.array([1,2,3])*np.array([4,5,6])array([ 4, 10, 18])>>> np.array([1,2,3])+np.array([4,5,6])array([5, 7, 9])>>> np.array([1,2,3]).dot([4,5,6])          #.dot表点积32

所以, f l a g 1 = [ a , b , c ] ⋅ [ x 0 , y 0 , 1 ] = a x 0 + b y 0 + c 当然,我们也可以写成flag1 = abc[0]*dots[0][0]+abc[1]*dots[0][1]+c,只是看上去不太优雅。

然后,得到了flag1和flag2的值,如果二者异号,那么就可以断定,直线在两个点的中间。也就是说,只要flag1*flag2<0,即可说明直线与线段有焦点。

当然,从实际的角度出发,我们可以不去考虑光线通过镜片的端点或者平行镜片并掠过这种情况,从而只要函数返回值小于零,就认定二者相交,否则不相交。然而,从数学的角度出发,直线和线段之间可能存在三种关系:不相交、有一个交点、线段在直线上。

虽然这个理解没什么实际价值,但对于python学习来说,却是非常有意义的一个例子,代码如下,看懂了这个代码,那么就差不多可以算是一个初级的python程序员了。

def isCross(abc,dots):
    abc = np.array(abc)
    flags = [abc.dot(list(p)+[1]) for p in dots]    #for表达式
    poses,negs,zeros = [0,0,0]
    for flag in flags:                              #循环语句
        if flag > 0:                                #判断语句
            poses += 1
        elif flag <0:
            negs += 1
        else:
            zeros += 1
    return poses*negs+zeros

这个短短的程序涵盖了循环语句、判断语句以及for表达式的内容,前两者是最基础的编程知识,后者是python中非常亮眼的一种功能。

首先来认识一下运算符+=poses += 1即为poses = poses + 1,即相当于将poses+1赋值给poses。赋值前后flag1在内存中的位置发生了变化,也就是说flag1已经不是原来的flag1了。在这里,等号也并不能读成等于,而是读成被赋值为。即poses被赋值为poses+1。前面略有提过,双等号==才表示真正的相等。

然后来看判断语句,对于表达式if a A elif b B else C,我们按照人类的语法去读即可:如果a成立,则执行A,如果b成立,则执行B,否则的话执行C。在上述代码中,也可以很方便地读成:如果flag>0,那么poses被赋值为poses+1;如果flag<0,那么negs变成negs+1;否则的话zeros变成zeros+1。

这几个变量顾名思义,poses表示正数的个数,negs表示负数的个数,zeros表示0的个数。

然后来看[abc.dot(list(p)+[1]) for p in dots],我们首先使用一种没有陌生字符的方式书写:

#这是伪代码,假设dots中有n个变量,表示创建flag1、flag2一直到flagn一共n个变量。
flag1 = abc.dot(list(dots[0])+[1])
flag2 = abc.dot(list(dots[1])+[1])
...
flagn = abc.dot(list(dots[n])+[1])

这个表达式非常规律,这n个变量相当于是在dots中循环一遍,然后逐个赋值。for p in dots表示的就是将dots中的元素取出,赋值给p,然后再对p进行操作abc.dot(list(p)+[1]),最后将所有操作得到的值包裹再一个list中。

最后再记一下这个表达形式 [abc.dot(list(p)+[1]) for p in dots],以后会经常用到。

最后来看for flag in flags,即拿出flags中的所有元素,循环操作其下方的代码块。flags中的元素即为两个点分别带入 a x + b y + c 之后的值。那么对于这两个点来说,如果一正一负,则poses*negs=1*1=1,此时代表直线和线段有一交点,否则这个值便为0。当poses*negs==0时,则zeros的个数表示端点与直线相交的个数,zeros为0,表示无交点,为1,表示有一个端点在直线上,为2表示两个端点都在直线上。

射线排序

现在,我们可以判断某一个线段与一条直线是否有交点了,那么如果空间中有多个平面镜,光线所在的直线又与许多平面镜有交点,那么应该如何找到最近的那个呢?最简单的方法是分别求取这些点到光源的距离,距离越近相交越早。但这样会产生一个问题,即难以判定这个最近点是否在光的传播路径上,如果这个点在光源的后面,那就比较尴尬了。

所以,比较稳妥的方法是,按照射线的方向对所有点进行排序,那么光源后面的那个点,就是光线传播过程中的第一个交点。

刚刚我们在判定直线与线段的交点时,提到了直线族的概念。发现对于a、b取值相同的一组直线来说,其c值的大小与直线族的顺序是密切相关的。如Fig2-2所示。其 c 1 到 c 4依次递减。

这启发我们需要构建出一组和光线想垂直的直线族[a,b,~],则对于空间中任意一点 ( x , y )其所对应的 a x + b y 的值即能够对射线上的点进行排序。

考虑到a、b的值可能为0,所以不适合求倒数,故采用[b,-a]作为特征直线族,用以评价点在射线上的位置,最终代码如下。

def sortByRay(abc,points):
    ab = np.array([abc[1],-abc[0]])         #特征直线族
    pDict = {ab.dot(point):point for point in points}
    keyList = list(pDict.keys())            #将pDict的兼职转化成列表
    keyList.sort(reverse=True)              #对键列表进行排序
    return [pDict[key] for key in keyList]

这里又涉及到了一个新的数据类型,即字典。在理解字典之前,我们可以先回顾一下列表,我们可以把列表想象成一组值和自然序列的一一对应。对于列表test = [a,b,c,d]来说,有如下的对应关系{0:a,1:b,2:c,3:d},所以我们可以通过test[0]来索引atest[1]来索引b,以此类推。

那么,现在我不想用自然数来索引了,我想通过一个标记来索引,所以希望能够创建一个伪列表

dic = {3:5,4:15,12:22},于是我们可以对此列表进行索引dic[3]==5,dic[4]==15,dic[12]==22。

这个伪列表就可以由字典来实现。这种索引关系就叫做键值对,我们通过一个键来索引一个值。

对于表达式pDict = {ab.dot(point):point for point in points}

表示通过pointpoints进行遍历,即对于每个points中的point都进行ab.dot(point)这样的点乘操作。于是得到了由特征直线族得到的特征值与点之间的一一对应关系。

pDict.keys()即可提取出字典中所有的键,pDict.values()可以提取出字典中所有的值。在此我们将所有的键提取出来之后,再将其转化为列表。

然后即可调用列表的排序函数,将所有的值进行排序。即keyList.sort(),其中reverse参数默认为False,此时为降序。我们选择True,此时表示升序。

线弧关系

目前,我们已经能够精确地衡量射线与线段之间的关系了,接下来需要思考如何确定射线与透镜的位置关系。这一点当然也要从交点说起。

首先,弧是圆的一部分,所以如果一条直线与弧有交点,那么必然与这段弧所在的圆有交点。而直线与圆的交点判定相对来说还是非常容易的,只要圆心到直线的距离小于半径即可。

而直线和圆的交点问题则可以归结为求解方程组:

# abc为光线参数;cir为圆参数
# point为光源位置,当其为[]时表示不考虑
def getCrossCircle(abc=[1,1,1],cir=[0,0,2],point=()):
    c=np.array(cir[0:2]+[1]).dot(abc)
    a2b2 = abc[0]**2+abc[1]**2
    delt = a2b2*cir[2]**2-c**2
    if delt<0: return []          #如果无交点,返回空列表[]
    else: delt=np.sqrt(delt)      #否则求解判别式
    plusMinus = lambda a,b : np.array(set([a-b,a+b]))  #定义函数plusMinus
    yCross = plusMinus(-abc[1]*c,abc[0]*delt)/a2b2*[1,1]+cir[1]
    xCross = plusMinus(-abc[0]*c,-abc[1]*delt)/a2b2*[1,1]+cir[0]
    if point==[]:
        return [(xCross[i],yCross[i]) for i in [0,1]]
    yFlag = (yCross-point[1])*abc[0] >= 0
    xFlag = (point[0]-xCross)*abc[1] >= 0
    zFlag = np.abs(xFlag)+np.abs(yFlag) > 0
    flag = yFlag*xFlag*zFlag
    return [(xCross[i],yCross[i])
            for i in range(len(yFlag)) if flag[i]]

这段程序虽然短,但信息量还是很大的,而且使用了一个lambda表达式。

plusMinus = lambda a,b : np.array([a-b,a+b])定义了一个名为plusMius的函数

这个函数写成常规形式即为:

def plusMinus(a,b)
    return np.array([a-b,a+b])

需要注意的是,lambda表达式的后面只能有一个表达式,即只能定义一行的函数。

在这段代码中,我们还看到了一个陌生的运算符set,这也是python的一种数据类型,集合。和我们数学上认识的集合一样,在集合中,不允许出现相同的值。所以,如果b==0的话,那么set(a,a)=set(a),即起到了去重的作用。然后再通过np.array将集合转换成可计算的数组数据。

此外,这里引入了比较运算符。我们目前所提到的运算都是数值型的,但实际上我们所接触到的运算还有其他的类型。例如,当我们进行判断的时候,if delt<0:中,<也是一种运算符,代表比较,如果delt的确小于0,那么将返回一个值True,否则自然返回一个False。其中,True表示真,False表示假,这个是符合上过英语课的小学生的直觉的。

类似的运算符有>,<,>=,<=,==,!=,都可以望文生义地理解,其中!=表示不等于。这些都是比较运算符,其返回值为TrueFalseTrueFalse是一种不同于数值的数据类型,叫布尔型。

关系运算符不仅可以作用于数值类型,还可以作用于其他数据类型,一般情况下的使用方法都是符合直觉的。

>>> 1==2
False
>>> [3,3]==[3,3]
True
>>> 3>3
False
>>> 3>=3
True

然后我们再来看这个算法的逻辑,由于我们求解的是直线和圆的交点,而真实的光线却是射线,那么必然要考虑交点和光源的位置关系。

故代码

yFlag = (yCross-point[1])*abc[0] >= 0
xFlag = (point[0]-xCross)*abc[1] >= 0

分别定义了这两个判据xFlagyFlag。但是当二者同时为0时,说明此时 x = x 0 , y = y 0 x=x_0,y=y_0 x=x0​,y=y0​,此时交点即光源,故不能算作光线与圆有交点。所以又有判据zFlag

只有当这三个判据都满足时,我们所得到的值才是有效的,故总判据与这三个分判据是'与'的关系,所以写为flag = yFlag*xFlag*zFlag

此外,我们并不知道交点的个数,当判别式为0的时候,lambda表达式将只有一个值传出,这时的xCross和yCross中分别只有一个元素;如果判别式大于0,则分别有两个元素。这里又涉及到array的一个优良特性,当维度不想等的两个变量进行计算时,其会自动对低维数据进行合理的扩张,例如

>>> np.array([1,2,3])+4
array([5, 6, 7])
>>> np.array([[1],[2],[3]])+4
array([[5],
       [6],
       [7]])

最后,又出现了一个似曾相识的表达式

return [(xCross[i],yCross[i]) for i in range(len(yFlag)) if flag[i]]

这个表达式可以很容易地读出来:遍历flag,如果flag的值为真,则将对应的交点坐标放入列表,并返回有效的交点坐标。

这是对我们之前所使用的[... for ... in ...]的一种扩张,这种写法简洁而强大,非常推荐使用。

点弧关系

一般来说,在一个光学系统中很少出现一整个球,大部分情况下是由部分球面组成的各种透镜。所以,作为二维的光路系统,可能更需要被处理的是光线和圆弧的关系,尤其是和劣弧的关系。

判定点在劣弧上的方法有很多种,例如弧ACB上任意一点关于AB的对称点如果落入圆内,则为劣弧;如果落到园外,则为优弧;如果在圆上,说明AB是直径,弧ACB为半圆。

在此,我们选取另一种方式。如图所示,E为对于劣弧上任意一点,其与AB中点D的连线必然小于AB,否则即在优弧上。

所以,代码为:

def isOnArc(point,arc):
    arc = np.array(arc)
    dAB = 0.5*np.linalg.norm(arc[0]-arc[1])             #AB/2长度
    dCrossA = np.linalg.norm(0.5*(arc[0]+arc[1])-point) #ED长度
    return dAB > dCrossA

因此,当满足劣弧判定时,线圆交点即为线弧交点。

def getCrossArc(abc=[1,1,1],arc=[[0,1],[0,-1],[1,0]],point=[]):
    if  point == []:
        return []
    crossDict = {np.sqrt((p[0]-point[0])**2+(p[1]-point[1])**2):p
                 for p in getCrossCircle(abc,arc2cir(arc),point)
                 if (isOnArc(p,arc) and (p!=point))}
    if crossDict == {}:
        return []
    return  crossDict[min(crossDict.keys())]

机灵的同学其实很早就注意到,在定义函数的时候,其传入参数竟然被赋了值。这在python中并不值得大惊小怪,此时输入的值便是默认值。

此外,函数在被调用的时候,我们当然可以通过参数的顺序进行传参,但也可以使用变量名来对参数进行赋值,此时参数的顺序便没有意义了。

例如,

test1 = getCrossArc([1,1,1],[[0,1],[0,-1],[1,0]],(0,0)]
test2 = getCrossArc(abc=[1,1,1],point=(0,0),
                    arc=[[0,1],[0,-1],[1,0]])

上述两种写法都能得到正确的结果。

以上就是python光学仿真实现光线追迹之空间关系的详细内容,更多关于python实现光线追迹的空间关系的资料请关注我们其它相关文章!

(0)

相关推荐

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

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

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

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

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

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

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

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

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

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

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

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

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

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

  • Python光学仿真wxpython透镜演示系统计算与绘图

    目录 计算与绘图 计算与绘图 这里的计算主要包括两个部分,分别是通过滚动条的参数得到光学器件的特征,这一点此前已经备述.其二则是光在传播过程中所产生的各种行为,反射折射函数也都已经讲过了,需要注意的就是确定边界. def getRay(self): self.rays,self.abcs,self.dots = [[],[],[]] sDot = self.source #光源为第一个点 sRay = rp.getABC(self.sourceDict['theta'],sDot) inPoin

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

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

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

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

  • python光学仿真学习wxpython创建手速测试程序

    滚动条是什么大家自然都是知道的,可以非常直观地显示数据的变化,或者可以非常方便地改变某些数值. 此前在介绍按钮.静态文本.输入文本这三个控件时,相对来说比较乏味,所以这次我们采用需求引导的模式.假如想编写一个软件用来检测打字速度,同时能够非常直观地通过滚动条来显示出来,应该怎么写? 我们大致需要三个控件,文本输入控件用来输入文字:静态文本控件用于显示速度:滚动条用来动态地显示速度.同时,还需要知道系统的时间,总之,代码如下 import wx import time #时间模块 class te

  • Python光学仿真学习处理高斯光束分布图像

    目录 通过python处理光斑图像 1 相关包与图像读取 2 图像截取 3显示强度 4数据拟合 问题 通过python处理光斑图像 1 相关包与图像读取 首先需要科学计算必备包numpy和画图包matplotlib.pyplot,我们通过后者进行图像数据的读取. plt.imread读取图片之后为数据格式为numpy数组,可以通过成员函数astype将整型数据变成浮点型,有利于后期处理. plt.imshow将img的数据加载到窗口,plt.show()显示绘图窗口,默认显示为伪彩图. pyth

随机推荐