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

 更新时间:2021年10月20日 00:00  点击:1247 作者:微小冷

折射与反射

光线与光学元件相互作用,无非只有两件事,反射和透射。而就目前看来,我们所常用的光学元件,也无非有两种表面,即平面和球面,二维化之后就简化成了射线与线段,射线与劣弧的关系。

平面反射

无论从哪个角度来看,平面的反射折射都要比球面更简单,而反射问题要比折射问题更简单,所以,我们首先处理平面的反射问题。

反射定律即入射角等于反射角,心念及此,最为循规蹈矩的思路必然是先找到入射光线和平面的夹角,然后用这个夹角和平面(在二维空间中是一条直线)在空间中的斜率,由这个斜率与入射角得到出射光的斜率,然后就可以得到出射光的方程。

这个方法的问题是需要反复使用三角函数和反三角函数,而三角函数和反三角函数并非严格意义上的互为相反,所以在传参的过程中,可能会遇到一些麻烦。

相对来说,比较不容易出错的方法是,寻找入射点关于法线的对称点,那么这个对称点与交点的连线,便是出射光的方程。

在这里插入图片描述

在这里插入图片描述

平面折射

折射与反射的思路如出一辙,最原始的想法仍旧是获取入射角,然后根据折射定律求出射角,然后再按照出射角解出出射光的表达式。这个思路的难点仍旧在三角函数与反函数的转化上。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

至此,我们发现折射与反射在表达形式上是相通的,如果令入射点关于法线做垂线,垂足为C,约定这条垂线与出射光线的交点为出射点B,那么出射点到垂足的距离BC与入射点到垂足的距离AC之间是满足比例关系的。当入射光线和反射光线的折射率相等时,这个比例为1,否则比例为 λ \lambda λ。

我们还能发现,这个 λ \lambda λ不一定有解,因为分母中有一个根号表达式,当内部的值小于0时,自然无解。这与我们的物理直觉是符合的,即并不是所有的入射光线都有折射光线,当折射光线消失的时候,就发生了全反射。

所以,当务之急是根据入射点找垂足,易得

在这里插入图片描述

那么对于我们所熟知的折射问题,即可令入射点关于反射平面做一次对称,再关于发现做一次定比延长线的对称,即可得到出射点。

python实现

至此,我们已经完全建立了一套反射与折射的关系,代码如下:

#得到点关于直线的对称点,k为比例系数
def getSymDot(point,line,k=1):
    return tuple((np.array(getPedal(point,line))*(1+k)-point)/k)
#得到直线的垂足
def getPedal(point,line):
    a,b,c=line
    x0,y0 = point
    y1 = (a**2*y0-a*b*x0-b*c)/(a**2+b**2)
    x1 = (b**2*x0-a*b*y0-a*c)/(a**2+b**2)
    return (x1,y1)

函数getSymDot即通过输入点和线来求解对称点,其思路是把点关于线对称的问题转化为点关于垂足对称的问题。所以引用了getPedal函数,这个函数通过输入一点和线来返回过点做线的垂线所得到的垂足。

所有代码都是对上述数学公式的简单复现。

def cataDioLine(abc=[1,-1,1],line=[2,-1,1],
                sPoint=[],cross=[],n1=1,n2=1.5):
    normal = [-line[1],line[0],line[1]*cross[0]-line[0]*cross[1]]#法线
    flecDot = getSymDot(sPoint,normal)
    flec=getABC([cross,flecDot])
    dPara = np.sqrt(line[0]**2+line[1]**2)
    dNormal = np.abs(np.array(normal).dot(list(sPoint)+[1]))/dPara#到法线距离
    dPane = np.abs(np.array(line).dot(list(sPoint)+[1]))/dPara#到反射面距离
    if dNormal == 0:
        return flec,abc
    delt = (n2/n1)**2*(1+(dPane/dNormal)**2)-1#判定全反射
    if delt>0:
        k =dPane/dNormal/np.sqrt(delt)
        fracDot = getSymDot(sPoint,normal,k)
        fracDot = getSymDot(fracDot,line)
        frac = getABC([cross,fracDot])
        return flec,frac
    return flec,[0,0,0]

