0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Python的自動(dòng)追蹤算法

新機(jī)器視覺(jué) ? 來(lái)源:新機(jī)器視覺(jué) ? 作者:新機(jī)器視覺(jué) ? 2022-10-19 14:17 ? 次閱讀

這里涉及攔截導(dǎo)彈的自動(dòng)跟蹤。最近,看到了一個(gè)挺有趣的自動(dòng)跟蹤算法,一個(gè)Python的簡(jiǎn)單模擬版本,分享給大家。

自動(dòng)追蹤算法,在我們?cè)O(shè)計(jì)2D射擊類(lèi)游戲時(shí)經(jīng)常會(huì)用到,這個(gè)聽(tīng)起來(lái)很高大上的東西,其實(shí)也并不是軍事學(xué)的專(zhuān)利,在數(shù)學(xué)上解決的話需要去解微分方程。

這個(gè)沒(méi)有點(diǎn)數(shù)學(xué)基礎(chǔ)是很難算出來(lái)的。但是我們有了計(jì)算機(jī)就不一樣了,依靠計(jì)算機(jī)極快速的運(yùn)算速度,我們利用微分的思想,加上一點(diǎn)簡(jiǎn)單的三角學(xué)知識(shí),就可以實(shí)現(xiàn)它。

好,話不多說(shuō),我們來(lái)看看它的算法原理,看圖:

be54c770-4f67-11ed-a3b6-dac502259ad0.png

由于待會(huì)要用pygame演示,他的坐標(biāo)系是y軸向下,所以這里我們也用y向下的坐標(biāo)系。

算法總的思想就是根據(jù)上圖,把時(shí)間t分割成足夠小的片段(比如1/1000,這個(gè)時(shí)間片越小越精確),每一個(gè)片段分別構(gòu)造如上三角形,計(jì)算出導(dǎo)彈下一個(gè)時(shí)間片走的方向(即∠a)和走的路程(即vt=|AC|),這時(shí)候目標(biāo)再在第二個(gè)時(shí)間片移動(dòng)了位置,這時(shí)剛才計(jì)算的C點(diǎn)又變成了第二個(gè)時(shí)間片的初始點(diǎn),這時(shí)再在第二個(gè)時(shí)間片上在C點(diǎn)和新的目標(biāo)點(diǎn)構(gòu)造三角形計(jì)算新的vt,然后進(jìn)入第三個(gè)時(shí)間片,如此反復(fù)即可。

假定導(dǎo)彈和目標(biāo)的初始狀態(tài)下坐標(biāo)分別是(x1,y1),(x,y),構(gòu)造出直角三角形ABE,這個(gè)三角形用來(lái)求∠a的正弦和余弦值,因?yàn)関t是自己設(shè)置的,我們需要計(jì)算A到C點(diǎn)x和y坐標(biāo)分別移動(dòng)了多少,移動(dòng)的值就是AD和CD的長(zhǎng)度,這兩個(gè)分別用vt乘cosa和sina即可。

計(jì)算sina和cosa,正弦對(duì)比斜,余弦鄰比斜,斜邊可以利用兩點(diǎn)距離公式計(jì)算出,即:

be7d0da2-4f67-11ed-a3b6-dac502259ad0.png

于是

be9584ea-4f67-11ed-a3b6-dac502259ad0.png

AC的長(zhǎng)度就是導(dǎo)彈的速度乘以時(shí)間即 |AC|=vt,然后即可計(jì)算出AD和CD的長(zhǎng)度,于是這一個(gè)時(shí)間片過(guò)去后,導(dǎo)彈應(yīng)該出現(xiàn)在新的位置C點(diǎn),他的坐標(biāo)就是老的點(diǎn)A的x增加AD和y減去CD。

于是,新的C點(diǎn)坐標(biāo)就是:

bebbfbac-4f67-11ed-a3b6-dac502259ad0.png

