Python小白的数学建模课-图论的基本概念$ j6 l% l |2 R7 _2 Z- ^
. T/ t; Y) l/ A! x- 图论中所说的图,不是图形图像或地图,而是指由顶点和边所构成的图形结构。
- 图论不仅与拓扑学、计算机数据结构和算法密切相关,而且正在成为机器学习的关键技术。
- 本系列结合数学建模的应用需求,来介绍 NetworkX 图论与复杂网络工具包的基本功能和典型算法。/ F0 k7 N) S) Q; [2 o
4 Z$ |+ G, T9 R8 x, Z( q
1. 图论1.1 图论是什么
+ i* h; i1 h- o1 C( P* _" E! @图论〔Graph Theory〕以图为研究对象,是离散数学的重要内容。图论不仅与拓扑学、计算机数据结构和算法密切相关,而且正在成为机器学习的关键技术。+ c( C8 B/ i/ X; s, y" |% X# S# E
( e H! j7 M6 ~图论中所说的图,不是指图形图像(image)或地图(map),而是指由顶点(vertex)和连接顶点的边(edge)所构成的关系结构。
/ u) g/ \, Y0 b8 a \! B% A; G
( h! b% i! Z- c, I( \图提供了一种处理关系和交互等抽象概念的更好的方法,它还提供了直观的视觉方式来思考这些概念。! V T' q9 {6 I6 P. A g3 K7 A4 h
$ X8 C+ _1 x4 k
1.2 NetworkX 工具包) j; `# {; t" O, b0 v5 z) Y
NetworkX 是基于 Python 语言的图论与复杂网络工具包,用于创建、操作和研究复杂网络的结构、动力学和功能。
' d! S s. _7 f9 W( z& K+ N/ ]
1 m, {% }+ F( @: TNetworkX 可以以标准和非标准的数据格式描述图与网络,生成图与网络,分析网络结构,构建网络模型,设计网络算法,绘制网络图形。: I; X, W$ f7 [, _
; ?) {2 t% M: C9 ~% dNetworkX 提供了图形的类、对象、图形生成器、网络生成器、绘图工具,内置了常用的图论和网络分析算法,可以进行图和网络的建模、分析和仿真。+ K Y3 d- E! P# M: N
# b3 _) N0 S# VNetworkX 的功能非常强大和庞杂,所涉及内容远远、远远地超出了数学建模的范围,甚至于很难进行系统的概括。本系列结合数学建模的应用需求,来介绍 NetworkX 图论与复杂网络工具包的基本功能和典型算法。
- s! v, n" I$ {
S G4 {8 e0 P2 p' Z' g![]()
/ h5 V( C, @3 ]2 k2、图、顶点和边的创建与基本操作5 U/ L, ~$ i1 G5 m
图由顶点和连接顶点的边构成,但与顶点的位置、边的曲直长短无关。 Networkx 支持创建简单无向图、有向图和多重图;内置许多标准的图论算法,节点可为任意数据;支持任意的边值维度,功能丰富,简单易用。 2.1 图的基本概念0 V( z. Q+ d, S K0 J( ]6 X
图(Graph):图是由若干顶点和连接顶点的边所构成关系结构。$ h: j7 P; ~$ d0 F- j; B) N! B) F
顶点(Node):图中的点称为顶点,也称节点。4 V- A0 Q* M+ r# O
边(Edge):顶点之间的连线,称为边。
k$ z1 ]0 q* ]% m) K平行边(Parallel edge):起点相同、终点也相同的两条边称为平行边。
3 A1 `9 {! D Z循环(Cycle):起点和终点重合的边称为循环。
4 \! W9 ^1 B# ?1 A: U有向图(Digraph):图中的每条边都带有方向,称为有向图。
. z" C7 V# ]$ p% [0 D! k) V1 U3 D: X无向图(Undirected graph):图中的每条边都没有方向,称为无向图。
- w4 p, a d; f赋权图(Weighted graph):图中的每条边都有一个或多个对应的参数,称为赋权图。该参数称为这条边的权,权可以用来表示两点间的距离、时间、费用。
: Y6 R1 P4 H: h! V. }0 E度(Degree):与顶点相连的边的数量,称为该顶点的度。
: T- O, H$ l5 L' t9 N f W
: h. ^- l. Q2 b: C2.2 图、顶点和边的操作. s a9 w1 r' O9 Q/ Z' q3 | f* q& }* B
Networkx很容易创建图、向图中添加顶点和边、从图中删除顶点和边,也可以查看、删除顶点和边的属性。
1 l, O) y( Z3 |% `' h' w. C+ v' d+ l4 m7 u1 R6 G7 w" j
2.2.1 图的创建Graph() 类、DiGraph() 类、MultiGraph() 类和 MultiDiGraph() 类分别用来创建:无向图、有向图、多图和有向多图。定义和例程如下:
) |9 ]" x6 q1 a7 k& I( E! N M9 u8 w$ |6 l% p9 t; I: q
class Graph(incoming_graph_data=None, **attr)
3 m% d) y: m) f. Q1 e) Himport networkx as nx # 导入 NetworkX 工具包
" L+ ^0 M/ ~) s- O! q0 n) v3 z6 h5 W
# 创建 图
; t9 p0 V' `7 J4 hG1 = nx.Graph() # 创建:空的 无向图! j( D) p. i0 \ P9 h5 m
G2 = nx.DiGraph() #创建:空的 有向图
9 c8 J# J; M R) Z7 CG3 = nx.MultiGraph() #创建:空的 多图/ |: j' n- r3 Z& ?1 n$ T
G4 = nx.MultiDiGraph() #创建:空的 有向多图* O8 ?0 K/ Z6 D2 w; x
3 x" w+ J' g/ a- S1 }! v# Q
" ?7 n' \2 x! c5 q
2.2.2 顶点的添加、删除和查看+ k# x( e) q1 j
图的每个顶点都有唯一的标签属性(label),可以用整数或字符类型表示,顶点还可以自定义任意属性。 顶点的常用操作:添加顶点,删除顶点,定义顶点属性,查看顶点和顶点属性。定义和例程如下: ' o7 L) ~, ?6 V p# }
Graph.add_node(node_for_adding, **attr)
' Q# n' }/ `( g/ u) E) u- HGraph.add_nodes_from(nodes_for_adding, **attr)
8 c7 x4 N6 F6 ?5 \Graph.remove_node(n)
x8 o; t$ N% w, a p1 ^8 |Graph.remove_nodes_from(nodes)
1 d' |& K* E5 L3 j, X! g9 ^% T7 V& O p& t. G( {
# 顶点(node)的操作
( P0 \# @7 Z* |& Z$ [# 向图中添加顶点
+ Y' f% I7 e( h/ e4 }G1.add_node(1) # 向 G1 添加顶点 1
8 N$ u; Y$ {) Z8 y; IG1.add_node(1, name='n1', weight=1.0) # 添加顶点 1,定义 name, weight 属性$ U# G# |0 a9 R2 Y5 @' }' G' @
G1.add_node(2, date='May-16') # 添加顶点 2,定义 time 属性8 l% k# G0 Z4 j, b8 J$ ^
G1.add_nodes_from([3, 0, 6], dist=1) # 添加多个顶点,并定义属性: _& l4 B( |6 c" v
G1.add_nodes_from(range(10, 15)) # 向图 G1 添加顶点 10~14- `5 f1 J" D& X
1 r1 V# i/ s3 e& t% S# 查看顶点和顶点属性2 r% j0 z0 `# h. n
print(G1.nodes()) # 查看顶点列表
' E' N+ x. C, T& T# [1, 2, 3, 0, 6, 10, 11, 12, 13, 14]6 o6 F/ j0 h" h4 e% v8 ^" p
print(G1._node) # 查看顶点属性
. F( g: S! K& e1 L+ Z. G# {1: {'name': 'n1', 'weight': 1.0}, 2: {'date': 'May-16'}, 3: {'dist': 1}, 0: {'dist': 1}, 6: {'dist': 1}, 10: {}, 11: {}, 12: {}, 13: {}, 14: {}}
: D$ ?1 D4 f% D& z z, k5 C1 V7 m" h1 d4 v. A4 Z, m
# 从图中删除顶点# b; n7 }8 }- q; C1 n$ W1 e
G1.remove_node(1) # 删除顶点
" Q$ E! X1 L! v! nG1.remove_nodes_from([1, 11, 13, 14]) # 通过顶点标签的 list 删除多个顶点
1 \5 H/ M+ ]& T% d: l' ]2 [0 e+ Z; wprint(G1.nodes()) # 查看顶点) K0 C0 p4 A! z( d
# [2, 3, 0, 6, 10, 12] # 顶点列表' v. P) W8 t/ @+ w: ]& _
2.2.3 边的添加、删除和查看边是两个顶点之间的连接,在 NetworkX 中 边是由对应顶点的名字的元组组成 e=(node1,node2)。边可以设置权重、关系等属性。 边的常用操作:添加边,删除边,定义边的属性,查看边和边的属性。向图中添加边时,如果边的顶点是图中不存在的,则自动向图中添加该顶点。 Graph.add_edge(u_of_edge, v_of_edge, **attr)% p6 m' w# y& z/ U9 p3 ?+ Y
Graph.add_edges_from(ebunch_to_add, **attr)
3 G- |7 S# U" e! ~Graph.add_weighted_edges_from(ebunch_to_add, weight=‘weight’, **attr)
$ w" m6 O, T; R C1 Y
% h$ A" q3 N. h& x3 O/ f) X% k1 f# 边(edge)的操作
; P8 g: W3 y$ K: U% q# 向图中添加边( M6 _- I( R1 O. d# {# |3 }8 P" O
G1.add_edge(1,5) # 向 G1 添加边,并自动添加图中没有的顶点
! M* m1 v4 n4 _% GG1.add_edge(0,10, weight=2.7) # 向 G1 添加边,并设置边的属性9 b. b7 V3 X, V) M0 \) W' ~0 `
G1.add_edges_from([(1,2,{'weight':0}), (2,3,{'color':'blue'})]) # 向图中添加边,并设置属性
' e# }6 e2 {5 d3 `0 }/ @* VG1.add_edges_from([(3,6),(1,2),(6,7),(5,10),(0,1)]) # 向图中添加多条边8 _: u+ L# p7 ^& A. c
G1.add_weighted_edges_from([(1,2,3.6),[6,12,0.5]]) # 向图中添加多条赋权边: (node1,node2,weight)
6 ]2 R7 M- e+ k' [. t& }print(G1.nodes()) # 查看顶点' ?2 `: y1 }8 [3 d) E
# [2, 3, 0, 6, 10, 12, 1, 5, 7] # 自动添加了图中没有的顶点
1 s2 `+ }, H, {* M y* t: ]% P( u0 S& G% ^3 T
# 从图中删除边( s4 C6 }6 D% u/ t
G1.remove_edge(0,1) # 从图中删除边 0-1
9 P6 P# W- }3 h% UG1.remove_edges_from([(2,3),(1,5),(6,7)]) # 从图中删除多条边
) u0 g0 p/ k6 j% C, N' w
" L. c( J7 k3 W" m/ ]$ T' t# 查看 边和边的属性
8 X% Y4 j/ D) g" j2 W0 zprint(G1.edges) # 查看所有的边- Z- R5 D, o# U
[(2, 1), (3, 6), (0, 10), (6, 12), (10, 5)]
( p& o% D* q/ R1 ]8 U p4 Vprint(G1.get_edge_data(1,2)) # 查看指定边的属性: d* x* K& ^( u* o
# {'weight': 3.6}: ] ], N7 z! V
print(G1[1][2]) # 查看指定边的属性
9 i, x; }) D: n4 A1 G+ I# {'weight': 3.6}
! N* `$ }4 u' E) ?. {' j5 kprint(G1.edges(data=True)) # 查看所有边的属性
5 H% o3 H2 _# |, B$ a6 e Q T3 ~# [(2, 1, {'weight': 3.6}), (3, 6, {}), (0, 10, {'weight': 2.7}), (6, 12, {'weight': 0.5}), (10, 5, {})]% G* Y4 Z& z9 a% x3 ?0 A
! R6 M3 g/ P' n; x! p2.2.4 查看图、顶点和边的信息" R% l$ @# `1 E e# y% o3 e% K
9 d }3 e8 H7 F' u2 P. w/ D A+ O
# 查看图、顶点和边的信息' ^& J+ \ r9 J0 h9 M
print(G1.nodes) # 返回所有的顶点 [node1,...]9 y! Q4 m! P( O( ?3 s
# [2, 3, 0, 6, 10, 12, 1, 5, 7]: I, Y# X( [" o' k8 q: { y
print(G1.edges) # 返回所有的边 [(node1,node2),...]# T$ F, D7 b$ c) |0 L; i) c
# [(2, 1), (3, 6), (0, 10), (6, 12), (10, 5)]
7 O; f% J5 r0 A4 h- @9 t9 X! kprint(G1.degree) # 返回各顶点的度 [(node1,degree1),...]$ o( T- b: Q& m' j! ?9 y2 c
# [(2, 1), (3, 1), (0, 1), (6, 2), (10, 2), (12, 1), (1, 1), (5, 1), (7, 0)]
, s& z& x$ d7 Y% h7 w( l" _4 v! d# wprint(G1.number_of_nodes()) # 返回顶点的数量
0 d1 A6 a+ S* ^3 `' S# 9
" t/ L( @0 E% H4 o1 |: Nprint(G1.number_of_edges()) # 返回边的数量$ D+ q/ z0 B& K; E$ V' |% @4 d
# 51 C3 v+ Y5 x' g, S
print(G1[10]) # 返回与指定顶点相邻的所有顶点的属性7 `9 C$ u0 y1 S* b1 y7 m8 J
# {0: {'weight': 2.7}, 5: {}}
2 N& y' W: G) u# \print(G1.adj[10]) # 返回与指定顶点相邻的所有顶点的属性
- E: U. m* E- T/ @% K6 }1 g8 w# {0: {'weight': 2.7}, 5: {}}: Y4 @0 Y# y9 G( |
print(G1[1][2]) # 返回指定边的属性, c& z- l8 K% t/ b0 h
# {'weight': 3.6}
% u/ A# d+ Q' t5 i, S3 |print(G1.adj[1][2]) # 返回指定边的属性& G% N% m; g/ C$ ^1 q) D
# {'weight': 3.6}! ]) B+ `- x" R2 [, X( v. H
print(G1.degree(10)) # 返回指定顶点的度
4 v& s! I0 m* Y/ `0 N, Q/ D# 2
! n) z1 q; u, S' h
- u) R s: L, P* n# j! G$ V( Gprint('nx.info:',nx.info(G1)) # 返回图的基本信息/ Y1 q% y% N' r+ X- h) o( s
print('nx.degree:',nx.degree(G1)) # 返回图中各顶点的度
* ?+ @; N$ ?- H5 Y- Xprint('nx.density:',nx.degree_histogram(G1)) # 返回图中度的分布
F$ n; W! R- e% w( D+ g7 x" z8 d& sprint('nx.pagerank:',nx.pagerank(G1)) # 返回图中各顶点的频率分布
8 W- B0 m& l. Y* A7 q# q
1 N9 B& @, R; p0 ~- d0 f V9 i
+ v: a+ A2 l- {" ]' V
8 C" T- |7 r. O8 s
$ @4 z1 [& T4 E9 }& g# |' ~' ^, U. u/ ?" y. g9 j/ W
2.3 图的属性和方法图的方法2 w% D% ]" h; _
) C- p$ S- o0 O- J% i. S+ _ i方法 说明
& C! B9 \$ T9 P: nG.has_node(n) 当图 G 中包括顶点 n 时返回 True2 N8 m$ L) \% `0 u( h
G.has_edge(u, v) 当图 G 中包括边 (u,v) 时返回 True
* `" J2 v. Q+ [; ?G.number_of_nodes() 返回 图 G 中的顶点的数量. y7 U; [# F5 G5 N- s1 O
G.number_of_edges() 返回 图 G 中的边的数量
9 @, l. G5 t6 c0 TG.number_of_selfloops() 返回 图 G 中的自循环边的数量! c9 ?) j& n/ p% J
G.degree([nbunch, weight]) 返回 图 G 中的全部顶点或指定顶点的度% x- r: L" E# K. [' J
G.selfloop_edges([data, default]) 返回 图 G 中的全部的自循环边8 m: a% p5 ]' R- W' c% p3 E
G.subgraph([nodes]) 从图 G1中抽取顶点[nodes]及对应边构成的子图
5 \$ ?# d! c: H9 o* S4 t! i2 k6 v8 ]! T" kunion(G1,G2) 合并图 G1、G2
. ?+ ?" {9 n* L" U+ s4 h" fnx.info(G) 返回图的基本信息' i1 C+ ]. o* n* {, M m P8 \# @3 C
nx.degree(G) 返回图中各顶点的度
$ D6 u* t- ~+ V# Vnx.degree_histogram(G) 返回图中度的分布
4 a8 Y, ?! a) l6 Nnx.pagerank(G) 返回图中各顶点的频率分布
2 j: O* @5 T: W) Y# q' e- U/ p0 ~nx.add_star(G,[nodes],**attr) 向图 G 添加星形网络
+ g+ m& o1 u% W/ O2 Pnx.add_path(G,[nodes],**attr) 向图 G 添加一条路径
# k. L8 _& a+ m2 n5 m! k0 Z7 Knx.add_cycle(G,[nodes],**attr) 向图 G 添加闭合路径
! u9 E- }) a6 V M$ [
8 Y- b5 q1 R' t( X* W7 P0 `% i7 x4 e1 }7 C# I0 e4 e4 m9 E
例程:4 i, N3 J) v5 S4 }- g" h
G1.clear() # 清空图G1$ t: \# d! F* h& p' o
nx.add_star(G1, [1, 2, 3, 4, 5], weight=1) # 添加星形网络:以第一个顶点为中心
8 j/ h+ o8 F% `: l( b# [(1, 2), (1, 3), (1, 4), (1, 5)]
* I9 Z3 T3 B4 I( u6 Pnx.add_path(G1, [5, 6, 8, 9, 10], weight=2) # 添加路径:顺序连接 n个节点的 n-1条边
& U* j% B4 ?9 V" I' \9 u. v# [(5, 6), (6, 8), (8, 9), (9, 10)]- H' y8 O5 Z8 d% }
nx.add_cycle(G1, [7, 8, 9, 10, 12], weight=3) # 添加闭合回路:循环连接 n个节点的 n 条边
/ p1 x# U, u" _; ], t5 N t7 M, v) N9 l# [(7, 8), (7, 12), (8, 9), (9, 10), (10, 12)]
& O: G% H; @3 K% G, p1 [print(G1.nodes) # 返回所有的顶点 [node1,...]8 Y# u" M: ^# z! ~8 F- t
nx.draw_networkx(G1)) d/ Q g6 m+ Y0 V# B
plt.show()) c4 {" v; o" n# J" K
G+ E6 t+ z7 Z* a. n/ ]/ |" l6 @G2 = G1.subgraph([1, 2, 3, 8, 9, 10])
# ^' t! o' x% @7 H; GG3 = G1.subgraph([4, 5, 6, 7])
$ V( q2 L o5 _& l$ u; k+ v1 [8 hG = nx.union(G2, G3)
" d) O* z$ p; G8 \) b) Gprint(G.nodes) # 返回所有的顶点 [node1,...]# [( r5 B2 U7 T' \6 K
# [1, 2, 3, 8, 9, 10, 4, 5, 6, 7]
+ ?# m; G- W! Q) {- \: U2 e3 W) {* i. q+ K
2 U% T+ b" Q+ d; o# q0 H3、图的绘制与分析3.1 图的绘制, }7 Z3 ^/ l' ~* V4 C3 T7 V; {
可视化是图论和网络问题中很重要的内容。NetworkX 在 Matplotlib、Graphviz 等图形工具包的基础上,提供了丰富的绘图功能。9 }' Q, P) t! Y4 v9 w
3 |' x9 S- E( i) o
本系列拟对图和网络的可视化作一个专题,在此只简单介绍基于 Matplotlib 的基本绘图函数。基本绘图函数使用字典提供的位置将节点放置在散点图上,或者使用布局函数计算位置。1 C3 P; \" |5 s% N
+ h: p$ j9 {" q2 p! @' N方法 说明
' w$ \9 s- M2 b4 l5 w+ O1 N. V2 S3 sdraw(G[,pos,ax]) 基于 Matplotlib 绘制 图 G
7 M0 N. [" B( \8 [0 fdraw_networkx(G[, pos, arrows, with_labels]) 基于 Matplotlib 绘制 图 G4 g9 s; A+ h& m' D: ~) q
draw_networkx_nodes(G, pos[, nodelist, . . . ]) 绘制图 G 的顶点
8 ^7 z) W, C) H: P! F% E$ tdraw_networkx_edges(G, pos[, edgelist, . . . ]) 绘制图 G 的边) w+ C! i# U. J O( `* }
draw_networkx_labels(G, pos[, labels, . . . ]) 绘制顶点的标签
) r( }% j; f! N, mdraw_networkx_edge_labels(G, pos[, . . . ]) 绘制边的标签
7 W& X7 ]. r5 ?' P
4 G6 o' \+ l/ p+ \( G7 d9 b
1 S: S- Z* ?/ j3 g8 ~其中,nx.draw() 和 nx.draw_networkx() 是最基本的绘图函数,并可以通过自定义函数属性或其它绘图函数设置不同的绘图要求。
/ \1 m- X9 G4 ~- V) b$ ldraw(G, pos=None, ax=None, **kwds) draw_networkx(G, pos=None, arrows=True, with_labels=True, **kwds) * N9 s$ ~1 |2 O( `
常用的属性定义如下:/ z* X6 J, b* r
7 c' {! A# A9 b; L/ G% z, V; V- Y
‘node_size’:指定节点的尺寸大小,默认300# y3 p) @/ R2 J' F
‘node_color’:指定节点的颜色,默认红色
$ H' r+ q; p, Q‘node_shape’:节点的形状,默认圆形6 i: W; x3 w. k! g! j
'‘alpha’:透明度,默认1.0,不透明0 H( M5 W( W+ n! F
‘width’:边的宽度,默认1.0+ P8 K5 l, ^7 r; A8 E! K( _" k
‘edge_color’:边的颜色,默认黑色' ]: i" e# i Y$ p7 \' Q6 m( {2 E
‘style’:边的样式,可选 ‘solid’、‘dashed’、‘dotted’、‘dashdot’# j ?0 h9 M7 A
‘with_labels’:节点是否带标签,默认True6 R' {8 g: s& }# I3 w
‘font_size’:节点标签字体大小,默认12 B- n. q: k. B/ E* O( ]6 }' F
‘font_color’:节点标签字体颜色,默认黑色4 c4 s" `: O2 a5 U# q9 f4 U
5 T. t/ ~* }: ~, N& Q : Q3 [" {3 j% V0 ?" ]
3.2 图的分析NetwotkX 提供了图论函数对图的结构进行分析
0 _' u% C) H+ O0 P% Y子图, C7 a- C/ g+ ?! R. w. [
- 子图是指顶点和边都分别是图 G 的顶点的子集和边的子集的图。
- subgraph()方法,按顶点从图 G 中抽出子图。例程如前。
4 L4 o9 q$ y7 p J2 e# g% e 连通子图
; [" Z: k8 Y8 I) [
- v7 }# T6 ]8 A; x( X& |- 如果图 G 中的任意两点间相互连通,则 G 是连通图。
- [color=rgba(0, 0, 0, 0.749019607843137)]connected_components()方法,返回连通子图的集合。$ u# f0 ^' T3 P9 Y
[color=rgba(0, 0, 0, 0.749019607843137)]G = nx.path_graph(4)[color=rgba(0, 0, 0, 0.749019607843137)]nx.add_path(G, [7, 8, 9])[color=rgba(0, 0, 0, 0.749019607843137)]# 连通子图[color=rgba(0, 0, 0, 0.749019607843137)]listCC = [len(c) for c in sorted(nx.connected_components(G), key=len, reverse=True)][color=rgba(0, 0, 0, 0.749019607843137)]maxCC = max(nx.connected_components(G), key=len)[color=rgba(0, 0, 0, 0.749019607843137)]print('Connected components:{}'.format(listCC)) # 所有连通子图[color=rgba(0, 0, 0, 0.749019607843137)]# Connected components:[4, 3][color=rgba(0, 0, 0, 0.749019607843137)]print('Largest connected components:{}'.format(maxCC)) # 最大连通子图[color=rgba(0, 0, 0, 0.749019607843137)]# Largest connected components:{0, 1, 2, 3}[color=rgba(0, 0, 0, 0.749019607843137)]强连通如果有向图 G 中的任意两点间相互连通,则称 G 是强连通图。strongly_connected_components()方法,返回所有强连通子图的列表。# 强连通G = nx.path_graph(4, create_using=nx.DiGraph())nx.add_path(G, [3, 8, 1])# 找出所有的强连通子图con = nx.strongly_connected_components(G)print(type(con),list(con))# <class 'generator'> [{8, 1, 2, 3}, {0}]弱连通如果一个有向图 G 的基图是连通图,则有向图 G 是弱连通图。weakly_connected_components()方法,返回所有弱连通子图的列表。# 弱连通G = nx.path_graph(4, create_using=nx.DiGraph()) #默认生成节点 0,1,2,3 和有向边 0->1,1->2,2->3nx.add_path(G, [7, 8, 3]) #生成有向边:7->8->3con = nx.weakly_connected_components(G)print(type(con),list(con))# <class 'generator'> [{0, 1, 2, 3, 7, 8}]
# |+ @8 O- s% {* o6 S- V: {
) q, `/ x, F+ `
t5 d" v p- O) h/ k
9 ~, E4 d3 R, m; K+ \: c8 N- c0 O
% J4 J6 z& T1 L+ u; s; M |