函数cataDioLine则是反射折射的实现函数。注意,在此引入的getABC并不是此前定义的通过点和角度求表达式的函数,而是通过两点转[a,b,c]的函数。

那么我们是否可以写一个同名函数来实现不同的功能呢?很遗憾的是,Python不支持函数的重载,所以只能将同名函数封装在一起:

def getABC(*par):
    if len(par)==1:     #此时传入的参数为点对dots=[(x0,y0),(x1,y1)]
        dots = par[0]
        abc = [dots[1][1]-dots[0][1],
                dots[0][0]-dots[1][0], 
                -np.linalg.det(dots)]
        return np.array(abc)/(np.sqrt(abc[0]**2+abc[1]**2))
    elif len(par)==2:   #此时传入的参数为点和角度(x0,y0),theta
        theta,sPoint = par
        a,b = [np.sin(theta),-np.cos(theta)]
        c = -(a*sPoint[0]+b*sPoint[1])
        return [a,b,c]

看到输入参数(*par),我们很多人可能会产生某些不是很美妙的联想,但不要兴奋,这只是python的一种传参方式。(*args)表示将传入的参数组成一个列表args;(**kargs)表示将传入的参数组成一个字典kargs。

弧面问题

光线在弧面上的反射问题,是典型的那种看似复杂实则简单的纸老虎问题,简单到我们只要找到法线就能轻松地转化为平面问题。

所以,问题被简单地转化为求解圆的切线问题——这个切线即反射平面。由于数学过程过于简单,就不写公式了,读者可以试着看代码反推公式。

#获取过交点的圆弧的切线
def getTangent(corss=[0,1],circle=[0,0,1]):
    a = corss[0]-circle[0]
    b = corss[1]-circle[1]
    c = -a*corss[0]-b*corss[1]
    return [a,b,c]

至此,我们就可以得到一个完整的折射反射问题的求解方案:

#光在直线或弧线表面的反折射
def cataDio(abc=[1,-1,1],dots=[(0,2),(2,2)],
            sPoint=[-2,-1],n1=1,n2=1.5):
    cross = getCross(abc,dots,sPoint)           #获取交点
    if cross == []:
        return [],[],[]
    if len(dots)==3:
        line = getTangent(cross,arc2cir(dots))  #圆上切线
    elif len(dots)==2:
        line = getABC(dots)
    flec,frac = cataDioLine(abc,line,sPoint,cross,n1,n2)
    return cross,flec,frac

当然,这里的getCross也需要重新写成不仅适合直线,而且适合弧线的形式:

def getCross(abc=[1,-1,0],dots=[[0,-1],[0,1],[0.5,0]],point=[]):
    if len(dots)==3:
        return getCrossArc(abc,dots,point)
    if len(dots)==2:
        return getCrossDots(abc,dots,point)

这时我们发现用两个点表示线段,三个点表示弧线还是比较舒服的一种做法,至少二者在表达形式上的统一似乎能为我们带来某种内心的愉悦。

以上就是python光学仿真实现光线追迹折射与反射的实现的详细内容,更多关于python光线追迹的资料请关注猪先飞其它相关文章!

原文出处:https://blog.csdn.net/m0_37816922/article/details/97885015

[!--infotagslink--]