只要一直反復(fù)循環(huán)執(zhí)行這個(gè)操作即可,好吧,為了更形象,把第一個(gè)時(shí)間片和第二個(gè)時(shí)間片放在一起看看:

bed5636c-4f67-11ed-a3b6-dac502259ad0.png

第一個(gè)是時(shí)間片構(gòu)造出的三角形是ABE,經(jīng)過(guò)一個(gè)時(shí)間片后,目標(biāo)從B點(diǎn)走到了D點(diǎn),導(dǎo)彈此時(shí)在C點(diǎn),于是構(gòu)造新的三角形CDF,重復(fù)剛才的計(jì)算過(guò)程即可。

圖中的角∠b就是導(dǎo)彈需要旋轉(zhuǎn)的角度,現(xiàn)實(shí)中只需要每個(gè)時(shí)間片修正導(dǎo)彈的方向就可以了,具體怎么讓導(dǎo)彈改變方向,這就不是我們需要研究的問(wèn)題了。

好,由于最近在用Python的pygame庫(kù)制作小游戲玩,接下來(lái)我們就用pygame來(lái)演示一下這個(gè)效果,效果如下圖:

bef4b7c6-4f67-11ed-a3b6-dac502259ad0.gif

很簡(jiǎn)單的代碼如下:

importpygame,sys
frommathimport*
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#導(dǎo)彈的初始發(fā)射位置
velocity=800#導(dǎo)彈速度
time=1/1000#每個(gè)時(shí)間片的長(zhǎng)度
clock=pygame.time.Clock()
old_angle=0
whileTrue:
foreventinpygame.event.get():
ifevent.type==pygame.QUIT:
sys.exit()
clock.tick(300)
x,y=pygame.mouse.get_pos()#獲取鼠標(biāo)位置,鼠標(biāo)就是需要打擊的目標(biāo)
distance=sqrt(pow(x1-x,2)+pow(y1-y,2))#兩點(diǎn)距離公式
section=velocity*time#每個(gè)時(shí)間片需要移動(dòng)的距離
sina=(y1-y)/distance
cosa=(x-x1)/distance
angle=atan2(y-y1,x-x1)#兩點(diǎn)線段的弧度值
x1,y1=(x1+section*cosa,y1-section*sina)
d_angle=degrees(angle)#弧度轉(zhuǎn)角度
screen.blit(missile,(x1-missile.get_width(),y1-missile.get_height()/2))
dis_angle=d_angle-old_angle#dis_angle就是到下一個(gè)位置需要改變的角度
old_angle=d_angle#更新初始角度
pygame.display.update()

如果僅把導(dǎo)彈考慮為一個(gè)質(zhì)點(diǎn)的話,那么以上算法就已經(jīng)足矣,我沒(méi)有做導(dǎo)彈的旋轉(zhuǎn),因?yàn)橐粋€(gè)質(zhì)點(diǎn)也不分頭尾不需要旋轉(zhuǎn),當(dāng)然這前提得是你加載的導(dǎo)彈圖片很小的時(shí)候不旋轉(zhuǎn)看起來(lái)也沒(méi)什么問(wèn)題。

但是在pygame里面做旋轉(zhuǎn)并不是一件容易的事情(也可能是我無(wú)知),好吧我們先把圖片替換成一張矩形的,再加入旋轉(zhuǎn)函數(shù)看看效果如何。

bfc428a8-4f67-11ed-a3b6-dac502259ad0.png

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

因?yàn)閳D片的坐標(biāo)點(diǎn)是它的左上角的點(diǎn),所以如果我們想讓圖片的坐標(biāo)固定在箭頭尖點(diǎn),那么把圖片實(shí)際打印位置x減少圖片長(zhǎng)度,y減少一半寬度就行。

但是實(shí)際運(yùn)行效果并不好:

bef4b7c6-4f67-11ed-a3b6-dac502259ad0.gif

大致方向相同,但是圖片箭頭的尖點(diǎn)并沒(méi)有一直跟隨鼠標(biāo),這是為什么呢。經(jīng)過(guò)我的研究(就因?yàn)檫@個(gè)問(wèn)題沒(méi)解決一直沒(méi)發(fā)布),

