QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 1566|回复: 0
打印 上一主题 下一主题

[数据结构与算法]16 什么,图这种数据结构把你难住了?!

[复制链接]
字体大小: 正常 放大
杨利霞        

5273

主题

82

听众

17万

积分

  • TA的每日心情
    开心
    2021-8-11 17:59
  • 签到天数: 17 天

    [LV.4]偶尔看看III

    网络挑战赛参赛者

    网络挑战赛参赛者

    自我介绍
    本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。

    群组2018美赛大象算法课程

    群组2018美赛护航培训课程

    群组2019年 数学中国站长建

    群组2019年数据分析师课程

    群组2018年大象老师国赛优

    跳转到指定楼层
    1#
    发表于 2020-4-26 15:17 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
    [数据结构与算法]16 什么,图这种数据结构把你难住了?!
    ( y  I7 g! p5 h, c2 n& W/ I你是不是和我一样,在学习数据结构与算法时,了解了一下图这种数据结构之后,根本不知道它的用武之地在哪里?
    ( ?& Y- Q& h; H# m, E* A5 O在我查了资料之后,现在我可以跟你讲讲,图可以这么用!
    ( b* d4 j: o3 b6 N( x) m& P4 z5 [9 C1 V& V
    概念介绍4 L! M/ m( R( G6 N4 Q
    ) o3 p' P' Q% O* v; h. t, r
    先来了解一下什么是图., A6 L, n$ K. }; @6 Z
    图,是一种非线性表数据结构.
    9 l2 i, \( Z$ U- x8 f! a3 h那么你可能会问了,什么是线性表结构哇,我怎么区别一种数据结构是线性的,还是非线性的呢.7 Q8 F! S$ a# @3 ?9 u7 [
    哈哈,还好我机智,在这篇文章之前就写了一篇文章来介绍,如果还有疑问,楼上雅座请: [数据结构与算法]14 搞不懂线性结构,非线性结构?# r( \8 P/ D( Q% y0 I
    & X8 T. S' B8 _9 b3 O
    在图中元素叫做顶点( vertex ),图中的一个顶点可以和其他任意顶点建立连接关系,这种建立的关系叫做边( edge )  W' q% O3 T+ f  B$ }
    3 a4 H2 x) M1 D7 F+ e5 X" S
    无向图
    % i; A, H0 [. F( m 1.jpg
    . p% P, a9 G$ U  [% D: E, w3 {8 r# t+ t* _4 ]% ]" h- @/ R5 D- m3 h! i
    上面给出的是无向图.看到这里你可能就觉得比较疑惑,这个无向图看起来没啥呀,怎么会有这种数据结构呢?
    ' U0 ^! {) @( [$ ?# i不知道你是怎么想的,我刚开始学的时候就有这种疑问,这是什么神仙数据结构哇,还会有应用场景?& L* e$ a& L# a
    ' W% N+ O& n, X! d
    既然有疑惑,那就给个应用场景:
    " S5 x! H( z( p* x* ]* C8 V假设,现在你和我是微信好友,那是不是应该你的好友列表里面有我,我的好友里面有你,这样咱们才是好友对不对~& L' p, p% o& s5 b# |1 s
    那在数据库中如何表示呢?吼~这个时候无向图就登场了
    2 C) N$ N( M4 r/ V# V5 M% ^- A你和我是微信好友,那就在咱俩之间来条线,表示咱俩之间有关系,一条线就解决了问题,真是完美至极啊
    $ n; n) [+ N8 u; b# v; \3 [1 z2 I假设,(怎么又是假设,哈哈哈)上图中表示的就是 A,B,C,D,E,F 之间的关系,那你可能就发现问题了,有的顶点线比较多,比如 D 有四条线,有的就相对少一些,比如 B 有两条线.这些线就表示顶点的度( degree ).这个概念有啥用?
    ; u1 F- c$ T& g( R# G能一眼看出来谁的好友多!那这个功能有啥用?(好吧,这个功能好像是有点儿鸡肋,不过也算是一个应用场景% O" O9 |" h! {. b) f

    7 j1 j6 r3 p5 h& N; U2 d$ D+ L有向图, ^  w2 o% }- F( v, n
    2.jpg 5 a3 h: a$ Z! y8 t: L
    看到上面的无向图,基础不错的小伙伴肯定会说了,我还知道有向图呢!
    ! s5 e% g: E) D+ y$ g呦呵,不错,有向图就是下面这个样子:
    0 z! g; E' H. B; K. D$ r
    9 K% d( d1 ]" C在无向图中,咱们知道一个顶点有多少条边,就说它的度为多少.
      H& J4 B) d: J, U; G$ v* F在有向图中呢,有指向顶点的,也有从顶点指出去的,基于无向图的概念,咱们把从这个顶点指出去的边称为出度,指向该顶点的边称为入度.
    3 U" B( |4 y, C! b* B- T6 A& d那么有向图会应用在哪些场景呢?微信好友这个场景是不太可以了
    8 R" B1 d2 k+ v' Z那么微博呢?$ N/ U; b# X1 m5 S% {
    微博和微信有什么不一样呢?微信是你和我是好友,那么咱们的好友列表里一定是要有彼此的,拉黑或者删除彼此了,那就不能互相发送消息了.' a2 s# ]8 @+ G4 Y3 G# J0 U8 f
    但是微博呢?你关注了我,并不代表我就要关注你对吧?看到这里有没有一种豁然开朗的感觉~% j$ J7 x3 Y& j' T
    那么我关注了多少人就是出度,多少人关注了我就是入度.
    / ], F( {# F- w) X( h$ K这样带入理解是不是会比较好一点儿?(我可真是个天才,哈哈哈* A  M6 D* e; N

    6 p6 }- |6 _- i, E$ c7 O带权图- B- G( U3 T! V% R+ S1 @& D
    3.jpg 1 q! P4 c) y4 h4 o
    看完了无向图,有向图,相信就有人说,我还见过带权图!(陈独秀给我坐下!
    % W  f, z9 P) M& U带权图长啥样呢?就下面这个样子:
    - Y& `* N# ?" R. H: x/ l! b* l* o% e' ?# W6 F
    懵逼了,这每条边上的数字是个什么鬼呦
    9 y3 ^6 {# v2 `, @& F别急,咱们来个场景:大家都玩 QQ 嘛?(别跟我说不玩,配合一下嘛…& `- G+ W- t  M. `( I* O
    玩 QQ 的话,一定知道有 QQ 空间,然后空间里面有个「谁在意我」「我在意谁」的功能,就是下图:
    8 B( W8 T9 N, C/ {$ [% ?! H& R- |( L$ ]; l: \' g. j
    那么有没有好奇过呢? QQ 怎么知道我在意谁,谁在意我呢?* s& C- y$ q% J2 I
    就是通过带权图哇- C7 O0 ^, W7 d9 v
    你访问了一个人的空间,这条边的权重就增加一点儿;别人访问了你的空间,那这条边的权重就增加一点儿;这段时间你们两个人聊天聊得比较频繁,来个小火花,顺便在你们两者之间的边权重增加一点儿.然后根据这些边的权重从大到小排序就得出了「谁在意我」「我在意谁」, S6 m7 p8 K+ v3 z& ~
    6 K! B5 Y* o) w3 g0 l* }6 L. E
    到这里,上面的一切理解都还 OK ?
    ' X6 X. |8 r0 k- _( K4 O  q4 F那咱们继续.图是怎么表示的呢?/ O1 a8 C. K3 q) B& W8 K
    图这种数据结构,再怎么画顶点,画边,到最后在物理结构上是怎么存储的呢?, n9 A  |7 z& q; b: s4 c' z) N
    别急,你所疑惑的,我都帮你想到了, s2 P: m  k# P5 {! R1 o

    ( R" L7 z1 Z7 n$ U+ p! b图的存储方法
    ' v: T) V6 A& I1 P: T4 d
    1 C3 x* y- h1 o9 i图的存储方法主要有以下两种:- z6 O# ]5 T! G. h
    ' H+ f4 s  ^9 U8 R1 ]' l& Y
    邻接矩阵; _. t5 w5 A+ i0 `
      A+ J& X1 S* ^8 P; `$ I: p0 U/ K8 \
    邻接矩阵的底层依赖一个二维数组.对于无向图来说,如果 i 与 j 之间有边,那就将 A[j] 和 A[j] 标记为 1 ;对于无向图来说,如果 i 指向 j ,那么 A[j] 值为 1 ,如果 j 指向 i ,那么 A[j] 值为 1 ;对于带权图来说, A[j] 存储的值就不是 1 了,而是对应的权重值.所以这是图最直观的一种存储方法.$ O; g( t, o' ^: \
    啥,你跟我说这还不直观?该不会是没有看下图吧:
    8 x. A& h" J, N0 I: |& x" M" d 4.jpg " l% o& A0 E* Z$ n
    但是你发现问题了嘛,这样看起来确实是直观了很多,但是很浪费空间有没有!比如无向图,如果 A[j] 为 1 ,那么 A[j] 肯定也是 1 ,多存储 A[j] 根本没啥必要.就像买东西,明明一块钱能买到的东西,为啥非要花两块钱?; o5 R; V9 C  y) c- h8 F6 c3 r
    所以如果使用邻接矩阵来表示的话,一定要清楚它的缺点.
    9 T! O) a' Q6 Q. u, W3 V% r) O# m/ d* a5 d
    但这并不是说,使用邻接矩阵来存储就没啥优点.这天底下哪儿有那么绝对的事情呢.
    5 G6 I) `* q  v6 i* v3 Y$ H首先,邻接矩阵的存储方式简单,直接,所以当我们需要获取两个顶点之间的关系时,相信我没有比这种存储结构更高效的了.! J* B# ]! j3 p5 e4 n
    还有就是使用邻接矩阵存储图的另外一个优点就是方便计算,因为可以将很多图的运算转换成矩阵之间的运算.: S( Z9 X# v3 f3 G' @

    ( @/ q& o8 |6 ~3 ~0 w. r+ c邻接表3 V. q& y) E: t+ q# u) D: p
    ! y) ~' w4 J# g  K( ?7 W
    先来看图:+ N: J# Q6 B0 G! O8 N' {
    5.jpg
    6 b. `* c3 ]# J: l
    - ?8 z3 T, \8 N" r  e) n4 R( f乍一看,这不是散列表嘛!每个顶点对应一条链表,链表中存储的是与这个顶点相连接的其他顶点.
    & Q) ]2 K& p3 b5 f2 S! ?$ z嘿嘿,直觉超棒!
    4 `) M& M7 `% y% \" b( {. p# D% t. Z% ], u& E/ }; a
    如果你对散列表熟悉的话,应该知道,在散列表中,如果链太长了,会导致冲突概率增大,复杂度也蹭的一下升高.而且吧,链表的存储方式你也知道,不是连续的,所以相对于数组来说, CPU 读取就会慢一些,相对于邻接矩阵的存储方式,在邻接表中查询两个顶点之间的关系就没那么高效了.
    / T6 y4 }2 V8 \) `' `所以在实际开发中要注意遇到这种情况该如何处理,或者在刚开始的时候就直接设计好实现方式.比如可以将邻接表中的链表改为平衡二叉树,或者红黑树.- D+ V: o! m' X9 E8 f, Y5 ~
    : d% B$ `8 \" O# z3 K  O5 ?' O
    我觉得对于数据结构来说,没有最好的,只有最合适的~# V9 |. w/ ]8 `: X) j

    1 c8 P3 b5 t# M- b参考# Y: Q& [0 J/ I! X- j$ o) ]
    6 G2 _8 ]/ w5 U
    极客时间—<数据结构与算法之美>" o; [9 v6 i7 h' l
    ————————————————
    $ g+ E; F+ ~; |: S  H版权声明:本文为CSDN博主「郑璐璐」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
    ' S' [8 _# H! A' v! u4 u原文链接:https://blog.csdn.net/zll_0405/article/details/105209800
    ( J; h3 j9 F2 f) t* c
    ; n/ ~9 a  ^& g, A* `* V1 \; R9 m& G. n: Q  `& M% Z- ], n! k
    zan
    转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-6-10 00:30 , Processed in 0.416723 second(s), 53 queries .

    回顶部