惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

T
Threat Research - Cisco Blogs
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
V
Vulnerabilities – Threatpost
GbyAI
GbyAI
P
Proofpoint News Feed
L
LINUX DO - 热门话题
P
Palo Alto Networks Blog
A
About on SuperTechFans
T
Tenable Blog
M
MIT News - Artificial intelligence
IT之家
IT之家
I
Intezer
D
DataBreaches.Net
爱范儿
爱范儿
T
Threatpost
C
CERT Recently Published Vulnerability Notes
云风的 BLOG
云风的 BLOG
博客园 - 三生石上(FineUI控件)
WordPress大学
WordPress大学
K
Kaspersky official blog
大猫的无限游戏
大猫的无限游戏
A
Arctic Wolf
Y
Y Combinator Blog
Cyberwarzone
Cyberwarzone
酷 壳 – CoolShell
酷 壳 – CoolShell
D
Darknet – Hacking Tools, Hacker News & Cyber Security
H
Help Net Security
Microsoft Security Blog
Microsoft Security Blog
Spread Privacy
Spread Privacy
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
AWS News Blog
AWS News Blog
博客园 - 聂微东
C
Check Point Blog
S
Securelist
有赞技术团队
有赞技术团队
雷峰网
雷峰网
aimingoo的专栏
aimingoo的专栏
Last Week in AI
Last Week in AI
Stack Overflow Blog
Stack Overflow Blog
MongoDB | Blog
MongoDB | Blog
D
Docker
G
GRAHAM CLULEY
T
The Exploit Database - CXSecurity.com
C
Cybersecurity and Infrastructure Security Agency CISA
T
Tailwind CSS Blog
L
Lohrmann on Cybersecurity
G
Google Developers Blog
C
Cyber Attacks, Cyber Crime and Cyber Security
L
LangChain Blog

博客园_首页