我發(fā)現(xiàn)原來(lái)是這個(gè)圖旋轉(zhuǎn)的機(jī)制問(wèn)題,我們看看旋轉(zhuǎn)后的圖片變成什么樣了:

c0099c94-4f67-11ed-a3b6-dac502259ad0.png

旋轉(zhuǎn)后的圖片變成了藍(lán)色的那個(gè)范圍,根據(jù)旋轉(zhuǎn)角度的不同,所變成的圖片大小也不一樣,我們看旋轉(zhuǎn)90的情況:

c043b488-4f67-11ed-a3b6-dac502259ad0.pngc05f1c14-4f67-11ed-a3b6-dac502259ad0.png

我們發(fā)現(xiàn),旋轉(zhuǎn)后的圖片不僅面積變大了,導(dǎo)彈頭的位置也變了。那應(yīng)該怎么解決這個(gè)問(wèn)題呢?思路是,每一次旋轉(zhuǎn)圖片以后,求出旋轉(zhuǎn)圖的頭位置(圖中的綠色箭頭點(diǎn)),然后把綠圖的打印位置移動(dòng)一下,下,x,y分別移動(dòng)兩個(gè)頭的距離,就可以讓旋轉(zhuǎn)后的導(dǎo)彈頭對(duì)準(zhǔn)實(shí)際我們參與運(yùn)算的那個(gè)導(dǎo)彈頭的位置,移動(dòng)后應(yīng)該是這樣的:

c0923806-4f67-11ed-a3b6-dac502259ad0.png

這樣,兩個(gè)導(dǎo)彈頭的點(diǎn)就一致了。接下來(lái)我們分析求旋轉(zhuǎn)后的導(dǎo)彈頭的算法。根據(jù)旋轉(zhuǎn)角度的不同,旋轉(zhuǎn)角在不同象限參數(shù)不一樣,所以我們分為這四種情況

1,2象限:

c0b08996-4f67-11ed-a3b6-dac502259ad0.png

3,4象限,它的旋轉(zhuǎn)只有正負(fù)0—180,所以3,4象限就是負(fù)角。

c10d8a60-4f67-11ed-a3b6-dac502259ad0.png

顯示圖片的時(shí)候我們將他移動(dòng)。

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

這里的(x1-width,y1-height/2)其實(shí)才是上圖中的(x1,y1)。

所以最后我們加入相關(guān)算法代碼,效果就比較完美了。

c15e32f8-4f67-11ed-a3b6-dac502259ad0.gif

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

importpygame,sys
frommathimport*
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#導(dǎo)彈的初始發(fā)射位置
velocity=800#導(dǎo)彈速度
time=1/1000#每個(gè)時(shí)間片的長(zhǎng)度
clock=pygame.time.Clock()
A=()
B=()
C=()
whileTrue:
foreventinpygame.event.get():
ifevent.type==pygame.QUIT:
sys.exit()
clock.tick(300)
x,y=pygame.mouse.get_pos()#獲取鼠標(biāo)位置,鼠標(biāo)就是需要打擊的目標(biāo)
distance=sqrt(pow(x1-x,2)+pow(y1-y,2))#兩點(diǎn)距離公式
section=velocity*time#每個(gè)時(shí)間片需要移動(dòng)的距離
sina=(y1-y)/distance
cosa=(x-x1)/distance
angle=atan2(y-y1,x-x1)#兩點(diǎn)間線段的弧度值
fangle=degrees(angle)#弧度轉(zhuǎn)角度
x1,y1=(x1+section*cosa,y1-section*sina)
missiled=pygame.transform.rotate(missile,-(fangle))
if0<=-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))?#鼠標(biāo)用一個(gè)紅色*代替
????pygame.display.update()

最后

這是一個(gè)簡(jiǎn)單的,用 Python 實(shí)現(xiàn)的自動(dòng)跟蹤算法,真正的導(dǎo)彈攔截跟蹤算法要復(fù)雜很多。

