Python实现简单的"导弹" 自动追踪原理解析

 更新时间:2021年03月16日 08:36:57   作者:酸菜鱼学Python  
这篇文章主要介绍了Python实现简单的"导弹" 自动追踪原理解析,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

自动追踪算法,在我们设计2D射击类游戏时经常会用到,这个听起来很高大上的东西,其实也并不是军事学的专利,在数学上解决的话需要去解微分方程,

这个没有点数学基础是很难算出来的。但是我们有了计算机就不一样了,依靠计算机极快速的运算速度,我们利用微分的思想,加上一点简单的三角学知识,就可以实现它。

好,话不多说,我们来看看它的算法原理,看图:

由于待会要用pygame演示,他的坐标系是y轴向下,所以这里我们也用y向下的坐标系。

算法总的思想就是根据上图,把时间t分割成足够小的片段(比如1/1000,这个时间片越小越精确),每一个片段分别构造如上三角形,计算出导弹下一个时间片走的方向(即∠a)和走的路程(即vt=|AC|),这时候目标再在第二个时间片移动了位置,这时刚才计算的C点又变成了第二个时间片的初始点,这时再在第二个时间片上在C点和新的目标点构造三角形计算新的vt,然后进入第三个时间片,如此反复即可。

假定导弹和目标的初始状态下坐标分别是(x1,y1),(x,y),构造出直角三角形ABE,这个三角形用来求∠a的正弦和余弦值,因为vt是自己设置的,我们需要计算A到C点x和y坐标分别移动了多少,移动的值就是AD和CD的长度,这两个分别用vt乘cosa和sina即可。

计算sina和cosa,正弦对比斜,余弦邻比斜,斜边可以利用两点距离公式计算出,即:

于是

AC的长度就是导弹的速度乘以时间即 |AC|=vt,然后即可计算出AD和CD的长度,于是这一个时间片过去后,导弹应该出现在新的位置C点,他的坐标就是老的点A的x增加AD和y减去CD。

于是,新的C点坐标就是:

只要一直反复循环执行这个操作即可,好吧,为了更形象,把第一个时间片和第二个时间片放在一起看看:

第一个是时间片构造出的三角形是ABE,经过一个时间片后,目标从B点走到了D点,导弹此时在C点,于是构造新的三角形CDF,重复刚才的计算过程即可,图中的角∠b就是导弹需要旋转的角度,现实中只需要每个时间片修正导弹的方向就可以了,具体怎么让导弹改变方向,这就不是我们需要研究的问题了

好,由于最近在用Python的pygame库制作小游戏玩,接下来我们就用pygame来演示一下这个效果,效果如下图:

很简单的代码如下:

import pygame,sys
from math import *
pygame.init()
screen=pygame.display.set_mode((800,700),0,32)
missile=pygame.image.load('element/red_pointer.png').convert_alpha()
x1,y1=100,600      #导弹的初始发射位置
velocity=800      #导弹速度
time=1/1000       #每个时间片的长度
clock=pygame.time.Clock()
old_angle=0
while True:
  for event in pygame.event.get():
    if event.type==pygame.QUIT:
      sys.exit()
  clock.tick(300)
  x,y=pygame.mouse.get_pos()     #获取鼠标位置,鼠标就是需要打击的目标
  distance=sqrt(pow(x1-x,2)+pow(y1-y,2))   #两点距离公式
  section=velocity*time        #每个时间片需要移动的距离
  sina=(y1-y)/distance
  cosa=(x-x1)/distance
  angle=atan2(y-y1,x-x1)       #两点线段的弧度值
  x1,y1=(x1+section*cosa,y1-section*sina)
  d_angle = degrees(angle)    #弧度转角度
  screen.blit(missile, (x1-missile.get_width(), y1-missile.get_height()/2))
  dis_angle=d_angle-old_angle     #dis_angle就是到下一个位置需要改变的角度
  old_angle=d_angle          #更新初始角度
  pygame.display.update()

如果仅把导弹考虑为一个质点的话,那么以上算法就已经足矣,我没有做导弹的旋转,因为一个质点也不分头尾不需要旋转,当然这前提得是你加载的导弹图片很小的时候不旋转看起来也没什么问题。但是在pygame里面做旋转并不是一件容易的事情(也可能是我无知),好吧我们先把图片替换成一张矩形的,再加入旋转函数看看效果如何

missiled = pygame.transform.rotate(missile, -(d_angle))
screen.blit(missiled, (x1-missile.get_width(), y1-missile.get_height()/2))

因为图片的坐标点是它的左上角的点,所以如果我们想让图片的坐标固定在箭头尖点,那么把图片实际打印位置x减少图片长度,y减少一半宽度就行。

但是实际运行效果并不好:

大致方向相同,但是图片箭头的尖点并没有一直跟随鼠标,这是为什么呢。经过我的研究(就因为这个问题没解决一直没发布),

我发现原来是这个图旋转的机制问题,我们看看旋转后的图片变成什么样了:

旋转后的图片变成了蓝色的那个范围,根据旋转角度的不同,所变成的图片大小也不一样,我们看旋转90的情况

我们发现,旋转后的图片不仅面积变大了,导弹头的位置也变了。那应该怎么解决这个问题呢?思路是,每一次旋转图片以后,求出旋转图的头位置(图中的绿色箭头点),然后把绿图的打印位置移动一下,下,x,y分别移动两个头的距离,就可以让旋转后的导弹头对准实际我们参与运算的那个导弹头的位置,移动后应该是这样的:

这样,两个导弹头的点就一致了。接下来我们分析求旋转后的导弹头的算法。根据旋转角度的不同,旋转角在不同象限参数不一样,所以我们分为这四种情况

1,2象限

3,4象限,它的旋转只有正负0—180,所以3,4象限就是负角

显示图片的时候我们将他移动

screen.blit(missiled, (x1-width+(x1-C[0]),y1-height/2+(y1-C[1])))

这里的(x1-width,y1-height/2)其实才是上图中的(x1,y1)

所以最后我们加入相关算法代码,效果就比较完美了

大功告成,最后附上全部的算法代码

import pygame,sys
from math import *
pygame.init()
font1=pygame.font.SysFont('microsoftyaheimicrosoftyaheiui',23)
textc=font1.render('*',True,(250,0,0))
screen=pygame.display.set_mode((800,700),0,32)
missile=pygame.image.load('element/rect1.png').convert_alpha()
height=missile.get_height()
width=missile.get_width()
pygame.mouse.set_visible(0)
x1,y1=100,600      #导弹的初始发射位置
velocity=800      #导弹速度
time=1/1000       #每个时间片的长度
clock=pygame.time.Clock()
A=()
B=()
C=()
while True:
  for event in pygame.event.get():
    if event.type==pygame.QUIT:
      sys.exit()
  clock.tick(300)
  x,y=pygame.mouse.get_pos()     #获取鼠标位置,鼠标就是需要打击的目标
  distance=sqrt(pow(x1-x,2)+pow(y1-y,2))   #两点距离公式
  section=velocity*time        #每个时间片需要移动的距离
  sina=(y1-y)/distance
  cosa=(x-x1)/distance
  angle=atan2(y-y1,x-x1)       #两点间线段的弧度值
  fangle=degrees(angle)        #弧度转角度
  x1,y1=(x1+section*cosa,y1-section*sina)
  missiled=pygame.transform.rotate(missile,-(fangle))
  if 0<=-fangle<=90:
    A=(width*cosa+x1-width,y1-height/2)
    B=(A[0]+height*sina,A[1]+height*cosa)

  if 90<-fangle<=180:
    A = (x1 - width, y1 - height/2+height*(-cosa))
    B = (x1 - width+height*sina, y1 - height/2)

  if -90<=-fangle<0:
    A = (x1 - width+missiled.get_width(), y1 - height/2+missiled.get_height()-height*cosa)
    B = (A[0]+height*sina, y1 - height/2+missiled.get_height())

  if -180<-fangle<-90:
    A = (x1-width-height*sina, y1 - height/2+missiled.get_height())
    B = (x1 - width,A[1]+height*cosa )

  C = ((A[0] + B[0]) / 2, (A[1] + B[1]) / 2)

  screen.fill((0,0,0))
  screen.blit(missiled, (x1-width+(x1-C[0]),y1-height/2+(y1-C[1])))
  screen.blit(textc, (x,y)) #鼠标用一个红色*代替
  pygame.display.update()

到此这篇关于Python实现简单的"导弹" 自动追踪原理解析的文章就介绍到这了,更多相关python自动追踪内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 常用的Python代码调试工具总结

    常用的Python代码调试工具总结

    今天给大家带来的是关于Python的相关知识,文章围绕着Python代码调试工具展开,文中有非常详细的介绍及代码示例,需要的朋友可以参考下
    2021-06-06
  • 对python打乱数据集中X,y标签对的方法详解

    对python打乱数据集中X,y标签对的方法详解

    今天就为大家分享一篇对python打乱数据集中X,y标签对的方法详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-12-12
  • Python实现扫描指定目录下的子目录及文件的方法

    Python实现扫描指定目录下的子目录及文件的方法

    这篇文章主要介绍了Python实现扫描指定目录下的子目录及文件的方法,需要的朋友可以参考下
    2014-07-07
  • python获取url的返回信息方法

    python获取url的返回信息方法

    今天小编就为大家分享一篇python获取url的返回信息方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-12-12
  • Python 高效编程技巧分享

    Python 高效编程技巧分享

    工作中经常要处理各种各样的数据,遇到项目赶进度的时候自己写函数容易浪费时间。Python 中有很多内置函数帮你提高工作效率。
    2020-09-09
  • python中os.remove()用法及注意事项

    python中os.remove()用法及注意事项

    在本篇内容里小编给大家分享的是一篇关于python中os.remove()用法及注意事项,有需要的朋友们可以跟着学习下。
    2021-01-01
  • python typing模块--类型提示支持

    python typing模块--类型提示支持

    这篇文章主要介绍python typing模块类型提示支持, typing 模块只有在python3.5以上的版本中才可以使用,pycharm目前支持typing检查,下面进入文章一起了解详细内容吧
    2021-10-10
  • python爬虫智能翻页批量下载文件的实例详解

    python爬虫智能翻页批量下载文件的实例详解

    在本篇文章里小编给大家整理的是一篇关于python爬虫智能翻页批量下载文件的实例详解内容,有兴趣的朋友们可以学习下。
    2021-02-02
  • Python变量作用域LEGB用法解析

    Python变量作用域LEGB用法解析

    这篇文章主要介绍了Python变量作用域LEGB用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-02-02
  • keras分类之二分类实例(Cat and dog)

    keras分类之二分类实例(Cat and dog)

    这篇文章主要介绍了keras分类之二分类实例(Cat and dog),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-07-07

最新评论