Plist 二进制格式 第30篇文章:一个大三计科生的自白 Manim如何在数学公式中完美显示中文? Docker 部署 RocketMQ 5 并发编程核心概念辨析 C#事务处理最佳实践:别再让“主表存了、明细丢了”的破事发生 CLI 是什么?为什么大厂突然集体卷命令行? 【从0到1构建一个ClaudeAgent】协作-自主Agent # linux红帽教程-手把手教学 UIImageView 设置图片不生效的原因排查 .NET生态下Native AOT兼容的Cron任务调度框架 Python 潮流周刊#147:Python 和 Ruby 的 JIT 故事 - 豌豆花下猫 可持久化线段树/主席树 学习笔记 如何实现 Claude Code 和 Codex 等 Agent CLI 的自动重试 - Newbe36524 WebSocket 连接池生产级实现:实时行情高可用与负载均衡 - Walter先生 关于代码注释的思考 MicroPython对接大模型:uopenai + 火山方舟实现文字聊天和图片理解 从词向量到大模型:NLP 技术演进浅记 LangChain使用deep agent并且加载SKILL 零成本打造专业域名邮箱:Cloudflare + Gmail 终极配置保姆级全攻略 【从0到1构建一个ClaudeAgent】协作-团队协议 - 程序员Seven 最小二乘问题详解20:无先验约束下的增量式SFM自由网平差 痞子衡嵌入式:大话双核i.MXRT1180之XIP应用里实现可靠Flash IAP的方法 AI Chat 封装, SemanticKerne.AiProvider.Unified 已发布 Windows下右键编辑js文件无法打开记事本——在注册表中使用环境变量 在后台服务中使用 Scoped 服务,为什么总是报错? H200 安装驱动并使用sglang启动模型 wireshark 抓包Trap上报告警内容 我用 AI 辅助开发了一系列小工具(2):图片压缩工具 [A Primer On MC and CC] 2.1 Memory Consistency 1 - 指令重排序和 SC 模型 Oracle数据库SCN推进技术详解与实践指南 玩转控件:封装个带图片的Label控件 Claude Code 4.7 真正该升级的不是模型,而是你的工作流 我用AI写了一个颜值拉满的桌面媒体播放器,全程没动一行代码,这就是AI编程新范式 5. WorkBuddy: 小龙虾的灵魂三件套,让你的小龙虾不只是工具 SQLite 分片方案实战:三种分片策略的深度对比 告别简陋 UI!一款基于 Fluent Design 和基于 WinUI 的开源免费、现代化的 Avalonia UI 控件库 关于二进制排列组合枚举的总结 AI开发-python-LangGraph框架(3-27-LangGraph从零实现大模型智能决策工作流) ElasticSearch主分片和副本分片概念详解 【002】HTTPS 粗解:证书、TLS 握手与对后端配置的影响 Hermes Agent 一周暴涨五万 Star,但我劝你别急着追 一个面向产品化的 Electron + Vue 3 桌面应用脚手架 明明连接的是Redis的DB0,为什么能查到DB3的数据? 【从0到1构建一个ClaudeAgent】协作-Agent团队 熟悉电子元器件之后,电子小白下一步该怎么走? MAF快速入门(23)通过C#类定义Skills .NET 高级开发 | 手写一个对象映射框架 FastAPI数据库ORM怎么选?我肝了三个Demo后,终于不再纠结了 mysqldump 参数拾遗:在遗忘与铭记之间 C# .NET 周刊|2026年3月5期 - InCerry 一文学习入门 ThingsBoard 开源物联网平台 - daidaidaiyu 如何为GIT设置全局勾子,为每次提交追加信息 - SKILL·NULL Number.isFinite和isFinite与isNaN()和isNaN的区别 - 南风晚来晚相识 PortSwigger SQL注入LAB2 - C2H5OH 推荐一个测试人必备的Skills,从功能到性能全搞定(附详细实操和安装下载方式) - 狂师 筑基期:掌握Odoo基础核心知识点02(Odoo XML 开发方式详解) - okkk!!! GLM模型这么火,咱们用vllm也咧一个呗! - 码甲哥不卷 深入理解 AbortController:从底层原理到跨语言设计哲学 - 革新 字符串学习笔记 - liduoduo2021 多租户系统框架的基础模块设计和分析设计 - 伍华聪 Apache SeaTunnel Zeta 为什么能做到“又快又稳”? - ApacheSeaTunnel AI开发-python-LangGraph框架(3-26-LangGraph基本概念及第一个简单样例) - 万笑佛 Vue 3 组件通信,别只会用 Props 和 Emits 了,这几个狠活儿你得看看 - 一名程序媛呀 ElasticSearch7.X版本配置密码 - huangSir-devops 用Manim实现动态交点计算--从一个动点问题说起 - wang_yb 团结引擎+Addressable+Instant Game打包抖音小游戏 - 威少小二orz function call 实战:让 LLM 自动判断 pod 异常、调用日志工具并完成故障分析 - it排球君 4.15 bubseek —— 让 Agent 的足迹,变成团队的洞察 - 老纪的技术唠嗑局 通过 C# 读取并导出 PDF 书签 - LAYONTHEGROUND 如何用 GitHub Actions 实现 Steam 自动化发布 - Newbe36524 【从0到1构建一个ClaudeAgent】并发-后台任务 - 程序员Seven .NET 高级开发 | 定制 ASP.NET Core 框架 - 痴者工良 电子小白:什么是运算放大器(运放) - Tlink zero2Agent:面向大厂面试的 Agent 工程教程,从概念到生产的完整学习路线 - 孤飞 堆上的ORW HC32F460 USB CDC通信异常:非对齐访问异常排查 20260413-Hyperbridge 攻击事件:发生在默克尔山上的验证绕过 - ACai_sec 那些喊着AI 要淘汰你的人,正在靠你的焦虑赚大钱! 深度学习进阶(八)Swin Transformer - 哥布林学者 最小二乘问题详解19:带先验约束的增量式SFM优化与实现 - charlee44 SnapTranslate 3.0 正式发布:全局划词翻译 + 完整英语学习闭环,一站式搞定查词、记词、复习 - TTGF .NET 官方团队发布的 .NET Agent Skills,告别 AI 编程幻觉! - 追逐时光者 AI工程范式的又一次演进:Harness Engineering - DeepSky丶 本地系统对接大模型智能体的若干尝试 第二本书出版了:《Transformer技术纵深:架构解析与前沿突破》 - 罗西的思考 【Azure Developer】IIS w3wp.exe 的 -m 参数:一个未被记录的管道模式标识 英雄帖招募 - 虾饺爱下棋 AI开发-python-langchain框架(3-24-Plan-and-Execute Agent) - 万笑佛 避免这些编程陷阱:七种让你代码失控的开发风格 - 暮色之狐 从写代码到问问题:2026年,AI如何重构数据科学工作流 C#从零开始: LumNote-重新定义单机Markdown编辑器 - LdotJdot 【FAQ】HarmonyOS SDK 闭源开放能力 —Media Library Kit - HarmonyOS_SDK 我用fastapi-scaff搭了个项目,两天工期缩到两小时,老板以为我开挂了 - 一名程序媛呀 Rudist v0.5.1 发布:AI 驱动的 Redis 客户端,更快、更直观 SqlSugar 接入 PostgreSQL pgvector 完整方案(增删改查 + 强类型相似度查询) - HarryPei 今天不想硬撑?来领一张《摆烂许可证》 - 汀、人工智能 一次 Android 抓包引出的疑问:Tor Browser 桌面模式下为何出现直连请求 做 AI 应用必懂:Function Call 和 Skills,到底差在哪? - it排球君 Linux实操--组管理、权限管理和定时任务 - NE_STOP
用SymPy自动因式分解:从面积拼图到代数恒等式
wang_yb · 2026-06-13 · via 博客园_首页

