加载中...

地址发布 老王说明书 宣传中心
此板块只作为纯讨论

正经话题,不搞色情!贤者时间必备
查看: 6188|回复: 29
收起左侧

[学习讨论] RPG游戏的寻路算法——从绯月讲起

[复制链接]
 楼主| 发表于 2023-12-25 18:31:47 | 显示全部楼层 |阅读模式
本帖最后由 navebayes 于 2023-12-26 12:02 编辑 5 z/ Q% s1 J5 Q1 f7 ?9 ?' A' C8 @(欢迎访问老王论坛:laowang.vip)
1 Q. a) g; B1 ?8 w. y1 y- O(欢迎访问老王论坛:laowang.vip)
最近正在玩个游戏,也算是国产之光绯月仙行录,这个游戏哪里都好就是bug太多,并且作者过于摆烂,以至于有很多玩家都认为这个游戏就是故意拖着吃赞助的(bushi)
. F+ @. B+ U! C# T! J& S言归正传,在游玩这个游戏的过程中,我在一个评论区里看到这样一段话:自从玩了绯月之后,对于其他RPG游戏都看不上眼了,因为这个游戏独创了自动寻路的功能,可以说是RPG类游戏的里程碑式壮举。
# F, P- C, v" g: }6 \, @我对此感到好奇,因为从前从来没有游玩RPG类游戏的经验,但我学过一点点算法,于是我打算用一些浅显易懂的方式说说自动寻路这样一个功能的实现。
: e; o5 p1 q% s  j6 [2 W0 @( S6 b, w. @(欢迎访问老王论坛:laowang.vip)
主流的寻路算法:深度优先,广度优先,Dijkstra,A* 等,我这里主要讲讲后两种。
: a% B, Y5 K. I4 p& I2 H0 |8 G; ?3 o2 t9 J(欢迎访问老王论坛:laowang.vip)
Dijkstra算法:这个算法是目前很多地图软件都在使用的算法,采用OPEN,CLOSE表的方式实现寻路功能。
# [+ }8 m: O; F) _8 g    创建两个表,OPEN, CLOSE。
$ w' v& ~) E- n) S: `OPEN表保存所有已生成而未考察的节点,CLOSED表中记录已访问过的节点。
; ?) y$ w7 u! t4 Z- a4 F0 _1. 访问路网中里起始点最近且没有被检查过的点,把这个点放入OPEN组中等待检查。0 e0 h6 R: G2 N  d8 B2 {(欢迎访问老王论坛:laowang.vip)
2. 从OPEN表中找出距起始点最近的点,找出这个点的所有子节点,把这个点放到CLOSE表中。# W* b& F" B7 ]. c* D/ ^7 T  r(欢迎访问老王论坛:laowang.vip)
3. 遍历考察这个点的子节点。求出这些子节点距起始点的距离值,放子节点到OPEN表中。
7 C+ P5 u6 ~4 P0 R8 R4. 重复2,3,步。直到OPEN表为空,或找到目标点。
; I+ P( l5 S" r" _) ~6 e7 J6 g4 d! f+ ~4 Z2 G3 j" j(欢迎访问老王论坛:laowang.vip)
实际写代码还是比较占行数的,直接给出链接如下。
: Y" ]& e4 C# j参考:https://blog.csdn.net/YiYeZhiNian/article/details/1222174509 @" H( G- T' W3 ~  y( v0 c(欢迎访问老王论坛:laowang.vip)
9 R" x" k" a% p/ T1 J9 c- @- g(欢迎访问老王论坛:laowang.vip)
用这个算法,我写过一个课程作业,具体就是对于各个城市的地铁最短路径规划,大致还是比较成功的。先说说个人对于Dijkstra算法设计地铁线路规划:5 l: y8 i8 t% _7 a; M' z(欢迎访问老王论坛:laowang.vip)
1.首先用爬虫爬到该城市的地铁网络,包含站点名称,该站点经纬度和线路图,并生成excel表格。用该excel表格生成pickle文件,方便直接调用。
2 M" F' S2 m: ?2 D7 `2.注册高德地图开发者账号(该功能需要实名),得到一个key用于调用api(每日上限100次,超过付费,我调试到后面不让我调试了...)
& {5 I6 U' E1 j. @# Y- p2 e3.输入始发地和目的地,并通过api返回两个位置的经纬度坐标。6 b& N$ y& e6 b8 h(欢迎访问老王论坛:laowang.vip)
4.比较两地理位置之间的最近地铁站,并根据Dijkstra算法实现路径规划。
% E+ c, P  F. v$ @4 I7 ]1 T( y5.Dijkstra算法的本质就是不断选择新顶点并更新已处理表,并将他和邻居节点进行比对,当所有节点都被处理后即为最短路径
" }1 P/ u* j8 W3 o8 j至此,已初步完成具体工作流程。
9 l( _" w2 G0 T# L
  1. def get_nearest_subway(data,longitude1,latitude1):7 x: T4 d( S: }7 a; z(欢迎访问老王论坛:laowang.vip)
  2.   #找最近的地铁站2 }; u* e8 c  D) J# g% b; b(欢迎访问老王论坛:laowang.vip)
  3.   longitude1=float(longitude1)
    ) P8 X3 r/ o3 G' r
  4.   latitude1=float(latitude1)
    4 n  |7 q  s  K4 Z% H! h- p% [0 ]# u# X" ]
  5.   distance=float('inf')
    " {# `7 {$ I' [" D
  6.   nearest_subway=None6 r! x6 e* ^: R4 y(欢迎访问老王论坛:laowang.vip)
  7.   for i in range(data.shape[0]):8 N# M/ \& A, U: R5 E- _7 a. N(欢迎访问老王论坛:laowang.vip)
  8.     site1=data.iloc[i]['name']    7 b1 Z! b; S) i5 o& J8 M  C+ w6 S(欢迎访问老王论坛:laowang.vip)
  9.     longitude=float(data.iloc[i]['longitude'])$ O& x" k6 I# }) _( w% R(欢迎访问老王论坛:laowang.vip)
  10.     latitude=float(data.iloc[i]['latitude'])    #分别将经纬度代入,计算与目标之间的欧氏距离
      G! f0 `  X3 D: Q3 m4 w
  11.     temp=geodesic((latitude1,longitude1), (latitude,longitude)).m   #temp对其遍历即可,这里对比各个地铁站的欧氏距离
    - v+ H0 z; C4 @3 N, O
  12.     if temp<distance:' {' O/ v; }% k0 c(欢迎访问老王论坛:laowang.vip)
  13.      distance=temp
    + \7 R3 k/ Y& P8 H0 Z* k$ A
  14.      nearest_subway=site11 ~: l' D+ D3 D! P6 {6 w(欢迎访问老王论坛:laowang.vip)
  15.      return nearest_subway  
复制代码
  1. def subway_line(start,end):    #创建点之间的距离, F7 j1 G: \& H# z4 {(欢迎访问老王论坛:laowang.vip)
  2.   file=open('graph.pkl','rb'): j7 i, Q7 k" x(欢迎访问老王论坛:laowang.vip)
  3.   graph=pickle.load(file)   #现在我们有了各个地铁站之间的距离存储在graph! P! H: C* n' K7 M" s' H1 @, F(欢迎访问老王论坛:laowang.vip)
  4.   costs={}    #创建节点的开销表,cost是指从start到该节点的距离/ {* v& m* q' J& ](欢迎访问老王论坛:laowang.vip)
  5.   parents={}; Q( I& K2 j& |2 L7 b  I( t* X(欢迎访问老王论坛:laowang.vip)
  6.   parents[end]=None
    * ~% H! ?: k0 z! E0 n5 F
  7.   for node in graph[start].keys():
    ! y* _( Q; B% c/ x
  8.     costs[node]=float(graph[start][node]). _" }( B. i# p+ X8 M* k3 ]8 ]. g(欢迎访问老王论坛:laowang.vip)
  9.     parents[node]=start
    ! h% M3 Z* J2 r% L+ z& k
  10.     costs[end]=float('inf')    #终点到起始点距离为无穷大
    5 T/ z' @5 C! V9 p
  11.     processed=[]      #记录处理过的节点list
    2 n5 k, _0 h* e6 n' Z* b) I7 N
  12.     shortest_path=dijkstra(start,end,graph,costs,processed,parents)
    , g- J7 w: L5 c! d) z
  13.     return shortest_path
复制代码
  1. #计算图中从start到end的最短路径
    * B4 V* _8 b" A
  2. def dijkstra(start,end,graph,costs,processed,parents):
    $ |) T  |! c7 ]) Z, f7 s
  3.     #查询到目前开销最小的节点; b7 P& I  H2 t* y6 S0 d: ](欢迎访问老王论坛:laowang.vip)
  4.     node=find_lowest_cost_node(costs,processed)& d( ~" w( D3 ^+ p! B(欢迎访问老王论坛:laowang.vip)
  5.     #使用找到的开销最小节点,计算它的邻居是否可以通过它进行更新
    * L0 g) l. M- }) ?- x
  6.     #如果所有的节点都在processed里面 就结束
    , h1 ~; d2 ]% X# A2 A' D
  7.     while node is not None:9 H' d5 t; X8 t9 H, y7 d(欢迎访问老王论坛:laowang.vip)
  8.     #获取节点的cost
    5 y4 I0 E! q) x# I
  9.       cost=costs[node]  #cost 是从node 到start的距离
    2 t+ \* |, ~9 W- ^. v( z# d
  10.       #获取节点的邻居) J1 ^2 o! D7 t+ R" O(欢迎访问老王论坛:laowang.vip)
  11.       neighbors=graph[node]1 o7 _4 v' j. f4 H(欢迎访问老王论坛:laowang.vip)
  12.       #遍历所有的邻居,看是否可以通过它进行更新/ ?, `  d' l- X: T/ w; q(欢迎访问老王论坛:laowang.vip)
  13.       for neighbor in neighbors.keys():1 h: T! [% m& C# d% S(欢迎访问老王论坛:laowang.vip)
  14.       #计算邻居到当前节点+当前节点的开销$ E2 W9 z) @' O4 K' f3 e. p! a(欢迎访问老王论坛:laowang.vip)
  15.         new_cost=cost+float(neighbors[neighbor])
    % [5 l  k8 b% m: z
  16.         if neighbor not in costs or new_cost<costs[neighbor]:
    ) k! ]( Y& g6 j0 [6 |  j* u  q/ ^2 o
  17.         costs[neighbor]=new_cost
    ' s' q$ }# o4 Y
  18.         #经过node到邻居的节点,cost最少0 J4 O1 w4 M. `& }(欢迎访问老王论坛:laowang.vip)
  19.         parents[neighbor]=node: D' t1 ]  f* |) R; R  x(欢迎访问老王论坛:laowang.vip)
  20.         #将当前节点标记为已处理( J* w0 ]! u3 G(欢迎访问老王论坛:laowang.vip)
  21.   processed.append(node)
    $ y% c5 K) P5 Q" D
  22.   #下一步继续找U中最短距离的节点  costs=U,processed=S: l* ~3 ~( h3 ?(欢迎访问老王论坛:laowang.vip)
  23.   node=find_lowest_cost_node(costs,processed)
复制代码
  1. def find_lowest_cost_node(costs,processed):# K* O3 u  t' S" l" T6 K(欢迎访问老王论坛:laowang.vip)
  2.     #初始化数据
    # r( }5 {0 \- X; |# ?. r* ^2 N# i. f
  3.   lowest_cost=float('inf') #初始化最小值为无穷大
    $ E. q4 d% w( x/ V8 p4 X
  4.   lowest_cost_node=None
    ' @, W- l  V4 O% u
  5.     #遍历所有节点
    1 m& D/ Z) ]0 ]
  6.   for node in costs:) P- o/ q; B* k* a  o5 |/ d" z- d(欢迎访问老王论坛:laowang.vip)
  7.     #如果该节点没有被处理
    + O+ ?8 I- m, L  |( `# ^
  8.     if not node in processed:
    ! ^9 V9 l  Z% v; _
  9.       #如果当前的节点的开销比已经存在的开销小,那么更新该节点为最小开销的节点1 s8 [! R2 Z% _$ F' _# D1 e(欢迎访问老王论坛:laowang.vip)
  10.       if costs[node]<lowest_cost:
    9 H, a! {3 k7 L! ]: s2 x1 j6 Y. A
  11.       lowest_cost=costs[node]! g4 V$ K) d3 W. W' {) Z- f(欢迎访问老王论坛:laowang.vip)
  12.       lowest_cost_node=node- m7 P0 r' a8 s8 |! Y(欢迎访问老王论坛:laowang.vip)
  13.     return lowest_cost_node
复制代码
上面这段基本上搬运的,主要是他代码已经写的很好了,注释也写的不错,但我写的时候爬虫调不出来(反爬虫技术可以的),最后是我手动去地图里找经纬度得到的结果。
/ c4 B5 H3 N/ F5 {9 K$ D0 \引用:https://blog.csdn.net/fengdu78/article/details/111570695
7 r' s7 T$ \0 @5 `: j/ E$ F* M+ c
/ U8 H' R# c0 d& c(欢迎访问老王论坛:laowang.vip)

0 Z) ]' j/ b( s1 E6 R

