|
马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。
您需要 登录 才可以下载或查看,没有帐号?免费注册
x
本帖最后由 navebayes 于 2023-12-26 12:02 编辑
* A+ V' ?) q. @3 C
' d2 j0 m: i( a最近正在玩个游戏,也算是国产之光绯月仙行录,这个游戏哪里都好就是bug太多,并且作者过于摆烂,以至于有很多玩家都认为这个游戏就是故意拖着吃赞助的(bushi)
! J, u! b" t7 {6 Z- u言归正传,在游玩这个游戏的过程中,我在一个评论区里看到这样一段话:自从玩了绯月之后,对于其他RPG游戏都看不上眼了,因为这个游戏独创了自动寻路的功能,可以说是RPG类游戏的里程碑式壮举。6 Q* a( A: n: C; h/ X8 M(欢迎访问老王论坛:laowang.vip)
我对此感到好奇,因为从前从来没有游玩RPG类游戏的经验,但我学过一点点算法,于是我打算用一些浅显易懂的方式说说自动寻路这样一个功能的实现。7 n; l/ \) S* n3 g2 {, S! P(欢迎访问老王论坛:laowang.vip)
2 P3 V8 o4 m/ `; ?(欢迎访问老王论坛:laowang.vip)
主流的寻路算法:深度优先,广度优先,Dijkstra,A* 等,我这里主要讲讲后两种。
, x# W- P+ V$ j' ?5 `* N5 R- h" k; }0 `6 q8 ]+ R, L(欢迎访问老王论坛:laowang.vip)
Dijkstra算法:这个算法是目前很多地图软件都在使用的算法,采用OPEN,CLOSE表的方式实现寻路功能。
# K3 u0 `' w" ~) P! G0 X3 a" S; Y 创建两个表,OPEN, CLOSE。
! v) ]* ~/ l1 T' ~! IOPEN表保存所有已生成而未考察的节点,CLOSED表中记录已访问过的节点。
+ K- b' e) _# n1. 访问路网中里起始点最近且没有被检查过的点,把这个点放入OPEN组中等待检查。
- h3 ]% O# k- z+ Q; u2 e% s0 }' Y& C4 }) S2. 从OPEN表中找出距起始点最近的点,找出这个点的所有子节点,把这个点放到CLOSE表中。
0 @5 F V7 c9 a1 r& y6 @. G3 [3. 遍历考察这个点的子节点。求出这些子节点距起始点的距离值,放子节点到OPEN表中。1 X5 [' t8 C6 x* q ]6 X, @(欢迎访问老王论坛:laowang.vip)
4. 重复2,3,步。直到OPEN表为空,或找到目标点。7 T# D: Y4 r9 [3 v, _0 M(欢迎访问老王论坛:laowang.vip)
' b, t, i) V9 I2 N) b. _- e实际写代码还是比较占行数的,直接给出链接如下。% s3 j7 Q, E0 s% @4 O3 P(欢迎访问老王论坛:laowang.vip)
参考:https://blog.csdn.net/YiYeZhiNian/article/details/122217450
7 P, ?( Z# F: S% {$ _) m9 ^. a$ ]" M0 u/ \$ G* F; n( U(欢迎访问老王论坛:laowang.vip)
用这个算法,我写过一个课程作业,具体就是对于各个城市的地铁最短路径规划,大致还是比较成功的。先说说个人对于Dijkstra算法设计地铁线路规划:: B' q* g5 X0 s% V, }! R9 ?. N(欢迎访问老王论坛:laowang.vip)
1.首先用爬虫爬到该城市的地铁网络,包含站点名称,该站点经纬度和线路图,并生成excel表格。用该excel表格生成pickle文件,方便直接调用。& s) Y/ l' z9 S( W(欢迎访问老王论坛:laowang.vip)
2.注册高德地图开发者账号(该功能需要实名),得到一个key用于调用api(每日上限100次,超过付费,我调试到后面不让我调试了...)2 G( p" G8 ~# C; \, E9 z(欢迎访问老王论坛:laowang.vip)
3.输入始发地和目的地,并通过api返回两个位置的经纬度坐标。& w0 N" z4 p6 B(欢迎访问老王论坛:laowang.vip)
4.比较两地理位置之间的最近地铁站,并根据Dijkstra算法实现路径规划。 e( ~7 J9 G% O h" L(欢迎访问老王论坛:laowang.vip)
5.Dijkstra算法的本质就是不断选择新顶点并更新已处理表,并将他和邻居节点进行比对,当所有节点都被处理后即为最短路径7 W5 n( s; Y* z) D(欢迎访问老王论坛:laowang.vip)
至此,已初步完成具体工作流程。9 t/ U1 `8 V0 c! Q" t# T* W(欢迎访问老王论坛:laowang.vip)
- def get_nearest_subway(data,longitude1,latitude1):
- `) I$ ]/ I6 G! I - #找最近的地铁站2 u( a. O, m# p9 L! d: d(欢迎访问老王论坛:laowang.vip)
- longitude1=float(longitude1)2 `& C* O% x8 k0 z(欢迎访问老王论坛:laowang.vip)
- latitude1=float(latitude1)& |4 O& s( N; v# L |9 i" j(欢迎访问老王论坛:laowang.vip)
- distance=float('inf'): ]; J+ \& @' Y# T" P! V(欢迎访问老王论坛:laowang.vip)
- nearest_subway=None# g1 U1 q. G0 x. i' `$ `(欢迎访问老王论坛:laowang.vip)
- for i in range(data.shape[0]):9 t! e. p: f: |+ y \8 l1 R$ y(欢迎访问老王论坛:laowang.vip)
- site1=data.iloc[i]['name'] + n/ t) R e( A$ [" V(欢迎访问老王论坛:laowang.vip)
- longitude=float(data.iloc[i]['longitude'])
9 D+ a- H9 ? s" f - latitude=float(data.iloc[i]['latitude']) #分别将经纬度代入,计算与目标之间的欧氏距离' [4 o" z; A+ Q) ]5 l0 g(欢迎访问老王论坛:laowang.vip)
- temp=geodesic((latitude1,longitude1), (latitude,longitude)).m #temp对其遍历即可,这里对比各个地铁站的欧氏距离" H+ e( n% D# `, F6 x) o(欢迎访问老王论坛:laowang.vip)
- if temp<distance:" F2 t& Y( J4 B% b3 E* ^ y(欢迎访问老王论坛:laowang.vip)
- distance=temp
, P4 Z4 k; N# p. } - nearest_subway=site1; I" `) _: O( J(欢迎访问老王论坛:laowang.vip)
- return nearest_subway
复制代码- def subway_line(start,end): #创建点之间的距离
3 O* A# ]5 t+ E4 s" ~ - file=open('graph.pkl','rb')
! M, d& W \* o+ B - graph=pickle.load(file) #现在我们有了各个地铁站之间的距离存储在graph2 e x" x5 r+ C0 q7 F* H4 c(欢迎访问老王论坛:laowang.vip)
- costs={} #创建节点的开销表,cost是指从start到该节点的距离
( ]4 S$ @+ T( o1 _# B - parents={}& ?5 d) n! H% Y% J1 W5 ?6 R+ c(欢迎访问老王论坛:laowang.vip)
- parents[end]=None
/ c9 h; e0 B1 Q1 \ - for node in graph[start].keys():9 F* R( c- O S; J8 C9 l(欢迎访问老王论坛:laowang.vip)
- costs[node]=float(graph[start][node])$ M% {; B9 N j2 B! e(欢迎访问老王论坛:laowang.vip)
- parents[node]=start. s1 w2 K- B F" ^, `: F9 b) b; ](欢迎访问老王论坛:laowang.vip)
- costs[end]=float('inf') #终点到起始点距离为无穷大6 ?: y; N7 C2 Y; G(欢迎访问老王论坛:laowang.vip)
- processed=[] #记录处理过的节点list. E8 \' T9 b5 c: y6 h- m(欢迎访问老王论坛:laowang.vip)
- shortest_path=dijkstra(start,end,graph,costs,processed,parents)
# I( I" C" Z! V7 }5 g - return shortest_path
复制代码- #计算图中从start到end的最短路径
7 o; f- S) C9 h2 E7 q - def dijkstra(start,end,graph,costs,processed,parents):
5 T/ ^, _4 ` {' I3 a+ z2 }8 j - #查询到目前开销最小的节点
8 d7 o" D; C% U9 _ - node=find_lowest_cost_node(costs,processed)+ v1 n0 l2 x8 C& {$ b1 w(欢迎访问老王论坛:laowang.vip)
- #使用找到的开销最小节点,计算它的邻居是否可以通过它进行更新; f! }1 N! ~5 Y5 D(欢迎访问老王论坛:laowang.vip)
- #如果所有的节点都在processed里面 就结束) J) R* R8 \% A* E# X; O/ i6 y4 P(欢迎访问老王论坛:laowang.vip)
- while node is not None:
7 W$ `1 \: `8 x4 u - #获取节点的cost
6 X5 u4 H' |2 u& L/ m+ X! w - cost=costs[node] #cost 是从node 到start的距离! `5 K2 y% X' X3 \1 I(欢迎访问老王论坛:laowang.vip)
- #获取节点的邻居
; B( _' q+ |# M8 {: L9 z; ? - neighbors=graph[node]
0 s8 Z8 }8 x( P$ U; e' ^( B - #遍历所有的邻居,看是否可以通过它进行更新
2 x3 m |6 y( n' S - for neighbor in neighbors.keys():
! q- F5 ~$ Y! p6 B4 n! c$ k, { - #计算邻居到当前节点+当前节点的开销
/ F( J, I/ \' V! \' Q! g; u8 ~ - new_cost=cost+float(neighbors[neighbor])1 S" [5 l9 v" s& q6 T! ~(欢迎访问老王论坛:laowang.vip)
- if neighbor not in costs or new_cost<costs[neighbor]:$ o/ p& W/ e- }2 V2 ?0 p# w, N(欢迎访问老王论坛:laowang.vip)
- costs[neighbor]=new_cost- u P& F4 i5 L3 a(欢迎访问老王论坛:laowang.vip)
- #经过node到邻居的节点,cost最少$ U( m1 b$ J I$ c(欢迎访问老王论坛:laowang.vip)
- parents[neighbor]=node3 `" k3 A; K+ D) [( ?* g( f' W(欢迎访问老王论坛:laowang.vip)
- #将当前节点标记为已处理0 i) T/ o3 Q4 h+ ]. z+ L5 T" [(欢迎访问老王论坛:laowang.vip)
- processed.append(node)
4 r* a3 T$ g+ O; E) k6 a( [ - #下一步继续找U中最短距离的节点 costs=U,processed=S
) c- d+ C9 Q' L u+ c - node=find_lowest_cost_node(costs,processed)
复制代码- def find_lowest_cost_node(costs,processed):$ ~4 R! O) m: n& S(欢迎访问老王论坛:laowang.vip)
- #初始化数据) j6 A- f% [1 _6 i. ]5 f2 \7 N(欢迎访问老王论坛:laowang.vip)
- lowest_cost=float('inf') #初始化最小值为无穷大
) h0 F- g K, d' F [7 A- b# [ - lowest_cost_node=None
& P0 L$ C7 W$ Y3 i' s7 W' G - #遍历所有节点
& h; C2 i: }9 x( Y4 o0 p/ p4 \* A - for node in costs:
9 ^5 c$ B2 d7 L. T- e/ t5 m: m& } - #如果该节点没有被处理( @" B4 ~; q6 S- y- z3 S$ u(欢迎访问老王论坛:laowang.vip)
- if not node in processed:
" M0 b" ]) [/ F, D - #如果当前的节点的开销比已经存在的开销小,那么更新该节点为最小开销的节点* G3 f0 h. z8 P7 D. u' l$ }: w" P9 u(欢迎访问老王论坛:laowang.vip)
- if costs[node]<lowest_cost:
4 u7 E" h3 P2 _' R - lowest_cost=costs[node]
# q7 z& ?/ V5 B! b- h8 ]. A5 M - lowest_cost_node=node+ N& L! c/ j+ t% {(欢迎访问老王论坛:laowang.vip)
- return lowest_cost_node
复制代码 上面这段基本上搬运的,主要是他代码已经写的很好了,注释也写的不错,但我写的时候爬虫调不出来(反爬虫技术可以的),最后是我手动去地图里找经纬度得到的结果。
* N7 I- y. H% Y$ k0 o/ t引用:https://blog.csdn.net/fengdu78/article/details/111570695
) F9 }& @+ n2 J& A/ d) R* g j3 d: Q# n5 t2 J, O, U(欢迎访问老王论坛:laowang.vip)
0 O. v3 T2 {% w. [0 B) E# s8 R3 b4 t. ]& I1 K. _9 t( U(欢迎访问老王论坛:laowang.vip)
A*算法:这个算法也是非常著名的算法,与Dijkstra算法相比,增加了启发式函数 ---- 启发函数的好坏直接决定了算法的效率和结果。由此衍生的D*(动态A算法)算法也被广泛运用于各类游戏中。D*的搜索效率高的原因就在于,当计划路径上某点被阻碍,因为是反向指针,可以定位到被堵节点的上一节点。也就是说只需重新搜索很小的一部分,其余部分仍然使用初始规划出的路径,大大提高了重规划的效率。2 ~ V" t" o3 D* g(欢迎访问老王论坛:laowang.vip)
5 z( J5 D9 K G4 r9 ]3 W/ [9 P8 x(欢迎访问老王论坛:laowang.vip)
& }6 H0 ^+ `" M8 E(欢迎访问老王论坛:laowang.vip)
额外补充-dfs&bfs逻辑
! \1 C) b; p$ o! M, W深度搜索(dfs)和广度搜索(bfs)的算法逻辑可以说是最具有代表性的,基本任何有关于寻路的算法都没法绕开他俩。我担心可能没了解这2个算法直接去看Djkta和A*会有些懵逼- n4 o2 O% l' A$ U* O(欢迎访问老王论坛:laowang.vip)
B5 O$ O. [/ @深度搜索(dfs) , M! I) m4 n0 H6 M(欢迎访问老王论坛:laowang.vip)
; P3 K$ J5 j8 Q, U0 ~dfs就和它字面意思一样,是往更深的地方找(deepfound)
* f7 p) \& D/ M6 x; ?# a( r它的核心思想很简单:, @5 f9 y: t2 N" `8 ?1 b, T(欢迎访问老王论坛:laowang.vip)
一直往前走,走不通就回头
3 O9 q. P. u6 x* |9 _4 T3 Z6 g- a
7 [3 a! W$ P. K1 c(欢迎访问老王论坛:laowang.vip)
顺序?当然是长幼啦,有长立长无长立幼 (1会先找2而非6,在2时会先找4而非5 ,直到4发现3 发现无路可走再back回去)7 z/ h# p5 }9 Z) b3 Y(欢迎访问老王论坛:laowang.vip)
大致伪代码如下
- W5 ~$ r! e! f9 X- input 地图
9 E$ r0 [% I5 F: P3 j1 j - create 已经过点
0 M# U% K. P; s& ` - create 结果存储( p0 w" @- b! m(欢迎访问老王论坛:laowang.vip)
p' Y) E! g* \- type node{ ^/ U4 m6 R" U+ O& G1 C) r* W(欢迎访问老王论坛:laowang.vip)
- node nextNode[];//下一节点们6 p) |/ i, a7 y& `5 _8 w9 E(欢迎访问老王论坛:laowang.vip)
- nodeval;//节点标记物6 h8 ^* _" P8 h7 D(欢迎访问老王论坛:laowang.vip)
- }0 f, {) C1 b- g# B* k8 @' ~4 w(欢迎访问老王论坛:laowang.vip)
! X2 Q1 `6 s! L5 t( t4 w- / K3 {5 e* I1 i4 ^. \(欢迎访问老王论坛:laowang.vip)
- //这里开始是函数) r8 E* |- J, x! r' d. x* |7 E(欢迎访问老王论坛:laowang.vip)
- fun dfs(node* nowPoint)
7 T# E- N: {: F6 t - {1 W N, F% ]. h' ?5 b' B2 r(欢迎访问老王论坛:laowang.vip)
- define u 为 nextNode[] size. l3 c$ f9 V% I3 x9 z; d(欢迎访问老王论坛:laowang.vip)
- int key;
# v$ p' q. ]) A1 X6 [0 J( p% q6 x
& s& p( @* ?4 M: ^) X7 w) \- for i in u {8 n5 D: }0 T' G1 c6 e/ R0 W(欢迎访问老王论坛:laowang.vip)
- if (nowPoint.noteval == target)
2 f) y5 u9 V0 o) i - {/ r* p$ {$ v) O) o' e(欢迎访问老王论坛:laowang.vip)
- 结果存储[0] = nowPoint.noteval;
5 d' {. O; T/ G0 @7 s2 @9 e2 p+ b0 T - return 0 ;
M, I. p% ]2 j. s+ M% _ - }
1 m8 n! T4 o" Y% ?' s4 P - else if( key = (dfs(nowPoint->nextNode[i])) != -1 ) //如果dfs这次没有返回负1(即 找到终点了)5 @* k) W3 a- w, O! o(欢迎访问老王论坛:laowang.vip)
- {$ O- A- U A- @% v9 _* W(欢迎访问老王论坛:laowang.vip)
- 结果存储[key] = nowPoint.noteval;
% S8 R* X+ V! P- y7 Z; K6 O: \ - return key+1; ; H4 V6 ?5 L7 m6 o/ v: E(欢迎访问老王论坛:laowang.vip)
- } ( n' g' i5 B r3 A2 Z(欢迎访问老王论坛:laowang.vip)
- else7 y4 t: F% A- F9 B& z% h(欢迎访问老王论坛:laowang.vip)
- {
2 C' k3 p3 W, C& \* w- b% f - /********************/: [ F; C J# n! k(欢迎访问老王论坛:laowang.vip)
- /** nothing **/4 ?3 l1 |+ T; p. t3 d9 t: i(欢迎访问老王论坛:laowang.vip)
- /********************/
6 U) P" b$ D! I% @0 `) l - }
+ c6 L( _) c8 b* d7 O
9 B6 a/ s* R+ ^1 {! d6 w3 m- . O. i8 ~% h3 R0 M, q) L(欢迎访问老王论坛:laowang.vip)
- }
+ t' ]3 b5 G, M+ a/ ]- N3 K - return -1;
% S3 L6 Q" u5 M* i* q# D - }
; D, z1 I2 B; f" ^
复制代码 就那么短,你只需要确定是不是就行了6 A2 {# E h& u/ @(欢迎访问老王论坛:laowang.vip)
是不是很简单:p 但是这就是一个比较原始的寻路算法的模样-顺其自然流
$ s6 r5 \8 q! S) A: Z
! Z& m- |/ `5 H' [在dfs算法中,你需要做的就是 询问+上抛+ q+ e2 E i# m2 y8 w! O(欢迎访问老王论坛:laowang.vip)
当然,dfs算法唯一能保证的就是‘找得到路’,这也是为什么纯粹的dfs算法不常用于生产环境中
( y Q* f4 T1 m4 E$ r$ C" s% C. P1 L$ Q( j+ A/ t s2 A(欢迎访问老王论坛:laowang.vip)
0 y" v% [2 z, u* A' N7 `" L广度搜索(bfs)
! y3 t1 L% P' J8 ^' q9 D i知道深度,广度就很容易联想了 先找周围嘛 无论如何,先找周围
3 V1 y7 F; E1 v* n6 F D
' y5 |7 S- w7 `8 T) C4 @这里不进行代码补充了,只简单地说一下逻辑
; [# y; i2 J9 D" R( f8 `: r/ d# q: W& G6 |(欢迎访问老王论坛:laowang.vip)
这个算法分以下几步:
; x) a4 r* u5 O0 K0,准备一个队列(就是那个queue),然后丢入我们的起点 命运的齿轮开始拨动% z, C& d2 r6 B1 h# q(欢迎访问老王论坛:laowang.vip)
( z- N1 g* {8 q+ Y" F(欢迎访问老王论坛:laowang.vip)
1,询问当前节点是否终点?
3 v: V U m5 y3 G2,将自己的nextNode 塞入队列中(除了访问过的)
! }# s& S* F0 Q8 |1 ~% }7 u3 V3,从队列里output一个节点,作为下一个‘当前节点’+ {" w2 e1 N7 N/ m, K' O8 r(欢迎访问老王论坛:laowang.vip)
! D7 M$ j. f4 p. q! @+ \. @5 Q(欢迎访问老王论坛:laowang.vip)
然后就是循环1~3
( u: |5 F8 q' d0 @9 p
# q; ^; K( O( r2 G是不是很简单?" ^: l m8 ?0 o* P+ Q+ c$ M1 X- l9 R(欢迎访问老王论坛:laowang.vip)
; A' _0 c1 h( v1 M这2个算法都属于随性流,一个适合终点远一个适合终点近。但无论如何这俩都暂时没有比较最优的功能* c/ P b8 q( G. b3 C(欢迎访问老王论坛:laowang.vip)
因为他们刚~满~十~八~岁~的审敛条件就只是找得到路9 N1 h6 i8 u" `0 T(欢迎访问老王论坛:laowang.vip)
, I; |6 T$ }& f& ` v6 @" O% P但你可以发现哦?如果将dfs的‘长幼判断’换成‘最优判断’,将呆值传递换为矩阵存储 就是dj算法了诶(a*也是类似的)
8 m0 \( B" g9 Y而bfs作为‘扩散式搜索’显然地在进行多点寻路时效用更加明显: q0 l* @" p t: v# }(欢迎访问老王论坛:laowang.vip)
如果觉得寻路算法很难的话,不妨先从dfs&bfs开始了解- i% |# W! u1 K# I3 B( _(欢迎访问老王论坛:laowang.vip)
1 b+ t% o( k. P( s, f" E9 V) {- B(欢迎访问老王论坛:laowang.vip)
' y L+ p# @! z: ?% Q- i(欢迎访问老王论坛:laowang.vip)
" c! \" v0 L: U# H: q% J! O- C(欢迎访问老王论坛:laowang.vip)
/ p: |& U$ V) L2 \' V9 G- h: H. a |
-
A*寻路算法
评分
-
查看全部评分
|