大家好,今天要解决一个痛点是关于 因式分解公式 的。

直接说问题:
Manim 展示 $ x2 + 5x + 6 = (x+2)(x+3) $ 的“十字相乘”面积模型,你需要先想好怎样把大矩形拆成四块$ (x^2)、(2x)、(3x)、(6) $,再手动计算每一块的边长和位置。

换成 $ x^2 + 7x + 12 $,布局又要重新算。

有没有办法只告诉程序“我要做这个多项式因式分解的拼图证明”,它就能自动生成所有矩形的尺寸和坐标?

,用 SymPy

它能够根据代数式的结构自动推导几何布局,让你从“一个公式写一套代码”升级为“一类公式共用一套生成器”。

1. 痛点场景还原:公式一换,布局全乱

先看一个“传统”手动布局的例子。

我们想讲 $ x^2 + 5x + 6 = (x+2)(x+3) $ 的面积模型,把大矩形分成四块:

from manim import *

class ManualX2Plus5xPlus6(Scene):
    def construct(self):
        x = 2  # 让 x 取某个具体值来画图
        # 手动指定四块的宽高
        # 大矩形宽 (x+3),高 (x+2)
        # 四块:x*x, 3*x, 2*x, 2*3
        sq_x2 = Rectangle(width=x, height=x, color=BLUE, fill_opacity=0.4)
        rec_3x = Rectangle(width=3, height=x, color=RED, fill_opacity=0.4)
        rec_2x = Rectangle(width=x, height=2, color=RED, fill_opacity=0.4)
        rec_6 = Rectangle(width=3, height=2, color=GREEN, fill_opacity=0.4)

        # 手动拼:左下角 sq_x2,右边 rec_3x,上边 rec_2x,右上 rec_6
        sq_x2.shift(LEFT * (3 / 2) + DOWN * (2 / 2))
        rec_3x.next_to(sq_x2, RIGHT, buff=0)
        rec_2x.next_to(sq_x2, UP, buff=0)
        rec_6.next_to(rec_3x, UP, buff=0)

        self.play(Create(sq_x2), Create(rec_3x), Create(rec_2x), Create(rec_6))
        self.wait(1)

要改成 $ x^2 + 7x + 12 = (x+3)(x+4) $,你需要改:

  • 常数项 334
  • 矩形 rec_3x 的宽度 → 4
  • 矩形 rec_2x 的高度 → 3
  • 小矩形 rec_6 的宽高 → 43
  • shift 里面的偏移量也要跟着变。

每一次改公式都是一次手工劳动,而且容易出错。

如果公式结构稍微复杂一点(比如二次项系数不是 1,或者有三项展开),这种手动模式基本就难以维护了。

2. SymPy 解决方案:因式分解→几何布局

SymPy 的杀手锏就在这里:它可以对多项式进行因式分解或展开,自动抽取出每一项的系数和次数。

然后我们根据这些信息,自动生成对应的矩形拼图布局。

整体流程是:

  1. 输入一个多项式字符串,如 "x**2 + 5*x + 6"
  2. 用 SymPy 对其进行因式分解,得到 (x + 2)*(x + 3)
  3. 从因式结果中读出两个一次式的常数项(2 和 3),从展开式中读出各项系数(1, 5, 6);
  4. 自动生成四个矩形块:$ x^2 $(边长 $ x \()、\) 2x $(宽 2 高 $ x \()、\) 3x $(宽 $ x $ 高 3)、6(宽 2 高 3),并根据这些边长自动算出左下角坐标;
  5. 将布局信息输出为一个统一的数据结构,供 Manim 使用。

这样,无论你输入 $ x^2 + 7x + 12 $ 还是 $ x^2 - 3x + 2 $,甚至带参数的 $ x^2 + (a+b)x + ab $,布局生成逻辑都完全一样——你只需要换一个多项式字符串。