! [. f. f! t+ d3 k; w/ g# \A*算法:这个算法也是非常著名的算法,与Dijkstra算法相比,增加了启发式函数 ---- 启发函数的好坏直接决定了算法的效率和结果。由此衍生的D*(动态A算法)算法也被广泛运用于各类游戏中。D*的搜索效率高的原因就在于,当计划路径上某点被阻碍,因为是反向指针,可以定位到被堵节点的上一节点。也就是说只需重新搜索很小的一部分,其余部分仍然使用初始规划出的路径,大大提高了重规划的效率。
$ ?8 ^, v0 t- O6 z" L$ _4 J8 [0 c; P5 F. M, `(欢迎访问老王论坛:laowang.vip)
/ [* q0 A' K2 N(欢迎访问老王论坛:laowang.vip)
额外补充-dfs&bfs逻辑
# O& h( w# t- ]( C深度搜索(dfs)和广度搜索(bfs)的算法逻辑可以说是最具有代表性的,基本任何有关于寻路的算法都没法绕开他俩。我担心可能没了解这2个算法直接去看Djkta和A*会有些懵逼
! j( h- W% v  E
, s6 h3 m8 r* j深度搜索(dfs) ) p4 ~6 U: H' t(欢迎访问老王论坛:laowang.vip)

3 g; _) [# N  l# fdfs就和它字面意思一样,是往更深的地方找(deepfound)' i. f$ D2 h7 M5 g3 L1 x; }(欢迎访问老王论坛:laowang.vip)
它的核心思想很简单:
8 q. x8 J+ H# S' ^一直往前走,走不通就回头
( P; Z& }# C0 `+ n0 b: w( [$ [, i! b$ s" k5 w- j(欢迎访问老王论坛:laowang.vip)
顺序?当然是长幼啦,有长立长无长立幼 (1会先找2而非6,在2时会先找4而非5 ,直到4发现3 发现无路可走再back回去)
% r; b+ e4 [4 F& u; _- V& e/ l6 w大致伪代码如下2 |4 M9 [$ r) O& W3 p! E; S2 Z! l(欢迎访问老王论坛:laowang.vip)
  1. input 地图' t. w% V* I8 T2 m& _# b(欢迎访问老王论坛:laowang.vip)
  2. create 已经过点7 J) X$ n) M- u" ~(欢迎访问老王论坛:laowang.vip)
  3. create 结果存储5 P1 a3 s8 b9 w4 Q: e! n; b2 V(欢迎访问老王论坛:laowang.vip)
  4. 9 U( l% L1 ]- I(欢迎访问老王论坛:laowang.vip)
  5. type node{
    ; W2 w; j0 W3 Q9 e1 e$ m( B
  6. node nextNode[];//下一节点们
    * Y$ ]5 y, E' _; f
  7. nodeval;//节点标记物" h! A2 u& |) _1 K- S% r& a6 ~(欢迎访问老王论坛:laowang.vip)
  8. }# u8 u& A" o1 `) i, G( ]6 P(欢迎访问老王论坛:laowang.vip)

  9. 9 R( J  p% R8 M& W5 s
  10. 9 \% b5 b# ^9 [3 T$ d) F+ |, w(欢迎访问老王论坛:laowang.vip)
  11. //这里开始是函数
    " I7 |% n" q5 F: g5 p. m
  12. fun dfs(node* nowPoint); L7 W# p& n* n  E$ N(欢迎访问老王论坛:laowang.vip)
  13. {
    , i& d0 X( ?- y$ p. d0 s# S4 y
  14. define u 为 nextNode[] size
    + B# B: A1 ]; x/ h; I* B
  15. int  key;& T! J3 G7 O: t2 a4 J(欢迎访问老王论坛:laowang.vip)

  16. / T7 i0 v; e! d  P+ Q3 ~
  17. for i in u {
    6 H$ e  t- n1 y) `/ E7 v* A
  18.       if (nowPoint.noteval == target)
    3 P' b9 g* `* `# o- e
  19.         {6 d: }* O: @* q$ O9 z: h(欢迎访问老王论坛:laowang.vip)
  20.             结果存储[0] = nowPoint.noteval;3 k6 X1 d% i8 j5 X& q(欢迎访问老王论坛:laowang.vip)
  21.             return 0 ;
    3 Z! n% K5 f5 e, R$ M9 l
  22.         }  . h% T! I0 p3 k/ a  X: F(欢迎访问老王论坛:laowang.vip)
  23.       else if( key = (dfs(nowPoint->nextNode[i])) != -1 ) //如果dfs这次没有返回负1(即 找到终点了)
    0 i9 X- V& G0 k$ w- W: Q( X
  24.             {7 N0 H7 v% o; E(欢迎访问老王论坛:laowang.vip)
  25.                 结果存储[key] = nowPoint.noteval;0 d, y' f& L  ]3 C7 m2 C; \(欢迎访问老王论坛:laowang.vip)
  26.                 return key+1;         6 x$ {1 }. c, E7 C. X(欢迎访问老王论坛:laowang.vip)
  27.             }   
    $ c; D; u# V9 I- ^. I3 [; y4 H# F
  28.         else! Y/ f: B- S- U$ i3 `(欢迎访问老王论坛:laowang.vip)
  29.             {
    % p& t! N1 V) O) R
  30.                 /********************/
    9 f* v/ c' O- r/ c3 C
  31.                 /**        nothing          **/, u5 Q, p, P' b(欢迎访问老王论坛:laowang.vip)
  32.                 /********************/
    4 @! _) B/ g* G, Y; X& s
  33.             }    # R# D: }6 i- k6 @! F% Y(欢迎访问老王论坛:laowang.vip)
  34. 4 O# h% N. R! r  k; b/ z(欢迎访问老王论坛:laowang.vip)
  35.         
    % }" x) u# |2 E& o- n
  36.    }   
    9 s1 l1 H! u* {8 a. j) |& [
  37.     return -1;  d/ u8 E; A' \- |8 s(欢迎访问老王论坛:laowang.vip)
  38. }8 i, z/ R9 r* w4 q& e3 ?5 ](欢迎访问老王论坛:laowang.vip)
