Python小白的数学建模课-图论的基本概念
' G$ M" b( R: U4 R8 G" r8 T8 @$ f' c/ ?( p6 T
- 图论中所说的图,不是图形图像或地图,而是指由顶点和边所构成的图形结构。
- 图论不仅与拓扑学、计算机数据结构和算法密切相关,而且正在成为机器学习的关键技术。
- 本系列结合数学建模的应用需求,来介绍 NetworkX 图论与复杂网络工具包的基本功能和典型算法。. Z/ b; q; \- q4 F" \
, y3 M8 u- T& ?0 C/ M7 H1. 图论1.1 图论是什么0 p$ m. Y$ D. W- b9 N
图论〔Graph Theory〕以图为研究对象,是离散数学的重要内容。图论不仅与拓扑学、计算机数据结构和算法密切相关,而且正在成为机器学习的关键技术。; A: P+ r4 L. k3 y( j
1 f% g" |2 d f- s- a$ Y4 c
图论中所说的图,不是指图形图像(image)或地图(map),而是指由顶点(vertex)和连接顶点的边(edge)所构成的关系结构。% X4 B) p5 l9 n- x5 Y; d
- b; `2 R& m# I# L/ u$ X图提供了一种处理关系和交互等抽象概念的更好的方法,它还提供了直观的视觉方式来思考这些概念。
3 G E) \' M2 v B" J3 ? ]% A0 ]0 J8 m
1.2 NetworkX 工具包
, p8 E9 S( m: _6 V) ?- K2 m2 uNetworkX 是基于 Python 语言的图论与复杂网络工具包,用于创建、操作和研究复杂网络的结构、动力学和功能。
, x0 a8 N( w, Q* p k3 S, e; @1 E
2 g+ @. f. Q$ `" B f. WNetworkX 可以以标准和非标准的数据格式描述图与网络,生成图与网络,分析网络结构,构建网络模型,设计网络算法,绘制网络图形。
$ a! |9 s/ @* N) t( j( Q0 p8 b$ E: Q! o
NetworkX 提供了图形的类、对象、图形生成器、网络生成器、绘图工具,内置了常用的图论和网络分析算法,可以进行图和网络的建模、分析和仿真。+ Z% Y3 L0 g5 H
, [- T+ F, g. O7 _ p( zNetworkX 的功能非常强大和庞杂,所涉及内容远远、远远地超出了数学建模的范围,甚至于很难进行系统的概括。本系列结合数学建模的应用需求,来介绍 NetworkX 图论与复杂网络工具包的基本功能和典型算法。
" K% h7 m) T6 M' O# q
1 `" A k/ v' S s% P" s1 a; Z6 {![]()
5 i3 J2 l+ v4 Y, M8 s X9 t2、图、顶点和边的创建与基本操作
2 h& K5 e, ?6 h5 l4 R: P( ~- t图由顶点和连接顶点的边构成,但与顶点的位置、边的曲直长短无关。 Networkx 支持创建简单无向图、有向图和多重图;内置许多标准的图论算法,节点可为任意数据;支持任意的边值维度,功能丰富,简单易用。 2.1 图的基本概念% S2 j- g. P, G% ]0 r9 ?
图(Graph):图是由若干顶点和连接顶点的边所构成关系结构。
! c% _( G/ n. Z% C c8 e- ^+ R顶点(Node):图中的点称为顶点,也称节点。
& k3 y) [0 S( f8 b) U) `+ w0 i边(Edge):顶点之间的连线,称为边。
% h. e& p( D8 f6 }* ]5 u! i7 \平行边(Parallel edge):起点相同、终点也相同的两条边称为平行边。
. _4 \: V) O* C. ?- S1 q循环(Cycle):起点和终点重合的边称为循环。, Z3 G+ b# p c! f' X2 L
有向图(Digraph):图中的每条边都带有方向,称为有向图。5 _/ _. W% O+ g% _( o. |8 R% T
无向图(Undirected graph):图中的每条边都没有方向,称为无向图。
0 I# o0 c, k' e% n# Q7 X赋权图(Weighted graph):图中的每条边都有一个或多个对应的参数,称为赋权图。该参数称为这条边的权,权可以用来表示两点间的距离、时间、费用。1 {5 T0 y: R, S f7 v, ^5 A- T
度(Degree):与顶点相连的边的数量,称为该顶点的度。- r# D) S. T( y
$ X( |( E5 G/ _9 ^- z5 l: f, S* `
2.2 图、顶点和边的操作( W2 N9 j* G8 Q! r9 H# z# C
Networkx很容易创建图、向图中添加顶点和边、从图中删除顶点和边,也可以查看、删除顶点和边的属性。
& u* t9 E: d/ }7 s; O
0 k$ M$ h5 p; C; ?" P, p' N% M2.2.1 图的创建Graph() 类、DiGraph() 类、MultiGraph() 类和 MultiDiGraph() 类分别用来创建:无向图、有向图、多图和有向多图。定义和例程如下:
# Y; N% `9 p5 I6 h2 l
1 L" I/ D* C# @+ l* I; p9 d5 X. w4 Mclass Graph(incoming_graph_data=None, **attr)& L* U$ |1 L) ^
import networkx as nx # 导入 NetworkX 工具包
! s: d% f. ]* z$ r; V4 x( u) b( i) M8 v- v
# 创建 图% p2 V& D5 X* u6 r- Y9 W0 \8 b6 |# l
G1 = nx.Graph() # 创建:空的 无向图
" E- i& U! l) r$ D8 ?- KG2 = nx.DiGraph() #创建:空的 有向图8 s s6 D2 `5 `/ F% S! f
G3 = nx.MultiGraph() #创建:空的 多图
* E9 N4 b" L A0 n1 p7 ?1 @- f/ ZG4 = nx.MultiDiGraph() #创建:空的 有向多图% O- C" g+ I9 f" W* z) B, R
* c1 P- ~0 G0 {- z1 m* U! ~; f' w
$ O7 v/ _; Y8 Q9 S8 M
2.2.2 顶点的添加、删除和查看
2 A: ]1 |# |6 ~* Y9 n+ ]4 L图的每个顶点都有唯一的标签属性(label),可以用整数或字符类型表示,顶点还可以自定义任意属性。 顶点的常用操作:添加顶点,删除顶点,定义顶点属性,查看顶点和顶点属性。定义和例程如下: $ s; X" {* }" l2 P9 Z! @
Graph.add_node(node_for_adding, **attr)
" x4 x1 R* V. T: qGraph.add_nodes_from(nodes_for_adding, **attr)2 n% }' m0 B) D$ M
Graph.remove_node(n)& l! _/ E. {% Z" r8 j
Graph.remove_nodes_from(nodes)
( v/ f- C. N; @/ L
/ j7 S8 N# }) B6 j# 顶点(node)的操作$ Y, v; d1 g& @ v3 k
# 向图中添加顶点
& N5 Y w; A1 z1 S l n, T/ xG1.add_node(1) # 向 G1 添加顶点 1% G0 p8 ]: t8 d4 [+ z
G1.add_node(1, name='n1', weight=1.0) # 添加顶点 1,定义 name, weight 属性
* a0 S( ]/ ?5 A( X/ MG1.add_node(2, date='May-16') # 添加顶点 2,定义 time 属性6 A) B: i# e' \$ a i( o! E
G1.add_nodes_from([3, 0, 6], dist=1) # 添加多个顶点,并定义属性
/ V8 g& c1 a1 P1 r! C; qG1.add_nodes_from(range(10, 15)) # 向图 G1 添加顶点 10~14
2 ^1 z, A8 [8 x
% M0 u L, m( R3 E% P! t6 l p# 查看顶点和顶点属性
3 |0 \8 H7 V! d7 Kprint(G1.nodes()) # 查看顶点列表
* \' J" {# m, ~# [1, 2, 3, 0, 6, 10, 11, 12, 13, 14]( I( D& j8 v7 y' {$ ^6 q( D
print(G1._node) # 查看顶点属性
2 s1 c+ {! e) T; F @9 T# {1: {'name': 'n1', 'weight': 1.0}, 2: {'date': 'May-16'}, 3: {'dist': 1}, 0: {'dist': 1}, 6: {'dist': 1}, 10: {}, 11: {}, 12: {}, 13: {}, 14: {}}
# y1 k+ k! ^$ k/ A3 T
; f7 N( |, j" { Q/ \6 w3 i# 从图中删除顶点( ^4 t3 j& w- ?: {& k
G1.remove_node(1) # 删除顶点 D+ U9 B0 \# t. E
G1.remove_nodes_from([1, 11, 13, 14]) # 通过顶点标签的 list 删除多个顶点' Y5 w( I! ^( p* e# y" G
print(G1.nodes()) # 查看顶点
% p9 Y Z/ ^% n# H4 R; @* S j. q% O# [2, 3, 0, 6, 10, 12] # 顶点列表* O2 B( r4 D& C7 P% F6 m) S2 z4 Q
2.2.3 边的添加、删除和查看边是两个顶点之间的连接,在 NetworkX 中 边是由对应顶点的名字的元组组成 e=(node1,node2)。边可以设置权重、关系等属性。 边的常用操作:添加边,删除边,定义边的属性,查看边和边的属性。向图中添加边时,如果边的顶点是图中不存在的,则自动向图中添加该顶点。 Graph.add_edge(u_of_edge, v_of_edge, **attr)* I Z. U2 O4 z* ]; X
Graph.add_edges_from(ebunch_to_add, **attr)3 W2 ^2 X u$ b( e. l( S: H1 Y0 D% r
Graph.add_weighted_edges_from(ebunch_to_add, weight=‘weight’, **attr)
( ?1 n8 W8 z* t3 F/ A
8 Q4 m5 s2 O- q! i/ Q" p# 边(edge)的操作: G$ E! e: V9 [0 [: l* a
# 向图中添加边
: Y/ ?- E. Z: HG1.add_edge(1,5) # 向 G1 添加边,并自动添加图中没有的顶点
* v$ R' z7 p7 F7 V! c9 \G1.add_edge(0,10, weight=2.7) # 向 G1 添加边,并设置边的属性
* l3 p, N7 L# n b7 A# ~+ ^& rG1.add_edges_from([(1,2,{'weight':0}), (2,3,{'color':'blue'})]) # 向图中添加边,并设置属性4 m* @0 G! L' n. j F
G1.add_edges_from([(3,6),(1,2),(6,7),(5,10),(0,1)]) # 向图中添加多条边
7 O o' I& V& A1 Z" }* f( DG1.add_weighted_edges_from([(1,2,3.6),[6,12,0.5]]) # 向图中添加多条赋权边: (node1,node2,weight)
7 M$ y; _8 V' m2 ?+ @0 nprint(G1.nodes()) # 查看顶点
' d& A" p0 Z- C9 C# [2, 3, 0, 6, 10, 12, 1, 5, 7] # 自动添加了图中没有的顶点$ ? o9 Q' `4 Z5 z i! K$ l
/ k' J \# z/ N2 n, f9 @
# 从图中删除边6 d, w0 G% v; k) }
G1.remove_edge(0,1) # 从图中删除边 0-1# h% N) V& a- s4 i- [3 w; A
G1.remove_edges_from([(2,3),(1,5),(6,7)]) # 从图中删除多条边
7 Y: S4 P9 q+ x X# H- N- _3 o) A5 Y$ Z/ p2 @" e
# 查看 边和边的属性6 D5 z) w/ b2 X; G5 _$ o
print(G1.edges) # 查看所有的边
, J& l$ B7 O$ y, W* M$ Z& D0 f[(2, 1), (3, 6), (0, 10), (6, 12), (10, 5)]
1 j/ d) Z( ]' O" ]. P" f( Vprint(G1.get_edge_data(1,2)) # 查看指定边的属性3 E; H3 u% ~5 b4 N$ p8 I: ?
# {'weight': 3.6}
1 v5 R' h! `7 }$ m: |1 q& i$ Jprint(G1[1][2]) # 查看指定边的属性
+ A k* Z( j- G3 e# {'weight': 3.6}
3 Z- r( A( l8 K% y cprint(G1.edges(data=True)) # 查看所有边的属性
1 w7 B7 T! M0 y/ I* p; c# [(2, 1, {'weight': 3.6}), (3, 6, {}), (0, 10, {'weight': 2.7}), (6, 12, {'weight': 0.5}), (10, 5, {})]+ H! t- f6 v$ u2 _" B" r. |$ E+ v
, ]: R8 e5 ]) }) f% x' A+ k2 w
2.2.4 查看图、顶点和边的信息
1 H0 V4 g9 s- }- z& f" @- F, I2 Q) I! K9 l9 ^
# 查看图、顶点和边的信息" k# p; w, o3 F4 \( M, j/ j0 ^) ^4 x
print(G1.nodes) # 返回所有的顶点 [node1,...] V2 \$ g9 [4 L/ v- h* d; T
# [2, 3, 0, 6, 10, 12, 1, 5, 7]
% {: A8 L, z/ L' b! ^, iprint(G1.edges) # 返回所有的边 [(node1,node2),...]
, ^, H% r [. V6 B; d$ M& j% P# [(2, 1), (3, 6), (0, 10), (6, 12), (10, 5)]& e( l X! R6 c
print(G1.degree) # 返回各顶点的度 [(node1,degree1),...], \, {0 C# Q6 O
# [(2, 1), (3, 1), (0, 1), (6, 2), (10, 2), (12, 1), (1, 1), (5, 1), (7, 0)]
0 n1 c* |0 [4 G' Q0 zprint(G1.number_of_nodes()) # 返回顶点的数量
8 u3 k6 B+ w0 D# 9 a7 R+ d/ M3 w! F- y* u: w/ n
print(G1.number_of_edges()) # 返回边的数量
7 b X U; j% u$ P" t) a' i# 5. w" y5 r' Z9 D9 Z
print(G1[10]) # 返回与指定顶点相邻的所有顶点的属性
C7 z0 O' T Q5 H+ k0 l# {0: {'weight': 2.7}, 5: {}}
/ d, [+ d1 R8 ]2 y$ C8 xprint(G1.adj[10]) # 返回与指定顶点相邻的所有顶点的属性
3 Q6 D/ }7 s4 y# {0: {'weight': 2.7}, 5: {}}
( n* M! J% [! |# z. J5 \print(G1[1][2]) # 返回指定边的属性/ V! @$ d5 s" N: j9 M2 K- }5 h
# {'weight': 3.6}
/ x1 v& B3 ~8 R. Sprint(G1.adj[1][2]) # 返回指定边的属性3 e3 J) t5 ?0 O: m
# {'weight': 3.6}- K; g2 y* B! c. a# R
print(G1.degree(10)) # 返回指定顶点的度1 ~" j# W2 W0 T" `* `# ]
# 2
8 r% J1 Y5 I4 p4 v
& H0 u* m0 q. A, a3 S1 L7 k6 Dprint('nx.info:',nx.info(G1)) # 返回图的基本信息
' _3 {' N7 Y( m# k. s' Q! c$ |print('nx.degree:',nx.degree(G1)) # 返回图中各顶点的度
* ]/ r: d, A2 p' G' x3 p- `. Mprint('nx.density:',nx.degree_histogram(G1)) # 返回图中度的分布7 O4 O+ s) J( e% p/ |: ^8 A
print('nx.pagerank:',nx.pagerank(G1)) # 返回图中各顶点的频率分布
( q. J/ i6 p' [' O, C( h9 S
$ X! k" a7 v- Z% P. r1 H& K" D! ?7 X/ T( x8 j" A# `
l2 a4 p3 b& q7 Z2 ?
8 C5 @- ^7 s* W4 G& v
5 Q. M$ k Z. h6 |" m: ]
2.3 图的属性和方法图的方法
+ e5 D) _" ^3 M, ^6 H! e: n. z
2 F0 K0 e8 T' ^% [5 {1 o& \. S方法 说明3 R6 b0 o0 p+ E( |: v" w
G.has_node(n) 当图 G 中包括顶点 n 时返回 True
7 q. ]9 l/ K. x# w4 \" v* R) SG.has_edge(u, v) 当图 G 中包括边 (u,v) 时返回 True
, C7 a2 e l2 y$ wG.number_of_nodes() 返回 图 G 中的顶点的数量
. N" L% q8 o% `/ j4 v; gG.number_of_edges() 返回 图 G 中的边的数量; \% H. L6 u$ z
G.number_of_selfloops() 返回 图 G 中的自循环边的数量% V& m- I; b) v8 F* W. j
G.degree([nbunch, weight]) 返回 图 G 中的全部顶点或指定顶点的度
# G+ n! |* M$ @G.selfloop_edges([data, default]) 返回 图 G 中的全部的自循环边
# J+ R8 H* h' u6 i6 s. cG.subgraph([nodes]) 从图 G1中抽取顶点[nodes]及对应边构成的子图
% n0 q3 Z% A+ g* I$ ~union(G1,G2) 合并图 G1、G2' P5 P u) }3 T3 u
nx.info(G) 返回图的基本信息0 D6 h. h D7 f( c7 P5 s/ i% c
nx.degree(G) 返回图中各顶点的度
( n6 I, K2 t l; n) Jnx.degree_histogram(G) 返回图中度的分布
5 }* h; N/ t+ [2 jnx.pagerank(G) 返回图中各顶点的频率分布
4 y" }6 v- T/ \/ ^- @5 Enx.add_star(G,[nodes],**attr) 向图 G 添加星形网络
( Y1 N: P) }6 m" lnx.add_path(G,[nodes],**attr) 向图 G 添加一条路径
4 w2 d8 n6 l+ g/ J9 {/ rnx.add_cycle(G,[nodes],**attr) 向图 G 添加闭合路径: L: b+ S3 c: N. |. [
g2 M, p( V- K. a! Y7 i1 [) {9 D$ M0 ]
例程:
) C4 W$ J$ R& d( f$ BG1.clear() # 清空图G1; O o2 o; y) Y
nx.add_star(G1, [1, 2, 3, 4, 5], weight=1) # 添加星形网络:以第一个顶点为中心
4 E! p* k7 b9 y# [(1, 2), (1, 3), (1, 4), (1, 5)]+ S5 s) p% J( r# p
nx.add_path(G1, [5, 6, 8, 9, 10], weight=2) # 添加路径:顺序连接 n个节点的 n-1条边
6 J# j; Z* S4 X( b2 x# [(5, 6), (6, 8), (8, 9), (9, 10)]# C4 Z$ I1 W& E. r
nx.add_cycle(G1, [7, 8, 9, 10, 12], weight=3) # 添加闭合回路:循环连接 n个节点的 n 条边
; F5 I. O, M9 P2 z/ ~# [(7, 8), (7, 12), (8, 9), (9, 10), (10, 12)]' G$ f4 J* J, ]3 o! h
print(G1.nodes) # 返回所有的顶点 [node1,...]" I4 H& b1 W' b% h5 B1 U. D9 m& {4 I
nx.draw_networkx(G1)
5 [% {' I, q, t% r- q, Dplt.show()
n* H: R3 `0 l: d& J
9 T, J: h- F- W* q9 I, RG2 = G1.subgraph([1, 2, 3, 8, 9, 10]) w+ i, i2 q3 [
G3 = G1.subgraph([4, 5, 6, 7])0 }4 z, D( W4 e# O- w
G = nx.union(G2, G3)
9 J4 F) m E' `% v. z6 l. }7 Wprint(G.nodes) # 返回所有的顶点 [node1,...]
, ?2 P# m" ]5 l( k* S |9 N' u9 p# [1, 2, 3, 8, 9, 10, 4, 5, 6, 7]! q3 h; m: A. \5 [3 `
( \1 t6 W) v% _3 t9 Z# F
! E) L# ^7 {: q/ L- t3、图的绘制与分析3.1 图的绘制% Z) ?6 F" m( t, ^9 w! ]& d
可视化是图论和网络问题中很重要的内容。NetworkX 在 Matplotlib、Graphviz 等图形工具包的基础上,提供了丰富的绘图功能。
6 h: K# Y; e( A1 y
1 E$ [1 K) p) [% N! t; r h1 p8 f本系列拟对图和网络的可视化作一个专题,在此只简单介绍基于 Matplotlib 的基本绘图函数。基本绘图函数使用字典提供的位置将节点放置在散点图上,或者使用布局函数计算位置。
8 I7 t1 o+ @+ N5 b; e# r
& ~) O" n7 p. S方法 说明) T) L! t( @( R+ ]0 j6 P3 ?
draw(G[,pos,ax]) 基于 Matplotlib 绘制 图 G2 A, _5 Q8 N( V* Z) S7 I
draw_networkx(G[, pos, arrows, with_labels]) 基于 Matplotlib 绘制 图 G
# P% i5 {$ ` L1 Mdraw_networkx_nodes(G, pos[, nodelist, . . . ]) 绘制图 G 的顶点
" z; i, a2 I+ X6 W7 ~3 y/ mdraw_networkx_edges(G, pos[, edgelist, . . . ]) 绘制图 G 的边
9 R) H/ C h' P1 J3 Wdraw_networkx_labels(G, pos[, labels, . . . ]) 绘制顶点的标签
/ w# Y/ Q# M0 }9 R1 Y4 S" ?3 Adraw_networkx_edge_labels(G, pos[, . . . ]) 绘制边的标签
; o; f4 X% N$ A: j1 L6 Z6 D# @/ L h0 c% ~% I* g
! _& Q7 V4 c) B5 g4 N0 j其中,nx.draw() 和 nx.draw_networkx() 是最基本的绘图函数,并可以通过自定义函数属性或其它绘图函数设置不同的绘图要求。
! P7 ]# @+ A6 T, I, V; odraw(G, pos=None, ax=None, **kwds) draw_networkx(G, pos=None, arrows=True, with_labels=True, **kwds) / `! ]7 P6 I, e( _, e
常用的属性定义如下:
4 D) w- G4 }' a+ s7 j, l0 c. z S! r: ?1 o7 V4 ~4 x8 b# F! g
‘node_size’:指定节点的尺寸大小,默认300
' T4 a5 ^, q6 d0 s‘node_color’:指定节点的颜色,默认红色
2 f/ o/ |+ H7 p% |0 C‘node_shape’:节点的形状,默认圆形4 y( Y2 _9 \: |! h4 V/ r
'‘alpha’:透明度,默认1.0,不透明
" x, q" M! q1 ]: I, z, W‘width’:边的宽度,默认1.0, W& t8 \0 m4 F6 d
‘edge_color’:边的颜色,默认黑色# E r1 ]3 N% j8 {" N8 Z
‘style’:边的样式,可选 ‘solid’、‘dashed’、‘dotted’、‘dashdot’
2 Q( p) G0 i8 F4 ?0 P$ E. \‘with_labels’:节点是否带标签,默认True' ^6 z" e* z& L# E$ h& F
‘font_size’:节点标签字体大小,默认12
0 [; Z8 h% ?/ A1 w6 g‘font_color’:节点标签字体颜色,默认黑色2 x; [" u# @/ H
& ~# k# a0 g+ Y3 G
- S8 j) e+ U" U* u
3.2 图的分析NetwotkX 提供了图论函数对图的结构进行分析* V7 E" y& s* q" ^
子图
1 }2 E, u# Z5 ~' C' t' ?" Y- 子图是指顶点和边都分别是图 G 的顶点的子集和边的子集的图。
- subgraph()方法,按顶点从图 G 中抽出子图。例程如前。: \" l3 M! e2 J0 U8 ?
连通子图1 u2 G: i2 r. \- a7 V z
# O. D2 G+ ~# O- K5 h$ }) F* X! u- 如果图 G 中的任意两点间相互连通,则 G 是连通图。
- [color=rgba(0, 0, 0, 0.749019607843137)]connected_components()方法,返回连通子图的集合。8 F7 k4 F3 X' g" Z
[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}]1 q4 u( O% e2 [/ S3 O
- b, |' }8 J; ]# k1 B' @0 q9 a
; G6 Q# h# p0 |: _0 Y5 M3 s
+ a3 F0 ^/ A' C1 A
8 P, p5 Y) Q1 a: x1 x/ Y |