下面是一段纯 SymPy 的布局生成器代码,不依赖 Manim,可以直接在 Jupyter 里跑:

from sympy import symbols, factor, expand, Poly
from sympy.abc import x

def generate_layout(poly_expr):
    """
    输入:一个关于 x 的多项式(可因式分解为 (x+a)(x+b))
    输出:四个矩形的布局信息列表,每个矩形包括 (标签, 宽, 高, x坐标, y坐标)
    """
    # 因式分解
    factored = factor(poly_expr)            # 例如 (x + 2)*(x + 3)
    # 确保是两个一次因式
    if not factored.is_Mul:
        raise ValueError("多项式不能分解为两个一次因式")
    factors = factored.args
    if len(factors) != 2:
        raise ValueError("暂时只支持两个一次因式")

    # 提取常数项 a, b
    a = None
    b = None
    for fac in factors:
        if fac.is_Add:
            # 形如 x + c,提取常数 c
            coeff_dict = Poly(fac, x).as_dict()
            c = -coeff_dict.get(0, 0) if 1 in coeff_dict and coeff_dict[1]==1 else None
            # 这里做一个简单处理:假定就是 x - a 形式
            # 我们用 Poly 提取根
            p = Poly(fac, x)
            if p.degree() == 1:
                root = -p.coeff_monomial(1) / p.coeff_monomial(0)  # 其实标准形式是 x - r
                # 更稳健:fac = x + a0 => a0
                const = p.coeff_monomial(1)  # 系数 1
                const_term = p.coeff_monomial(0)  # a0
                if const == 1:
                    val = const_term
                else:
                    val = const_term / const
            if a is None:
                a = val
            else:
                b = val

    # 展开式获取各项
    expanded = expand(poly_expr)
    poly = Poly(expanded, x)
    coeff_x2 = poly.coeff_monomial(2)  # x^2 系数
    coeff_x  = poly.coeff_monomial(1)  # x 系数
    const    = poly.coeff_monomial(0)  # 常数项

    # 现在已知大矩形的宽 = (x + a),高 = (x + b),但我们是取具体 x 值画图
    # 这里的布局用符号 a, b 表达(注意 a,b 是常数数值,不是符号)
    # 返回布局,宽和高用数值表达式(含 x),留给 Manim 代入 x_val
    layout = [
        {"label": "x^2", "w": "x", "h": "x", "x0": 0, "y0": 0},
        {"label": f"{b}x", "w": b, "h": "x", "x0": "x", "y0": 0},
        {"label": f"{a}x", "w": "x", "h": a, "x0": 0, "y0": "x"},
        {"label": f"{a*b}", "w": b, "h": a, "x0": "x", "y0": "x"},
    ]
    return layout, a, b

# 示例:x^2 + 5x + 6
layout, a, b = generate_layout(x**2 + 5*x + 6)
print("a =", a, " b =", b)
for item in layout:
    print(item)

输出:

a = 2  b = 3
{'label': 'x^2', 'w': 'x', 'h': 'x', 'x0': 0, 'y0': 0}
{'label': '3x', 'w': 3, 'h': 'x', 'x0': 'x', 'y0': 0}
{'label': '2x', 'w': 'x', 'h': 2, 'x0': 0, 'y0': 'x'}
{'label': '6', 'w': 3, 'h': 2, 'x0': 'x', 'y0': 'x'}

可以看到,矩形的宽高和坐标全部由 ab 决定,而 a,b 是从 SymPy 因式分解中自动提取的。换一个多项式,只需改一行输入,布局自动生成。

3. Manim 联动实战:通用因式分解拼图动画

现在我们把这个自动布局引擎接入 Manim,做一个通用场景:你只需要在开头指定多项式,动画就会自动展示对应的面积拼图,并标注公式。

代码如下:

from manim import *
from sympy import factor, Poly
from sympy.abc import x