審核編輯:彭靜
聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4697

    瀏覽量

    68085
  • 自動(dòng)跟蹤
    +關(guān)注

    關(guān)注

    0

    文章

    21

    瀏覽量

    9330
  • python
    +關(guān)注

    關(guān)注

    54

    文章

    4758

    瀏覽量

    84293

原文標(biāo)題:太強(qiáng)了,手?jǐn)]一款導(dǎo)彈跟蹤算法(Python版)

文章出處:【微信號(hào):vision263com,微信公眾號(hào):新機(jī)器視覺(jué)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    DRV2605:自動(dòng)諧振追蹤要點(diǎn)剖析

    DRV260x系列器件使用一種名為自動(dòng)諧振追蹤的特殊LRA控制算法自動(dòng)諧振追蹤使用LRA的反電動(dòng)勢(shì) (back-EMF) 來(lái)檢測(cè)和
    發(fā)表于 10-29 11:36 ?9017次閱讀
    DRV2605:<b class='flag-5'>自動(dòng)</b>諧振<b class='flag-5'>追蹤</b>要點(diǎn)剖析

    盤(pán)點(diǎn)史上最全的Python算法

    本文是一些機(jī)器人算法(特別是自動(dòng)導(dǎo)航算法)的Python代碼合集。其主要特點(diǎn)有以下三點(diǎn):選擇了在實(shí)踐中廣泛應(yīng)用的算法;依賴最少;容易閱讀,容
    的頭像 發(fā)表于 02-21 10:04 ?5982次閱讀

    Python的Apriori算法和FP-Growth算法是什么

    [源碼和文檔分享]基于Python實(shí)現(xiàn)的Apriori算法和FP-Growth算法的頻繁項(xiàng)集挖掘的研究與實(shí)現(xiàn)
    發(fā)表于 06-04 12:49

    追蹤算法自行車(chē)模型

    追蹤算法自行車(chē)模型自行車(chē)模型是對(duì)汽車(chē)運(yùn)動(dòng)描述的一種簡(jiǎn)化方法,其基于幾個(gè)假設(shè):忽略車(chē)輛垂直方向上的運(yùn)動(dòng)假設(shè)前面的兩個(gè)輪胎具有一致的角度和轉(zhuǎn)速,后面也如此,這樣車(chē)輛就可以簡(jiǎn)化成自行車(chē)結(jié)構(gòu)車(chē)輛運(yùn)動(dòng)
    發(fā)表于 08-17 08:58

    什么是純追蹤算法?

    什么是純追蹤算法?
    發(fā)表于 11-22 06:08

    什么是基追蹤算法?基于改進(jìn)基追蹤方法的信號(hào)去噪

    追蹤(basis pursuit)算法是一種用來(lái)求解未知參量L1范數(shù)最小化的等式約束問(wèn)題的算法。字典的構(gòu)造 對(duì)于觀測(cè)到的離散信號(hào)s∈H,H為Hilbert空間,給定H中的字典φ={φγ,γ∈Γ},其中Γ為指標(biāo)集,φγ為H中的基
    發(fā)表于 12-01 16:04 ?1.1w次閱讀
    什么是基<b class='flag-5'>追蹤</b><b class='flag-5'>算法</b>?基于改進(jìn)基<b class='flag-5'>追蹤</b>方法的信號(hào)去噪

    蟻群算法python編程實(shí)現(xiàn)

    本文主要介紹了Python編程實(shí)現(xiàn)蟻群算法詳解,涉及螞蟻算法的簡(jiǎn)介,主要原理及公式,以及Python中的實(shí)現(xiàn)代碼,具有一定參考價(jià)值。
    發(fā)表于 02-02 10:36 ?7412次閱讀
    蟻群<b class='flag-5'>算法</b><b class='flag-5'>python</b>編程實(shí)現(xiàn)

    光線追蹤算法匯總

    在基本光線追蹤算法中,只追蹤有限數(shù)目的光線。
    的頭像 發(fā)表于 05-11 15:58 ?1.7w次閱讀
    光線<b class='flag-5'>追蹤</b><b class='flag-5'>算法</b>匯總

    組合29個(gè)簡(jiǎn)單Python代碼塊,自動(dòng)發(fā)現(xiàn)新算法

    本文提出了一種基于演化算法的搜索策略,將其AAD中實(shí)現(xiàn)。AAD可以基于Python的子集作為語(yǔ)法結(jié)構(gòu),組合成復(fù)雜度相對(duì)較高的程序(循環(huán),嵌套塊,嵌套函數(shù)調(diào)用等),并生成可執(zhí)行的Python代碼。在本文中使用AAD來(lái)發(fā)現(xiàn)數(shù)組/向量
    的頭像 發(fā)表于 04-19 13:47 ?3451次閱讀

    Python模擬導(dǎo)彈自動(dòng)追蹤的代碼實(shí)例

    自動(dòng)追蹤算法,在我們制作射擊類(lèi)游戲時(shí)經(jīng)常會(huì)用到。這個(gè)聽(tīng)起來(lái)很高大上的東西,其實(shí)并不是軍事學(xué)的專(zhuān)利,從數(shù)學(xué)上來(lái)說(shuō)就是解微分方程,這個(gè)沒(méi)有點(diǎn)數(shù)學(xué)基礎(chǔ)是很難算出來(lái)的。但是我們有了計(jì)算機(jī)就不一樣了,依靠
    的頭像 發(fā)表于 06-13 16:35 ?1385次閱讀
    <b class='flag-5'>Python</b>模擬導(dǎo)彈<b class='flag-5'>自動(dòng)</b><b class='flag-5'>追蹤</b>的代碼實(shí)例

    Python實(shí)現(xiàn)所有算法-基本牛頓法

    Python實(shí)現(xiàn)所有算法-二分法 Python實(shí)現(xiàn)所有算法-力系統(tǒng)是否靜態(tài)平衡 Python實(shí)現(xiàn)所有算法
    的頭像 發(fā)表于 07-13 10:40 ?1572次閱讀

    Python實(shí)現(xiàn)自動(dòng)駕駛

    今天來(lái)一個(gè)好玩一點(diǎn)的,汽車(chē)已經(jīng)能夠自動(dòng)駕駛了,Python怎么能沒(méi)有呢?這不,必須安排上。 一、安裝環(huán)境 gym是用于開(kāi)發(fā)和比較強(qiáng)化學(xué)習(xí)算法的工具包,在python中安裝gym庫(kù)和其中
    發(fā)表于 06-06 10:43 ?1次下載
    <b class='flag-5'>Python</b>實(shí)現(xiàn)<b class='flag-5'>自動(dòng)</b>駕駛

    [源代碼]Python算法詳解

    [源代碼]Python算法詳解[源代碼]Python算法詳解
    發(fā)表于 06-06 17:50 ?0次下載

    如何使用Python進(jìn)行圖像識(shí)別的自動(dòng)學(xué)習(xí)自動(dòng)訓(xùn)練?

    如何使用Python進(jìn)行圖像識(shí)別的自動(dòng)學(xué)習(xí)自動(dòng)訓(xùn)練? 使用Python進(jìn)行圖像識(shí)別的自動(dòng)學(xué)習(xí)和自動(dòng)
    的頭像 發(fā)表于 01-12 16:06 ?500次閱讀

    Python建模算法與應(yīng)用

    上成為理想的腳本語(yǔ)言,特別適用于快速的應(yīng)用程序開(kāi)發(fā)。本文將詳細(xì)介紹Python在建模算法中的應(yīng)用,包括常見(jiàn)的建模算法Python在建模中的優(yōu)勢(shì)、常用庫(kù)以及實(shí)際案例。
    的頭像 發(fā)表于 07-24 10:41 ?369次閱讀