相关文章

  • 关于C#基础知识回顾--反射(一)

    其实说白了,反射就是能知道我们未知类型的类型信息这么一个东西.没什么神秘可讲!反射的核心是System.Type。System.Type包含了很多属性和方法,使用这些属性和方法可以在运行时得到类型信息...2020-06-25
  • 关于C#反射 你需要知道的

    这篇文章主要介绍了C#反射的相关知识,文中讲解的非常详细,代码帮助大家更好的参考学习,感兴趣的朋友可以了解下...2020-06-25
  • C# 通过反射获取类型的字段值及给字段赋值的操作

    这篇文章主要介绍了C# 通过反射获取类型的字段值及给字段赋值的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-01-19
  • .NET/C#如何使用反射注册事件详解

    反射是.NET中的重要机制,下面这篇文章主要给大家介绍了关于.NET/C#如何使用反射注册事件的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧...2021-09-22
  • asp.net反射简单应用实例

    这篇文章主要介绍了asp.net反射简单应用,结合完整实例形式分析了asp.net反射的原理与简单使用方法,需要的朋友可以参考下...2021-09-22
  • php面向对象之反射功能与用法分析

    这篇文章主要介绍了php面向对象之反射功能与用法,结合实例形式简单分析了php5面向对象反射的概念及具体用法,需要的朋友可以参考下...2017-04-03
  • 深入解析Java反射之基础篇

    JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,这篇文章主要给大家介绍了关于Java反射之基础篇的相关资料,需要的朋友可以参考下...2021-11-18
  • 关于C#基础知识回顾--反射(二)

    其实说白了,反射就是能知道我们未知类型的类型信息这么一个东西.没什么神秘可讲!反射的核心是System.Type。System.Type包含了很多属性和方法,使用这些属性和方法可以在运行时得到类型信息...2020-06-25
  • C++反射的一种实现方法详解

    这篇文章主要给大家介绍了关于C++反射的一种实现方法,文中通过示例代码介绍的非常详细,对大家学习或者使用C++具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧...2020-04-25
  • C#泛型和反射实例解析

    这篇文章主要介绍了C#泛型和反射实例解析,对于C#初学者理解泛型和反射有很好的帮助借鉴作用,需要的朋友可以参考下...2020-06-25
  • 前端JavaScript中的反射和代理

    这篇文章主要介绍的是前端JavaScript中的反射和代理,本文主要围绕JavaScript中的内置Reflect、JavaScript中的内置Proxy、Proxy实现观察者模式三个话题展开文章内容,需要的朋友可以参一考下...2021-10-22
  • Asp.net中使用DapperExtensions和反射来实现一个通用搜索

    这篇文章主要介绍了Asp.net中使用DapperExtensions和反射来实现一个通用搜索功能,非常不错,具有参考解决价值,需要的朋友可以参考下...2021-09-22
  • unity shader实现玻璃折射效果

    这篇文章主要为大家详细介绍了unity shader实现玻璃折射效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-06-25
  • java 利用反射获取内部类静态成员变量的值操作

    这篇文章主要介绍了java 利用反射获取内部类静态成员变量的值操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-12-04
  • 深入反射生成数组的详解

    本篇文章是对反射生成数组进行了详细的分析介绍,需要的朋友参考下...2020-06-25
  • 用户控件(ASCX)向网页(ASPX)传值使用反射实现

    用户控件向网页传递值,网上的方法有很多,本文尝试一下使用反射来实现,感兴趣的朋友可以参考下哈,希望可以帮助到你...2021-09-22
  • C# 反射与dynamic最佳组合示例代码

    这篇文章主要介绍了C# 反射与dynamic最佳组合示例代码,需要的朋友可以参考下...2020-06-25
  • C#反射实例学习及注意内容

    C#反射的入门学习首先要明白C#反射提供了封装程序集、模块和类型的对象等等需要的朋友可以参考下...2021-09-22
  • C#通过反射创建自定义泛型

    这篇文章主要介绍了C#通过反射创建自定义泛型的方法,需要的朋友可以参考下...2020-06-25
  • Python光学仿真数值分析求解波动方程绘制波包变化图

    这篇文章主要为大家介绍了Python光学仿真通过数值分析求解波动方程并绘制波包变化图的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助...2021-10-20