class AutoFactorPuzzle(Scene):
    def construct(self):
        # ========== 1. 在这里输入你要讲的多项式 ==========
        poly_expr = x**2 + 5 * x + 6  # 试着改成 x**2 + 7*x + 12 看看!
        x_val = 1.5  # 动画中 x 的具体取值(可调)

        # ========== 2. SymPy 自动分析多项式 ==========
        factored = factor(poly_expr)
        if not factored.is_Mul or len(factored.args) != 2:
            raise ValueError("多项式需能分解为两个一次因式 (x+a)(x+b)")
        # 提取 a, b
        roots = []
        for fac in factored.args:
            p = Poly(fac, x)
            if p.degree() != 1:
                raise ValueError("因式必须是一次式")
            # 一次多项式 ax + b,all_coeffs() 返回 [a, b]
            coeffs = p.all_coeffs()
            # 根为 -b/a
            root = -coeffs[1] / coeffs[0]
            roots.append(root)
        a, b = float(roots[0]), float(roots[1])  # 常数项 a, b

        # ========== 3. 根据 a, b 生成矩形布局 ==========
        # 四个矩形:x^2, b*x, a*x, a*b
        blocks = [
            {"label": "x^2", "w": x_val, "h": x_val, "x0": 0, "y0": 0, "color": BLUE},
            {
                "label": f"{abs(b)} \\times x",
                "w": abs(b),
                "h": x_val,
                "x0": x_val,
                "y0": 0,
                "color": RED,
            },
            {
                "label": f"{abs(a)} \\times x",
                "w": x_val,
                "h": abs(a),
                "x0": 0,
                "y0": x_val,
                "color": YELLOW,
            },
            {
                "label": f"{abs(b)} \\times {abs(a)}",
                "w": abs(b),
                "h": abs(a),
                "x0": x_val,
                "y0": x_val,
                "color": GREEN,
            },
        ]

        # ========== 4. Manim 绘制 ==========
        # 创建四个内部矩形
        for blk in blocks:
            rect = Rectangle(
                width=blk["w"],
                height=blk["h"],
                color=blk["color"],
                fill_opacity=0.4,
                stroke_width=1,
            )
            # 左下角坐标转换为 Manim 中心坐标
            rect.move_to([blk["x0"] + blk["w"] / 2, blk["y0"] + blk["h"] / 2, 0])

            # 偏移一个固定位置
            rect.shift(LEFT * 2 + DOWN)

            label = MathTex(f"{blk['label']}", font_size=16)
            label.move_to(rect.get_center())
            self.play(Create(rect), Write(label))

        self.wait(0.5)

        # 显示等式
        eq = MathTex(
            f"{{x}}^2 + {poly_expr.coeff(x,1)}x + {poly_expr.coeff(x,0)}",
            "=",
            f"(x{a if a<0 else abs(a)})(x{b if b<0 else abs(b)})",
            font_size=20,
        ).shift(DOWN * 1.5)
        self.play(Write(eq))
        self.wait(2)

关键解释

  • 多形式输入只改 poly_expr = ... 一行,整段代码就能绘制出对应的十字相乘面积图;
  • SymPy 的 factor 自动拆出一次因式,Poly 帮我们提取系数,ab 直接成为矩形的边长参数;
  • 整个布局的坐标逻辑只写了一次,通过 abx_val 计算出 blocks,没有手动偏移。

试一下把 poly_expr 改成 x**2 + 7*x + 12,重新运行,动画将立刻变成 $ x^2 \((蓝)、\) 4x \((红)、\) 3x \((黄)、\) 12 $(绿)的四块拼图,等式也变为 $ x^2+7x+12=(x+4)(x+3) $。

4. 效果展示说明

运行上面的 AutoFactorPuzzle,你会看到:

  1. 内部四块矩形依次浮现:蓝色 $ x^2 $ 在左下,右侧是一块红色 $ bx $,上方是一块 黄色 $ ax $,右上角是绿色常数块 $ ab $。所有块之间严丝合缝,因为它们的坐标和边长都来自同一个因式分解结果。
  2. 标签 $ x^2 \(、\) bx \(、\) ax \(、\) ab $ 精准居中,清晰对应代数项。
  3. 下方显示因式分解的等式,左边是展开式,右边是因式乘积,学生可以立刻将图形与公式对应起来。

更重要的是:换一个多项式只需要改一行代码,动画逻辑完全复用。

想一次性展示多个公式的对比?只需要建几个不同的场景实例即可。

这让“制作一套完整的因式分解几何证明微课”从一天的工作量缩短到一小时。

5. 小结

SymPy 在这个场景中扮演的角色,是一个代数结构的自动解析器

传统方法也能拼出图,但你得用大脑去“解析”公式,再手动翻译成矩形的宽高和位置;

SymPy 帮你自动完成“因式分解 → 提取系数 → 生成几何参数”的全过程,让你专注于教学设计和动画美化。

这个模式可以轻松扩展到:

  • 二次三项式的十字相乘面积模型(即本文的例子);
  • 完全平方和平方差公式的拼图证明;
  • 多项式乘法的面积表示(如 $ (x+a)(x+b)(x+c) $ 的体积模型);
  • 甚至更复杂的恒等式,只要你能把代数结构映射为空间分割。