复制代码
就那么短,你只需要确定是不是就行了! y" M& U7 o1 h& O; C(欢迎访问老王论坛:laowang.vip)
是不是很简单:p 但是这就是一个比较原始的寻路算法的模样-顺其自然流1 x7 i, a! e3 s! r(欢迎访问老王论坛:laowang.vip)
2 m! p  x) F# z3 Q# Z" k8 }/ s(欢迎访问老王论坛:laowang.vip)
在dfs算法中,你需要做的就是 询问+上抛
, |1 P- h: R7 C2 f4 b) j当然,dfs算法唯一能保证的就是‘找得到路’,这也是为什么纯粹的dfs算法不常用于生产环境中
% D6 Q: C# ^& L. ?$ P9 _0 X' X! D0 P' q; Q. i(欢迎访问老王论坛:laowang.vip)

7 _4 [0 R- Q! Y广度搜索(bfs)
2 M" t# w* i+ q$ C3 g# b* k; `知道深度,广度就很容易联想了 先找周围嘛 无论如何,先找周围; w( q* s+ a, i! G8 V(欢迎访问老王论坛:laowang.vip)

! p0 l1 [1 f* |" W这里不进行代码补充了,只简单地说一下逻辑
# O, C8 o+ D2 N" U
& ?4 p& N, J, d5 ~这个算法分以下几步:& A8 P/ N, A6 H- {5 w5 l& B0 Q(欢迎访问老王论坛:laowang.vip)
0,准备一个队列(就是那个queue),然后丢入我们的起点 命运的齿轮开始拨动
! |, N2 V2 R8 A( N' ?: G
/ B; S7 B$ \! Q1,询问当前节点是否终点?$ Q$ j  |/ \  ^* L! z(欢迎访问老王论坛:laowang.vip)
2,将自己的nextNode 塞入队列中(除了访问过的)
1 z: Y3 \8 l: J# R( J/ A2 F3,从队列里output一个节点,作为下一个‘当前节点’/ C* o+ `# g" T& \(欢迎访问老王论坛:laowang.vip)
5 s; E: g  V& ]& d3 [* T(欢迎访问老王论坛:laowang.vip)
然后就是循环1~3
% Z/ f" N& o* X5 y, ], U9 V1 Q4 [
, E# X# |3 _! m5 E3 i是不是很简单?
8 L( U  g, x  t. n, x* Y, k3 d$ V$ n2 f6 @(欢迎访问老王论坛:laowang.vip)
这2个算法都属于随性流,一个适合终点远一个适合终点近。但无论如何这俩都暂时没有比较最优的功能: k, t5 ?' m% E5 o; T(欢迎访问老王论坛:laowang.vip)
因为他们刚~满~十~八~岁~的审敛条件就只是找得到路
: u) p& x" T" \: x4 R# o2 w* i0 B9 _5 _$ q2 P3 x( b" o1 y- k1 m(欢迎访问老王论坛:laowang.vip)
但你可以发现哦?如果将dfs的‘长幼判断’换成‘最优判断’,将呆值传递换为矩阵存储 就是dj算法了诶(a*也是类似的)" `  j7 ]5 J9 e4 @( ^4 o8 g(欢迎访问老王论坛:laowang.vip)
而bfs作为‘扩散式搜索’显然地在进行多点寻路时效用更加明显
* v( S- L9 L1 u+ V" p# B1 B  y如果觉得寻路算法很难的话,不妨先从dfs&bfs开始了解6 F; j3 k# ]4 g- [: ^; T) t(欢迎访问老王论坛:laowang.vip)
" M. ^- t+ C, }* Y" {(欢迎访问老王论坛:laowang.vip)
  K3 W$ k  S# [% |( V2 B(欢迎访问老王论坛:laowang.vip)
* H/ T/ j2 U9 E# {3 [# e5 E% {(欢迎访问老王论坛:laowang.vip)
6 h$ L" D; y/ K( k5 @(欢迎访问老王论坛:laowang.vip)

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?免费注册

x

评分

参与人数 1软妹币 +235 收起 理由
navebayes + 235 cheese!!

查看全部评分

本帖被以下淘专辑推荐:

回复

使用道具 举报

发表于 2023-12-25 20:31:04 手机版 | 显示全部楼层
Applcu 发表于 2023-12-25 20:19! h7 W/ \- D" N  D3 f+ g0 g(欢迎访问老王论坛:laowang.vip)
我在想怎么算说明运作逻辑.....真贴上去么?这个编辑器到底怎么用啊  D) B2 {- C) ~+ x- k(欢迎访问老王论坛:laowang.vip)
...
$ N; e) w1 Y, B3 K: ~1 R(欢迎访问老王论坛:laowang.vip)
有一个代码框功能,可以用那个。主要编辑的话建议用正八经的md编辑器(我用的是国产的typora)
回复 支持 3 反对 0

使用道具 举报

发表于 2023-12-25 20:25:06 | 显示全部楼层
navebayes 发表于 2023-12-25 20:11
( x; B- c. w  C9 N" |& ^申请失败,原因:对算法本身描述过少,请更具体地说明算法的运作逻辑和方式,如果可以最好给出部分低代码$ l( P' g) J; i( ]/ e(欢迎访问老王论坛:laowang.vip)
" v: H3 z6 ?' z& I$ m0 W  o) @# u(欢迎访问老王论坛:laowang.vip)
...
4 U1 ?' [8 [2 y2 c(欢迎访问老王论坛:laowang.vip)
啊这。不如解释一下单源最短路径问题中对于负权值和负权值回路为什么不能使用上述算法$ y- A: L% B; ?  @# K, D2 |(欢迎访问老王论坛:laowang.vip)

/ ^$ X8 ?4 H: Y! K: \这玩意考试考得头昏脑涨,我tm 深度生成树画错了,越想越气 怎么会这么蠢' n5 b9 Y- w  V9 H% @& ](欢迎访问老王论坛:laowang.vip)
回复 支持 2 反对 0

使用道具 举报

 楼主| 发表于 2023-12-25 23:28:47 | 显示全部楼层
昔有佳人公孙氏 发表于 2023-12-25 23:085 A) ?; g; w/ d4 s) _8 ?; x(欢迎访问老王论坛:laowang.vip)
其实目前,本科生能接触到的寻路算法 无外乎迪杰斯特拉和弗洛伊德 一个不吃负值 一个不吃负环
* b. Y# H; l. r) m- b/ S& K但其实寻找最 ...
7 x- H9 M1 I( ~' ]5 e- ]1 k3 \(欢迎访问老王论坛:laowang.vip)
黏菌那个早有耳闻,确实是大自然的力量。我之后还想写点加密算法,比如AES,RSA,ECC之类的,其实这个地铁的算法也是我的一个课设,我大概花了两天就搞完了,我的主专业课是自控/单片机/数电/模电/嵌入式之类的,更偏自动化这个方向,算法我研究的不深。这个当毕业论文可能还是有一些勉强的,个人感觉工科是要做出一点实物的东西,当然不同学校可能要求不尽相同。
+ b& f' A6 p: _. a
5 ?0 u3 R- f! U) R' m8 X9 R( m3 }: V( [  Y; y(欢迎访问老王论坛:laowang.vip)
回复 支持 1 反对 0

使用道具 举报

 楼主| 发表于 2023-12-25 21:58:28 | 显示全部楼层
先说说个人对于Dijkstra算法设计地铁线路规划:
" w3 }$ X% Y- x) T. ?5 B& ~  d1.首先用爬虫爬到该城市的地铁网络,包含站点名称,该站点经纬度和线路图,并生成excel表格。用该excel表格生成pickle文件,方便直接调用。
. }( \7 I+ c2 Z  E2.注册高德地图开发者账号(该功能需要实名),得到一个key用于调用api(每日上限100次,超过付费,我调试到后面不让我调试了...): i) s; k! @' D4 I, ^2 s+ g+ V4 T(欢迎访问老王论坛:laowang.vip)
3.输入始发地和目的地,并通过api返回两个位置的经纬度坐标。
' O& g+ u6 x6 }4.比较两地理位置之间的最近地铁站,并根据Dijkstra算法实现路径规划。& M# }7 J" b6 R! M5 ?5 V& D(欢迎访问老王论坛:laowang.vip)
5.Dijkstra算法的本质就是不断选择新顶点并更新已处理表,并将他和邻居节点进行比对,当所有节点都被处理后即为最短路径
: i) V+ p# ?2 i# k$ u# {至此,已初步完成具体工作流程。% Z* j, Y, S- v(欢迎访问老王论坛:laowang.vip)
  1. def get_nearest_subway(data,longitude1,latitude1):6 F8 E" W" H. B/ L+ z(欢迎访问老王论坛:laowang.vip)
  2.     #找最近的地铁站
    ) }9 K' W6 r0 U) I( d3 p
  3.     longitude1=float(longitude1)
    * m. ?; r2 A/ K; a$ n3 q
  4.     latitude1=float(latitude1)' y9 y; l2 M6 m8 o* x+ o! L1 T1 N(欢迎访问老王论坛:laowang.vip)
  5.     distance=float('inf')
    ! {" z) u! @. F" j( v0 |
  6.     nearest_subway=None- {! n) i9 r+ y8 @9 g4 u, M(欢迎访问老王论坛:laowang.vip)
  7.     for i in range(data.shape[0]):" X% h7 C5 s0 q( g- h8 j' u9 r! }(欢迎访问老王论坛:laowang.vip)
  8.   site1=data.iloc[i]['name']    9 N8 {+ u% n& Z: C, E. B0 }7 T(欢迎访问老王论坛:laowang.vip)
  9.   longitude=float(data.iloc[i]['longitude'])  @7 h. ?3 O  Q* u+ i+ l(欢迎访问老王论坛:laowang.vip)
  10.   latitude=float(data.iloc[i]['latitude'])    #分别将经纬度代入,计算与目标之间的欧氏距离
    ) b/ A# Z, n' g( c! L. @# L- H
  11.   temp=geodesic((latitude1,longitude1), (latitude,longitude)).m   #temp对其遍历即可,这里对比各个地铁站的欧氏距离7 P; ^( X, e: i' l( W(欢迎访问老王论坛:laowang.vip)
  12.   if temp<distance:
    6 T: R% F! ~1 ~# ~$ f
  13.     distance=temp  \( A7 g5 i+ _(欢迎访问老王论坛:laowang.vip)
  14.     nearest_subway=site12 B* b1 H! g" ]4 |9 f. `5 U(欢迎访问老王论坛:laowang.vip)
  15.     return nearest_subway  
复制代码
  1. def subway_line(start,end):    #创建点之间的距离, b- Y$ A4 `* T5 Q: u+ Y) Z' a(欢迎访问老王论坛:laowang.vip)
  2.     file=open('graph.pkl','rb')
    . G. w, {' B& t
  3.     graph=pickle.load(file)   #现在我们有了各个地铁站之间的距离存储在graph$ O& M0 v5 L9 J& G8 ]" i(欢迎访问老王论坛:laowang.vip)
  4.     costs={}    #创建节点的开销表,cost是指从start到该节点的距离
    8 ?1 w5 A3 Z! ?4 p4 i  s/ J
  5.     parents={}1 M" b1 e  _/ O7 D4 w* K, h(欢迎访问老王论坛:laowang.vip)
  6.     parents[end]=None5 J5 D" [2 Y! z' c(欢迎访问老王论坛:laowang.vip)
  7.     for node in graph[start].keys():/ O' x. h, L2 r2 M8 I9 l(欢迎访问老王论坛:laowang.vip)
  8.   costs[node]=float(graph[start][node])/ b" e% |9 U, r" Z* ~(欢迎访问老王论坛:laowang.vip)
  9.   parents[node]=start/ g- ~: \% u8 j3 y" T' ?" O(欢迎访问老王论坛:laowang.vip)
  10.     costs[end]=float('inf')    #终点到起始点距离为无穷大
    + l: Y2 b6 _( B- P. d2 O
  11.     processed=[]      #记录处理过的节点list
    ! x+ z2 o6 o. m  l: h5 b
  12.     shortest_path=dijkstra(start,end,graph,costs,processed,parents)& p, f( g: H) s% S7 m% |(欢迎访问老王论坛:laowang.vip)
  13.     return shortest_path
复制代码
  1. #计算图中从start到end的最短路径# {% a, }4 [( F3 u( |(欢迎访问老王论坛:laowang.vip)
  2. def dijkstra(start,end,graph,costs,processed,parents):' H( y3 z6 G! J% X1 k1 X(欢迎访问老王论坛:laowang.vip)
  3.     #查询到目前开销最小的节点
    + h3 R3 [, T; G  |
  4.     node=find_lowest_cost_node(costs,processed): i3 L% B6 I% o(欢迎访问老王论坛:laowang.vip)
  5.     #使用找到的开销最小节点,计算它的邻居是否可以通过它进行更新
    6 Q; V  l/ x8 O7 ?7 C2 {
  6.     #如果所有的节点都在processed里面 就结束
    + u' ]5 ~' D" C; W8 F
  7.     while node is not None:
    , f1 Z, p6 Q8 s$ S: q
  8.   #获取节点的cost5 ^- l  b9 P6 s% o2 M- }& t(欢迎访问老王论坛:laowang.vip)
  9.   cost=costs[node]  #cost 是从node 到start的距离
    * m. ^' ~) H9 Q9 V- q. c% p
  10.   #获取节点的邻居
    $ J4 ?/ \' c0 x/ s. T5 A; i/ F. z
  11.   neighbors=graph[node]% S, G# T6 X- B( w8 Z. i(欢迎访问老王论坛:laowang.vip)
  12.   #遍历所有的邻居,看是否可以通过它进行更新2 W# V0 R2 U- \+ E& ^% I  C(欢迎访问老王论坛:laowang.vip)
  13.   for neighbor in neighbors.keys():+ ^8 n/ d" A; }1 L; g(欢迎访问老王论坛:laowang.vip)
  14.     #计算邻居到当前节点+当前节点的开销
    ! q/ c6 {: I$ J6 r3 x3 B4 E
  15.     new_cost=cost+float(neighbors[neighbor])+ V% h: Y* j6 [2 g(欢迎访问老王论坛:laowang.vip)
  16.     if neighbor not in costs or new_cost<costs[neighbor]:
    1 ]* j+ b9 N$ u
  17.       costs[neighbor]=new_cost
    5 b; [! ?" L2 a, g( L6 ]
  18.       #经过node到邻居的节点,cost最少
    . |7 h$ k3 K  Q% R7 m
  19.       parents[neighbor]=node1 }+ M/ ~: J/ M(欢迎访问老王论坛:laowang.vip)
  20.   #将当前节点标记为已处理
    " g- D+ C! w3 T# `% I
  21.   processed.append(node)
    $ M* k2 Z0 m) n3 E8 r4 D  C( n
  22.   #下一步继续找U中最短距离的节点  costs=U,processed=S
    3 d. c% [. ^% x2 ^) c; I9 e
  23.   node=find_lowest_cost_node(costs,processed)
复制代码

" T7 X- v$ c3 o4 k3 B: I2 p- Y( Y8 H: c: F(欢迎访问老王论坛:laowang.vip)
  1. def find_lowest_cost_node(costs,processed):* [7 I# v* P; t& `% ]6 q(欢迎访问老王论坛:laowang.vip)
  2.     #初始化数据; q- ^; o& b( O8 I6 B, w. |(欢迎访问老王论坛:laowang.vip)
  3.     lowest_cost=float('inf') #初始化最小值为无穷大
    - C7 ]+ p7 A8 C' P+ |
  4.     lowest_cost_node=None
    " r1 {- s. r4 H$ T3 e' B/ {
  5.     #遍历所有节点
    2 W* A! g; H3 D% x5 B9 F
  6.     for node in costs:8 v9 O* d. G8 U3 V8 y0 n(欢迎访问老王论坛:laowang.vip)
  7.   #如果该节点没有被处理" i, s! U0 p4 q$ |/ A: e& _(欢迎访问老王论坛:laowang.vip)
  8.   if not node in processed:
    4 G7 H" z, |8 r; P2 D$ `- ]
  9.     #如果当前的节点的开销比已经存在的开销小,那么更新该节点为最小开销的节点# g' y  T! u2 ?7 H1 C  i( O1 q(欢迎访问老王论坛:laowang.vip)
  10.     if costs[node]<lowest_cost:- X4 G. _. |2 T0 H- v(欢迎访问老王论坛:laowang.vip)
  11.       lowest_cost=costs[node]6 {+ j2 v( ]/ |6 e1 N9 @! E(欢迎访问老王论坛:laowang.vip)
  12.       lowest_cost_node=node
    - c" A' @) g* T/ i
  13.     return lowest_cost_node
复制代码
上面这段基本上搬运的,主要是他代码已经写的很好了,注释也写的不错,但我写的时候爬虫调不出来(反爬虫技术可以的),最后是我手动去地图里找经纬度得到的结果。
# R( S& _5 e6 y8 W: ]% M3 o引用:https://blog.csdn.net/fengdu78/article/details/111570695
( k# a, {0 O' G
回复 支持 1 反对 0

使用道具 举报

发表于 2023-12-25 20:11:24 手机版 | 显示全部楼层

修改修改

本帖最后由 navebayes 于 2023-12-25 20:14 编辑 * F4 a; i. U+ G  Y/ |2 ]( K" h% Y$ y. F& e(欢迎访问老王论坛:laowang.vip)

; P0 ^7 t5 W' H# Z0 H申请失败,原因:对算法本身描述过少,请更具体地说明算法的运作逻辑和方式,如果可以最好给出部分低代码
/ o- F! M' |+ \& k9 a! F
5 @; L1 S  u" e8 Z4 [  vps:只要做滴好,王爷有赏啊
回复 支持 1 反对 0

使用道具 举报

 楼主| 发表于 2023-12-25 20:19:00 | 显示全部楼层
navebayes 发表于 2023-12-25 20:110 S" h3 Y" K0 Z(欢迎访问老王论坛:laowang.vip)
申请失败,原因:对算法本身描述过少,请更具体地说明算法的运作逻辑和方式,如果可以最好给出部分低代码: M$ o7 R9 H& `(欢迎访问老王论坛:laowang.vip)

! h! ~9 f: v4 H7 F' }. n ...
/ w0 T6 p) W  T2 R% j(欢迎访问老王论坛:laowang.vip)
我在想怎么算说明运作逻辑.....真贴上去么?这个编辑器到底怎么用啊% e- n# F0 ?3 N(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

发表于 2023-12-25 20:35:28 手机版 | 显示全部楼层
昔有佳人公孙氏 发表于 2023-12-25 20:25
( g( o1 i# [$ a2 ]8 D  W: D6 ]& R) t: Q啊这。不如解释一下单源最短路径问题中对于负权值和负权值回路为什么不能使用上述算法# F- B# u# e) p, Z3 F(欢迎访问老王论坛:laowang.vip)

) o- N2 I, Y: {1 K+ \5 P这玩意考试考得头 ...
8 r  l! v$ n( [0 i+ x/ O(欢迎访问老王论坛:laowang.vip)
看作者本人嘛,我不想干涉思路..
  q, f; n/ k, @( ?" ?. u, P(程序:哥们这可不是什么负权圈,这是时空之门啊啊啊啊啊(然后获得了每过一年减一岁的黄金体验))
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-12-25 20:43:49 | 显示全部楼层
昔有佳人公孙氏 发表于 2023-12-25 20:25  I* ]! t: Z4 _9 X. c* }( u(欢迎访问老王论坛:laowang.vip)
啊这。不如解释一下单源最短路径问题中对于负权值和负权值回路为什么不能使用上述算法
1 T9 Y, u4 g+ l6 L" Z
6 \+ z+ R4 t1 |  Z" i7 M这玩意考试考得头 ...

# \$ k5 j, S3 o1 T( {% _1 g这应该是考专业课考破防了1 L& ~: `) `$ N+ q  t(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-12-25 20:48:19 | 显示全部楼层
昔有佳人公孙氏 发表于 2023-12-25 20:258 A, |1 q/ P4 I) U) f/ f(欢迎访问老王论坛:laowang.vip)
啊这。不如解释一下单源最短路径问题中对于负权值和负权值回路为什么不能使用上述算法
! u5 l4 b# x& o% c. u$ ?8 f  s$ n& q" r2 o0 M(欢迎访问老王论坛:laowang.vip)
这玩意考试考得头 ...

! \; G5 k% q0 `" }/ {负权不就卡bug了么,越来越低就成环了4 p# Z  j9 V2 F& u! D(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

发表于 2023-12-25 20:56:03 手机版 | 显示全部楼层
Applcu 发表于 2023-12-25 20:48
5 ^9 T+ i, v/ ?" [) J. z& e! `负权不就卡bug了么,越来越低就成环了
  @+ _0 Q9 J" C1 O2 w% j1 |! V& [(欢迎访问老王论坛:laowang.vip)
宝宝,把剩下的内容补上嘛
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-12-25 21:15:16 | 显示全部楼层
navebayes 发表于 2023-12-25 20:56
( P" i) b9 M8 ^5 Z; y$ I* P宝宝,把剩下的内容补上嘛
1 q0 c( U" C5 f! w(欢迎访问老王论坛:laowang.vip)
真是男童啊.....  }9 z& G/ S- @3 K3 F(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

发表于 2023-12-25 21:37:16 手机版 | 显示全部楼层
Applcu 发表于 2023-12-25 21:15
0 F  t, _: }1 ^6 I1 }0 s真是男童啊.....
" I( c+ x' Z" |4 v$ f; r(欢迎访问老王论坛:laowang.vip)
诽谤管理封七天
回复 支持 反对

使用道具 举报

发表于 2023-12-25 22:34:33 | 显示全部楼层
Applcu 发表于 2023-12-25 21:580 {/ R5 @" q( R$ Y, M; U(欢迎访问老王论坛:laowang.vip)
先说说个人对于Dijkstra算法设计地铁线路规划:
1 V$ M* R6 c# u5 Y1.首先用爬虫爬到该城市的地铁网络,包含站点名称,该站点 ...
8 X3 A: s8 d8 _/ d6 M(欢迎访问老王论坛:laowang.vip)
可以加入正文喵3 F: Y' g& E+ W, J* e(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-12-25 22:42:13 | 显示全部楼层
昔有佳人公孙氏 发表于 2023-12-25 20:25# w! H8 g9 t0 C/ Z4 e(欢迎访问老王论坛:laowang.vip)
啊这。不如解释一下单源最短路径问题中对于负权值和负权值回路为什么不能使用上述算法. W% h  _1 A% O2 `(欢迎访问老王论坛:laowang.vip)

- i8 C, J+ o/ @+ F* ]这玩意考试考得头 ...
6 P% `' V1 N  C( Y(欢迎访问老王论坛:laowang.vip)
我也头大了.....但本质还是因为Dijkstra算法认为,如果有一条最短路线可以直达目的地,那么从旁边走另外的路将会直接不考虑,但在负权边这里,反而会减小cost,所以Dijkstra算法失效了,因为无法发现这条最短的路径$ I) }, ~( y8 ~9 u. H: q(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-12-25 22:51:56 | 显示全部楼层
navebayes 发表于 2023-12-25 22:34
$ \* n7 Q4 x+ H( d* O' s+ |/ Q可以加入正文喵

7 Q1 S9 `9 ~9 Z! S% I加了加了
5 B7 }: ~: C* h  a" K% B$ l% k
回复 支持 反对

使用道具 举报

发表于 2023-12-25 23:02:44 | 显示全部楼层
Applcu 发表于 2023-12-25 22:51
. |6 Z# V8 ]9 J+ g5 B  E. M加了加了
' l+ v4 {& t8 m5 b(欢迎访问老王论坛:laowang.vip)
介意让我改一下吗https://pic.imgdb.cn/item/6589996fc458853aef070f38.gif/ ~1 m( C1 G% `# T(欢迎访问老王论坛:laowang.vip)
回复 支持 反对

使用道具 举报

发表于 2023-12-25 23:08:15 | 显示全部楼层
其实目前,本科生能接触到的寻路算法 无外乎迪杰斯特拉和弗洛伊德 一个不吃负值 一个不吃负环# g  q4 b% p; r: S" C% [(欢迎访问老王论坛:laowang.vip)
但其实寻找最短路径的算法里还有很知名的黏菌算法。1 h* C9 z6 ]3 ?2 o# d: C7 M(欢迎访问老王论坛:laowang.vip)
有兴趣的可以去B站搜搜黏菌走迷宫,看看在大自然顶级的算法编辑下的单源最短路径问题的解法# U/ g6 U$ C2 e* A; {(欢迎访问老王论坛:laowang.vip)
不是说将奖励放置位置对应于日本的各大城市,两天内的黏菌路线就基本和目前日本的城市规划大致相似了吗,当然不太严谨是真的 0 d5 r9 f9 o! j9 j(欢迎访问老王论坛:laowang.vip)
就好像楼主说的做了地铁最短路径规划, 这玩意我舍友和你类似,她做的事全国铁路路径规划,涉及到价格,时间,反正做到最后是你输入起始地址和目的地,就能得出花费最少,时间最少,转站最少的路线,C++课设 我记得很清楚,我记得老师说的评价是,如果再加上运筹学的内容,说不定可以当毕业论文了,不出意外她拿了满分。
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-12-25 23:30:40 | 显示全部楼层
navebayes 发表于 2023-12-25 23:02# E+ x6 a% p0 G) G5 R  F4 i4 X8 g: \4 I% i(欢迎访问老王论坛:laowang.vip)
介意让我改一下吗

* N6 w# x7 Z3 v0 t: X可以可以
9 X/ j; Y( j% {  v0 {
回复 支持 反对

使用道具 举报

发表于 2023-12-26 12:03:53 | 显示全部楼层
Applcu 发表于 2023-12-25 23:30: z2 y# d0 ^9 u/ y(欢迎访问老王论坛:laowang.vip)
可以可以
+ |( T4 ^3 J& ~% H(欢迎访问老王论坛:laowang.vip)
排版改了下喵,东西加好了喵
! I& p, G8 I5 H' B2 b$ G7 j% A
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 免费注册
点击进行验证

本版积分规则

我们不生产资源,只做资源的搬运工。

tags标签-春满四合院-AvGood-Archiver-小黑屋- |网站地图