小沐:"戈戈,你说令狐冲学独孤九剑,师父风清扬是怎么传授的?"戈戈:"风清扬说,剑招要烂熟于心,才能无招胜有招。你问这个干啥?"小沐:"我在想,如果当年风清扬有一张完整的招式流程图,令狐冲学起来是不是更快?可我画流程图太慢了,改一次就得重画!"
戈戈:"哈哈,你这是想当风清扬,却卡在了绘图上!用 Python 几行代码就能搞定,改起来也不用重来——这才叫'以不变应万变'!"小沐:"这招比独孤九剑还厉害,快教我!"

江湖上有句话:"练剑不画图,方向全凭悟。"可惜大多数人既没有令狐冲的天赋,也没有风清扬的耐心。
好在 Python 给了我们另一条路——用代码"写"出流程图,修改只需动几行,布局自动调整,堪称程序员版的"独孤九剑"。本文将介绍三大门派:Graphviz、Schemdraw 和 Matplotlib,从快速出图到精细定制,各有绝学,总有一招适合你。
一、环境准备
学武之前先备好兵器。以下是本文三大门派所需的全部依赖,一次装好,后顾无忧。
pip install graphvizpip install diagramspip install matplotlibpip install schemdraw
二、使用 Graphviz 绘制基础流程图
2.1 功能介绍
Graphviz 是流程图界的"少林派"——资历最老、功力最深。它通过 DOT 语言描述图结构,Python 的 graphviz 库则充当翻译官,让我们用纯 Python 指挥节点和箭头各就各位。你只需说"这里放个菱形判断框",它就乖乖照办,布局还自动对齐。
2.2 安装依赖
2.3 示例代码
以下示例绘制一张"武林大会比武场次"流程图,从报名资格审查,到分组对决,再到最终夺魁,一图尽览赛制全貌:
import graphviz# 召唤有向图,武林大会正式开幕dot = graphviz.Digraph( comment='武林大会比武流程', format='png')# 各门派选手统一着装,英雄帖样式先定好dot.attr('node', shape='box', style='rounded,filled', fillcolor='#AED6F1', fontname='Microsoft YaHei')dot.attr('edge', fontname='仿宋')dot.attr('graph', fontname='Microsoft YaHei')# 武林大会流程节点,一一点名入场dot.node('start', '武林大会开幕', shape='ellipse', fillcolor='#A9DFBF')dot.node('reg', '各派递交英雄帖\n(报名)')dot.node('qualify', '资格审核\n(是否达标?)', shape='diamond', fillcolor='#F9E79F')dot.node('reject', '取消资格\n逐出赛场', fillcolor='#F1948A') # 没资格就别来丢人dot.node('group', '抽签分组\n(初赛对阵表)')dot.node('prelim', '初赛对决\n(三场两胜)', fillcolor='#D7BDE2')dot.node('advance', '晋级?', shape='diamond', fillcolor='#F9E79F')dot.node('elim', '败者淘汰\n饮酒压惊', fillcolor='#F1948A') # 输了也别伤心,酒管够dot.node('semi', '半决赛\n(华山论剑)', fillcolor='#D7BDE2')dot.node('final', '决赛\n(天下第一之争)', fillcolor='#D7BDE2')dot.node('champ', '武林盟主诞生!', shape='ellipse', fillcolor='#A9DFBF') # 众望所归# 用箭头串起整场江湖盛宴dot.edge('start', 'reg')dot.edge('reg', 'qualify')dot.edge('qualify', 'reject', label='不达标') # 滥竽充数者,请退dot.edge('qualify', 'group', label='达标入围')dot.edge('group', 'prelim')dot.edge('prelim', 'advance')dot.edge('advance', 'elim', label='落败') # 英雄迟暮,下次再来dot.edge('advance', 'semi', label='晋级')dot.edge('semi', 'final')dot.edge('final', 'champ')# 封笔收招,图谱传世dot.render('wulin_tournament', view=False, cleanup=True)print("武林大会赛制流程图已生成,江湖儿女,各凭本事!")
代码运行结果如下:
image
三、使用 Schemdraw 绘制精美流程图
3.1 功能介绍
如果说 Graphviz 是"少林派"——威猛扎实但规矩多,那 schemdraw 更像"武当派"——优雅从容,一招一式娓娓道来。它支持像写武功心法一样,逐步叠加元素,方向、间距皆可指定,每一步都所见即所得,特别适合绘制教程类、演示类流程图。
3.2 安装依赖
pip install schemdraw matplotlib
3.3 示例代码
用一段"少年闯江湖拜师学艺"的函数逻辑,用 Schemdraw 演示一套行云流水的流程:
import matplotlibimport schemdrawimport schemdraw.elements as elmfrom schemdraw.flow import flowmatplotlib.rcParams['font.sans-serif'] = ['Microsoft YaHei', 'SimHei', 'SimSun', 'Arial Unicode MS']matplotlib.rcParams['axes.unicode_minus'] = Falseschemdraw.config(font='Microsoft YaHei', unit=1.9)EDGE, BG = '#243447', '#f5f7fb'START, PROCESS, DECISION = '#ff7a59', '#6bb8f2', '#ffc857'SUCCESS, ALT, NEUTRAL = '#22b573', '#8f63c9', '#98a1a8'STYLE = dict(edge_color=EDGE, linewidth=1.8)def node(kind, text, xy, fill, w, h): cls = {'p': flow.Process, 'd': flow.Decision, 't': flow.Terminal}[kind] return cls(w=w, h=h, fill=fill, **STYLE).at(xy).label(text)def link(d, a, b, color=EDGE, label=None, side='right', ofst=(0, 0)): e = elm.Line(arrow='->', color=color, lw=1.8).at(a).to(b) d += e.label(label, loc=side, ofst=ofst, color=color, fontsize=13) if label else ewith schemdraw.Drawing(fontsize=10, bgcolor=BG, fontcolor='#18212b', aspect='equal') as d: mx, bx, sx, y0, g = 0, 7.6, 15.2, 0, -3.4 start = node('t', '少年离家\n闯荡江湖', (mx, y0), START, 3.0, 1.2) meet = node('p', '偶遇神秘老人', (mx, y0 + g), PROCESS, 2.8, 1.4) accept = node('d', '老人愿意\n收徒?', (mx, y0 + g * 2), DECISION, 3.0, 1.8) train = node('p', '深山苦练\n三年内功', (mx, y0 + g * 3), PROCESS, 2.8, 1.4) secret = node('p', '获传武林秘籍', (mx, y0 + g * 4), PROCESS, 2.8, 1.4) end_win = node('t', '称霸武林\n封神结局', (mx, y0 + g * 5), START, 3.0, 1.2) wander = node('p', '继续流浪\n误打误撞', (bx, y0 + g * 2), PROCESS, 2.8, 1.4) lucky = node('d', '捡到\n神功秘籍?', (bx, y0 + g * 3), DECISION, 3.0, 1.8) self_learn = node('p', '对着秘籍\n野蛮自学', (bx, y0 + g * 4), PROCESS, 2.8, 1.4) end_ok = node('t', '野路子出奇迹\n也算一代宗师', (bx, y0 + g * 5), '#39c27a', 3.0, 1.2) go_home = node('t', '打道回府\n另谋出路', (sx, y0 + g * 3), NEUTRAL, 3.0, 1.2) for n in [start, meet, accept, train, secret, end_win, wander, lucky, self_learn, end_ok, go_home]: d += n link(d, start.S, meet.N); link(d, meet.S, accept.N); link(d, train.S, secret.N); link(d, secret.S, end_win.N) link(d, accept.S, train.N, SUCCESS, '愿意', 'left', (-0.55, 0)); link(d, wander.S, lucky.N); link(d, self_learn.S, end_ok.N) link(d, lucky.S, self_learn.N, SUCCESS, '捡到了', 'left', (-0.6, 0)) link(d, accept.E, wander.W, ALT, '不收', 'top', (0, 0.5)) link(d, lucky.E, go_home.W, ALT, '没捡到', 'bottom', (0, -0.45)) d.save('jianghu_flow.png', dpi=160) print('流程图已生成,代码已整理为紧凑版。')
代码运行结果如下:
image
四、使用 Matplotlib 手动绘制自定义流程图
4.1 功能介绍
前两派各有门规,若遇上"特立独行"的需求——比如公司专属品牌色、非标准节点形状,或者要把流程图塞进数据报告里和折线图并排——matplotlib 才是真正的"独行侠"。它没有固定套路,画布、坐标、颜色、箭头,全凭你一笔一划自由发挥,自由度拉满,代价是多写几行代码。
4.2 安装依赖
4.3 示例代码
换个有趣的题材——绘制一张"郭靖学功夫升级路线"流程图。
import matplotlib.pyplot as pltfrom matplotlib.patches import FancyBboxPatchplt.rcParams['font.sans-serif'] = ['Microsoft YaHei', 'SimHei', 'SimSun', 'Arial Unicode MS', 'DejaVu Sans']plt.rcParams['axes.unicode_minus'] = FalseFIG_BG = '#f3f5f8'TEXT = '#1f2937'EDGE = '#7a8492'ARROW = '#545f6d'MERGE = '#8c48b7'def draw_box(ax, x, y, w, h, text, color, fs=10.5): patch = FancyBboxPatch( (x - w / 2, y - h / 2), w, h, boxstyle='round,pad=0.12,rounding_size=0.12', facecolor=color, edgecolor=EDGE, linewidth=1.4 ) ax.add_patch(patch) ax.text(x, y, text, ha='center', va='center', fontsize=fs, fontweight='bold', color=TEXT) return {'x': x, 'y': y, 'w': w, 'h': h}def anchor(node, side, frac=0.5): if side == 'top': return node['x'] + (frac - 0.5) * node['w'], node['y'] + node['h'] / 2 if side == 'bottom': return node['x'] + (frac - 0.5) * node['w'], node['y'] - node['h'] / 2 if side == 'left': return node['x'] - node['w'] / 2, node['y'] + (frac - 0.5) * node['h'] return node['x'] + node['w'] / 2, node['y'] + (frac - 0.5) * node['h']def link(ax, src, dst, src_side='bottom', dst_side='top', color=ARROW, rad=0.0): ax.annotate( '', xy=anchor(dst, dst_side), xytext=anchor(src, src_side), arrowprops=dict( arrowstyle='-|>', color=color, lw=1.8, mutation_scale=12, shrinkA=6, shrinkB=8, connectionstyle=f'arc3,rad={rad}', ), )def section_label(ax, x, y, text): ax.text( x, y, text, ha='center', va='center', fontsize=10, color='#8a5d00', bbox=dict(facecolor='#fff7db', edgecolor='#d7a01e', boxstyle='round,pad=0.3', linewidth=1.2) )fig, ax = plt.subplots(figsize=(8.4, 11), facecolor=FIG_BG)ax.set_xlim(0, 8)ax.set_ylim(0, 11)ax.set_facecolor(FIG_BG)ax.axis('off')section_label(ax, 2.0, 10.45, '师承路线')section_label(ax, 6.0, 10.45, '内功路线')L1 = draw_box(ax, 2.0, 9.8, 3.0, 0.72, '江南七怪启蒙\n(十年磨一剑)', '#b8d8c7')L2 = draw_box(ax, 2.0, 8.55, 3.0, 0.72, '洪七公传授\n降龙十八掌', '#99ccb0')L3 = draw_box(ax, 2.0, 7.3, 3.0, 0.72, '马钰点拨\n全真内功心法', '#99ccb0')L4 = draw_box(ax, 2.0, 6.05, 3.0, 0.72, '周伯通传授\n左右互搏术', '#eadb97')R1 = draw_box(ax, 6.0, 9.8, 2.85, 0.72, '九阴真经\n(强行背下来的)', '#b4c8d8')R2 = draw_box(ax, 6.0, 8.55, 2.85, 0.72, '降龙十八掌\n内劲贯通', '#97bdd7')R3 = draw_box(ax, 6.0, 7.3, 2.85, 0.72, '左右互搏\n双掌齐发', '#97bdd7')R4 = draw_box(ax, 6.0, 6.05, 2.85, 0.72, '内外兼修\n天下无敌', '#7daed0')C1 = draw_box(ax, 4.0, 4.75, 3.45, 0.74, '华山论剑\n(正式踏入顶尖高手圈)', '#c8bcd2')C2 = draw_box(ax, 4.0, 3.45, 3.45, 0.74, '镇守襄阳\n以武学护天下苍生', '#bea9cc')C3 = draw_box(ax, 4.0, 2.15, 3.45, 0.74, '郭大侠威名远播\n天下第一人', '#99ccb0')C4 = draw_box(ax, 4.0, 0.95, 3.05, 0.64, '大结局', '#99ccb0')for a, b in [(L1, L2), (L2, L3), (L3, L4), (R1, R2), (R2, R3), (R3, R4), (C1, C2), (C2, C3), (C3, C4)]: link(ax, a, b)ax.annotate('', xy=anchor(C1, 'top', 0.28), xytext=anchor(L4, 'bottom', 0.50), arrowprops=dict(arrowstyle='-|>', color=MERGE, lw=2.0, mutation_scale=12, shrinkA=6, shrinkB=8))ax.annotate('', xy=anchor(C1, 'top', 0.72), xytext=anchor(R4, 'bottom', 0.50), arrowprops=dict(arrowstyle='-|>', color=MERGE, lw=2.0, mutation_scale=12, shrinkA=6, shrinkB=8))plt.title('郭靖学功夫升级路线图', fontsize=13, fontweight='bold', pad=1, color='#111827')plt.tight_layout(pad=1.0)plt.savefig('guojing_kungfu.png', dpi=170, bbox_inches='tight', facecolor=FIG_BG)print('流程图已优化:连线端点已贴边并统一美化。')plt.show()
代码运行结果如下:
image
五、三种方案对比与选型建议
三大门派各怀绝技,没有高下之分,只有适不适合。就像《笑傲江湖》里,令狐冲凭独孤九剑纵横江湖,田伯光靠快刀也能称霸一方——用对了兵器,才是真本事。
| | | |
|---|
| Graphviz | | | |
| Schemdraw | | | |
| Matplotlib | | | |
选型建议:
- 需要精细样式控制或与数据可视化结合 → 选 Matplotlib
小沐:"戈戈,我现在感觉自己就是令狐冲,掌握了独孤九剑——Graphviz、Schemdraw、Matplotlib,三招在手,流程图天下无敌!"
戈戈:"哈哈,剑法已传,记住风清扬的话:熟能生巧,以代码驭图,方能无招胜有招。"小沐:"明白了!这就去给项目文档也画一张'武功秘籍'流程图!"

觉得有用请点赞 + 在看,转发给需要画流程图的朋友~ 评论区聊聊你平时用什么工具画流程图?
结语
如果您觉得这些文字有一点点用处,请给作者点个赞或关个注;╮( ̄▽ ̄)╭如果您有技术问题探讨,评论处留言。//(ㄒoㄒ)//谢谢各位童鞋们啦( ´ ▽ ` )ノ ( ´ ▽ `` )っ!更多精彩文章详见:CSDN博客:爱看书的小沐微信公众号:杨小羊爱阅读