# 📚 多层感知机(MLP)—— 从零理解神经网络(详细版) # 🎯 本章学习目标 | 目标 | 说明 | |------|------| | **理解** | 什么是函数——机器学习的核心数学工具 | | **理解** | 单层感知机为什么只能解决线性问题(含数学证明) | | **理解** | 多层感知机如何解决非线性问题(数学原理) | | **掌握** | 前向传播:数据如何在网络中流动(含矩阵维度分析) | | **掌握** | 反向传播:网络如何从错误中学习(完整链式法则推导) | | **掌握** | 用 NumPy 从零实现一个 MLP(不依赖任何框架) | | **理解** | 激活函数、损失函数、学习率等核心概念(含求导过程) | --- # 📖 第一部分:数学基础 —— 你必须知道的概念 ## 1.1 什么是函数? ### 1.1.1 函数的定义 **函数**是一种对应规则:输入一个值,经过某种规则,输出一个值。 数学定义: ``` 设 D 是一个非空数集,如果对于 D 中的每一个数 x,按照某种规则 f, 都能确定唯一一个数 y 与之对应,那么我们就说 f 是定义在 D 上的函数, 记作: y = f(x) 其中: - x 称为自变量(输入) - y 称为因变量(输出) - D 称为定义域(所有可能的输入) - f 称为对应规则(函数本身) ``` ### 1.1.2 函数的直观理解 ``` y = f(x) 可以想象为一个"加工机器": 输入 x --> [ f ] --> 输出 y 例子: - y = 2x + 1 是一个加工机器:输入 x,输出 2x+1 - y = x^2 是一个加工机器:输入 x,输出 x^2 - y = sin(x) 是一个加工机器:输入 x,输出 sin(x) ``` ### 1.1.3 神经网络本质上是一个复杂的函数 ``` 传统函数: y = f(x) = 2x + 1 神经网络(想象): y = f(x) = ReLU(W2 * ReLU(W1 * x + b1) + b2) 这是一个多层嵌套的函数! x = 输入(比如一张图片的像素) y = 输出(比如图片是数字几的概率) ``` ### 1.1.4 多元函数 神经网络处理的是多元函数,即有多个输入的函数: ``` z = f(x, y) = x + 2y 例子: 输入:x = 3, y = 5 输出:z = 3 + 2*5 = 13 在神经网络中: 输入可能是 784 个数(x1, x2, x3, ..., x784) 输出可能是 10 个数(y1, y2, ..., y10) z1 = f1(x1, x2, ..., x784) z2 = f2(x1, x2, ..., x784) ... z10 = f10(x1, x2, ..., x784) ``` --- ## 1.2 导数 —— 变化的度量 ### 1.2.1 导数的物理意义 导数描述的是"变化率": ``` 导数的定义: f'(x) = lim(h->0) [f(x+h) - f(x)] / h 含义: - 当 h 趋近于 0 时,差商 [f(x+h) - f(x)] / h 趋近于一个常数 - 这个常数就是函数 f 在点 x 处的导数 - 导数表示:在点 x 处,f 的值随着 x 的增加而变化的快慢 几何意义:切线的斜率 物理意义:瞬时变化率 ``` ### 1.2.2 常见函数的导数 ``` 1. 常数函数:f(x) = C f'(x) = 0 2. 幂函数:f(x) = x^n f'(x) = n * x^(n-1) 例子: f(x) = x^2 -> f'(x) = 2x f(x) = x^3 -> f'(x) = 3x^2 3. 指数函数:f(x) = e^x f'(x) = e^x 4. 对数函数:f(x) = ln(x) f'(x) = 1/x 5. 三角函数: f(x) = sin(x) -> f'(x) = cos(x) f(x) = cos(x) -> f'(x) = -sin(x) ``` ### 1.2.3 导数的运算法则 ``` 加法法则:[f(x) + g(x)]' = f'(x) + g'(x) 乘法法则:[f(x) * g(x)]' = f'(x) * g(x) + f(x) * g'(x) 除法法则:[f(x) / g(x)]' = [f'(x) * g(x) - f(x) * g'(x)] / g(x)^2 链式法则(最重要):[f(g(x))]' = f'(g(x)) * g'(x) ``` ### 1.2.4 链式法则详解 链式法则是理解反向传播的数学基础,必须深入理解! ``` 如果 y = f(u),且 u = g(x) 那么 y = f(g(x)) 是 x 的复合函数 链式法则: dy/dx = dy/du * du/dx 也可以写成: d/dx [f(g(x))] = f'(g(x)) * g'(x) 直观理解: dy/dx = dy/du * du/dx = (y 相对 u 的变化率) * (u 相对 x 的变化率) ``` **例子1**: ``` y = (2x + 1)^3 令 u = 2x + 1,则 y = u^3 dy/du = 3u^2 = 3(2x+1)^2 du/dx = 2 所以 dy/dx = 3u^2 * 2 = 6(2x+1)^2 = 6(2x+1)^2 ``` **例子2**: ``` y = e^(3x) 令 u = 3x,则 y = e^u dy/du = e^u = e^(3x) du/dx = 3 所以 dy/dx = e^(3x) * 3 = 3e^(3x) ``` ### 1.2.5 偏导数 当函数有多个自变量时,我们需要偏导数: ``` 设 z = f(x, y) = x^2 + 3xy + y^2 对 x 的偏导数(把 y 看作常数): ∂z/∂x = 2x + 3y 对 y 的偏导数(把 x 看作常数): ∂z/∂y = 3x + 2y 例子:计算 f(1, 2) 的偏导数 ∂z/∂x|(1,2) = 2*1 + 3*2 = 8 ∂z/∂y|(1,2) = 3*1 + 2*2 = 7 ``` --- ## 1.3 线性代数基础 ### 1.3.1 向量 **向量**是一串数字的有序排列: ``` 向量 a = [a1, a2, ..., an] 称为 n 维向量 例子: 速度 v = [vx, vy, vz] = [3, 4, 0] 是一个三维向量 像素 p = [255, 128, 64, ...] 是一个 784 维向量 向量的维度:向量中元素的个数 向量运算: 加法:[1,2] + [3,4] = [4,6] 数乘:2 * [1,2] = [2,4] 点积:[1,2] · [3,4] = 1*3 + 2*4 = 11 ``` **向量的点积(内积)**: ``` 向量 a = [a1, a2, ..., an] 向量 b = [b1, b2, ..., bn] 点积 a · b = a1*b1 + a2*b2 + ... + an*bn = Σ(ai*bi) 几何意义: a · b = |a| * |b| * cos(θ) 其中 |a| 是向量 a 的长度,θ 是两个向量的夹角 当两个向量垂直时(θ = 90°),cos(90°) = 0,点积为 0 ``` ### 1.3.2 矩阵 **矩阵**是一个数字的矩形表格: ``` 矩阵 A = [[a11, a12, ..., a1n], [a21, a22, ..., a2n], ... [am1, am2, ..., amn]] 称为 m 行 n 列的矩阵,记作 A(m x n) 例子: 图片(28x28)是一个 28 行 28 列的矩阵: A = [[0, 128, 255, ..., 0], [255, 0, 128, ..., 45], ... [45, 100, 0, ..., 78]] ``` ### 1.3.3 矩阵乘法 这是神经网络最核心的运算! ``` 矩阵 A (m x n) 与矩阵 B (n x k) 的乘积 C (m x k): C = A @ B 其中 C 的每个元素: C[i,j] = Σ(A[i,:] * B[:,j]) = A[i,1]*B[1,j] + A[i,2]*B[2,j] + ... + A[i,n]*B[n,j] ``` **例子**: ``` A (2x3) = [[1, 2, 3], B (3x2) = [[7, 8], [4, 5, 6]] [9, 10], [11, 12]] C = A @ B (2x2): C[0,0] = 1*7 + 2*9 + 3*11 = 7 + 18 + 33 = 58 C[0,1] = 1*8 + 2*10 + 3*12 = 8 + 20 + 36 = 64 C[1,0] = 4*7 + 5*9 + 6*11 = 28 + 45 + 66 = 139 C[1,1] = 4*8 + 5*10 + 6*12 = 32 + 50 + 72 = 154 C = [[58, 64], [139, 154]] ``` ### 1.3.4 神经网络中的矩阵运算 ``` 输入向量 x (784,) 权重矩阵 W (784 x 128) 偏置向量 b (128,) 计算:z = x @ W + b x 是 1 行 784 列的矩阵 (1 x 784) W 是 784 行 128 列的矩阵 (784 x 128) x @ W 是 1 行 128 列的矩阵 (1 x 128) z 是 1 行 128 列的矩阵 (1 x 128) b 是 1 行 128 列的矩阵 (1 x 128),与 z 相加 批量计算时(batch_size = 64): X (64 x 784) @ W (784 x 128) = Z (64 x 128) ``` ### 1.3.5 矩阵的转置 ``` 矩阵 A 的转置 A^T:行列互换 A (m x n) -> A^T (n x m) 例子: A = [[1, 2, 3], A^T = [[1, 4], [4, 5, 6]] [2, 5], [3, 6]] 重要公式: (A @ B)^T = B^T @ A^T 在反向传播中经常用到! ``` --- # 📖 第二部分:生物神经网络与人工神经元 ## 2.1 生物大脑中的神经元 ### 2.1.1 神经元的结构 ``` 树突(接收信号) / | \ ╱ ╱ ╲ ╲ ┌─────────────────────────┐ │ 细胞体 │ │ (处理输入,产生输出) │ └─────────────────────────┘ │ │ 轴突 │ 突触(连接到其他神经元) ``` **各部分功能**: - **树突**:接收来自其他神经元的信号 - **细胞体**:汇总所有输入,如果超过阈值则产生信号 - **轴突**:传输细胞体产生的信号 - **突触**:神经元之间的连接点,决定信号传递的"强度" ### 2.1.2 神经元的工作原理 ``` 简化模型: 1. 多个神经元的信号通过树突传入 2. 每个输入信号有一个"权重"(突触强度) 3. 细胞体把所有信号加权求和 4. 如果总和 > 阈值,神经元激活,输出信号 5. 信号通过轴突传给其他神经元 数学表示: y = 激活函数(Σ wi*xi - θ) 其中: xi = 第 i 个输入信号 wi = 第 i 个突触的权重(权重越大,该信号越重要) θ = 阈值(决定神经元何时激活) 激活函数 = 把总和转换成输出 ``` ### 2.1.3 生活中的类比:部门开会投票 ``` 想象一个部门开会决策: 输入(每个人的意见): - 员工 A 的意见(重要性 0.8) - 员工 B 的意见(重要性 0.5) - 员工 C 的意见(重要性 0.9) 汇总计算: 总分 = 0.8*意见A + 0.5*意见B + 0.9*意见C 决策: 如果 总分 > 阈值(比如 1.5) 则 部门决定"通过" 否则 部门决定"不通过" 这就是人工神经网络的原理! ``` --- ## 2.2 人工神经元(感知机) ### 2.2.1 感知机的数学模型 **感知机(Perceptron)**是最简单的人工神经元模型,由 Frank Rosenblatt 于 1958 年提出。 ``` 感知机的数学模型: 输入:x1, x2, ..., xn(n 个输入) 权重:w1, w2, ..., wn(每个输入对应一个权重) 偏置:b(阈值,取负号后称为偏置) 计算: z = x1*w1 + x2*w2 + ... + xn*wn + b 用向量形式:z = x · w + b 其中 x = [x1, x2, ..., xn],w = [w1, w2, ..., wn] 激活函数: 如果 z > 0,则 y = 1(激活) 如果 z <= 0,则 y = 0(不激活) 这叫"阶跃函数": y = { 1, if z > 0 { 0, if z <= 0 ``` ### 2.2.2 感知机的向量形式 ``` 输入向量:x = [x1, x2, ..., xn] (1 x n) 权重向量:w = [w1, w2, ..., wn] (1 x n) 加权求和:z = x · w + b = Σ(xi * wi) + b 这就是两个向量的点积再加上偏置! ``` ### 2.2.3 用代码实现感知机 ```python import numpy as np class Perceptron: """单个感知机(人工神经元)""" def __init__(self, input_size): """ 初始化感知机 input_size: 输入的维度(有多少个输入) """ # 初始化权重为小的随机数 self.w = np.random.randn(input_size) * 0.01 # 初始化偏置为 0 self.b = 0 def forward(self, x): """ 前向传播:计算感知机的输出 x: 输入向量,比如 (784,) """ # 计算加权求和:z = x · w + b z = np.dot(x, self.w) + self.b # 阶跃激活函数 if z > 0: return 1 else: return 0 def __call__(self, x): """让对象可以像函数一样调用""" return self.forward(x) ``` ### 2.2.4 感知机的几何意义 ``` 感知机本质上是一个超平面: z = x · w + b = 0 这个方程定义了一个 n 维空间中的超平面(n-1 维子空间) 对于二维输入(x1, x2): w1*x1 + w2*x2 + b = 0 是一条直线 对于三维输入(x1, x2, x3): w1*x1 + w2*x2 + w3*x3 + b = 0 是一个平面 感知机的作用: - 把输入空间分成两部分 - 一边输出 1,一边输出 0 ``` --- # 📖 第三部分:单层感知机的局限——数学证明 ## 3.1 感知机可以解决什么问题? ### 3.1.1 AND 门(与门) ``` AND 门的真值表: 输入 A 输入 B 输出 ───────────────── 0 0 0 0 1 0 1 0 0 1 1 1 <- 只有两个都是 1 时才输出 1 我们可以用感知机实现 AND 门: 设 w1 = 0.5, w2 = 0.5, b = -0.7 验证: A=0, B=0: z = 0*0.5 + 0*0.5 - 0.7 = -0.7 <= 0 -> y=0 [OK] A=0, B=1: z = 0*0.5 + 1*0.5 - 0.7 = -0.2 <= 0 -> y=0 [OK] A=1, B=0: z = 1*0.5 + 0*0.5 - 0.7 = -0.2 <= 0 -> y=0 [OK] A=1, B=1: z = 1*0.5 + 1*0.5 - 0.7 = 0.3 > 0 -> y=1 [OK] 几何解释: 在二维坐标系中,直线 0.5*x1 + 0.5*x2 - 0.7 = 0 把 (0,0), (0,1), (1,0) 分到一边(输出 0) 把 (1,1) 分到另一边(输出 1) ``` ### 3.1.2 OR 门(或门) ``` OR 门的真值表: 输入 A 输入 B 输出 ───────────────── 0 0 0 0 1 1 1 0 1 1 1 1 <- 至少有一个是 1 时输出 1 用感知机实现:w1 = 0.5, w2 = 0.5, b = -0.3 验证: A=0, B=0: z = 0*0.5 + 0*0.5 - 0.3 = -0.3 <= 0 -> y=0 [OK] A=0, B=1: z = 0*0.5 + 1*0.5 - 0.3 = 0.2 > 0 -> y=1 [OK] A=1, B=0: z = 1*0.5 + 0*0.5 - 0.3 = 0.2 > 0 -> y=1 [OK] A=1, B=1: z = 1*0.5 + 1*0.5 - 0.3 = 0.7 > 0 -> y=1 [OK] ``` ### 3.1.3 NOT 门(非门) ``` NOT 门的真值表: 输入 A 输出 ───────── 0 1 1 0 用感知机实现:w1 = -0.5, b = 0.3 验证: A=0: z = 0*(-0.5) + 0.3 = 0.3 > 0 -> y=1 [OK] A=1: z = 1*(-0.5) + 0.3 = -0.2 <= 0 -> y=0 [OK] ``` --- ## 3.2 XOR 门——感知机的致命缺陷 ### 3.2.1 XOR 的定义 ``` XOR(异或)门的真值表: 输入 A 输入 B 输出 ───────────────── 0 0 0 0 1 1 1 0 1 1 1 0 <- 两者不一样时输出 1 特点:相同输出 0,不同输出 1 ``` ### 3.2.2 尝试用感知机实现 XOR ``` 假设存在感知机使得 XOR 成立,即存在 w1, w2, b 使得: 当 (A,B) = (0,0): w1*0 + w2*0 + b <= 0 当 (A,B) = (0,1): w1*0 + w2*1 + b > 0 当 (A,B) = (1,0): w1*1 + w2*0 + b > 0 当 (A,B) = (1,1): w1*1 + w2*1 + b <= 0 从第一个条件:b <= 0 从第二个条件:w2 + b > 0 从第三个条件:w1 + b > 0 从第四个条件:w1 + w2 + b <= 0 把 b <= 0 代入第二个:w2 + b > 0 -> w2 > -b >= 0 -> w2 > 0 把 b <= 0 代入第三个:w1 + b > 0 -> w1 > -b >= 0 -> w1 > 0 所以 w1 > 0, w2 > 0, b <= 0 但第四个条件 w1 + w2 + b <= 0 由于 w1 > 0, w2 > 0, b <= 0 如果 b <= 0 但非常接近 0,比如 b = -ε 那么 w1 + w2 - ε <= 0 -> w1 + w2 <= ε 这意味着 w1 和 w2 必须非常小,接近 0 但根据 w1 > 0, w2 > 0 且 w1 + w2 <= ε 这不可能同时满足第二个 w1 + b > 0 -> w1 - ε > 0 -> w1 > ε 和第三个 w2 - ε > 0 -> w2 > ε 所以 w1 + w2 > 2ε,与 w1 + w2 <= ε 矛盾! 因此:感知机无法实现 XOR! ``` ### 3.2.3 XOR 的几何解释 ``` AND 门(线性可分): B ^ 1 | * | | | | +----------> A 0 0 可以用一条直线(超平面)把 *(输出1)和 。(输出0)分开 OR 门(线性可分): B ^ 1 | * * | | | | +----------> A 0 0 可以用一条直线把 *(输出1)和 。(输出0)分开 XOR 门(线性不可分): B ^ 1 | * . | . | * | | +----------> A 0 0 *(输出1)在对角线:(0,1) 和 (1,0) .(输出0)在对角线:(0,0) 和 (1,1) 无法用一条直线把 * 和 . 分开! 必须用两条直线(折线)才能分开 ``` ### 3.2.4 线性可分与线性不可分 ``` 定义: 一组数据点 {x(i)} 被称为线性可分的,如果存在一个超平面 w · x + b = 0 能够把所有正类点分到一边,所有负类点分到另一边。 换句话说: 如果存在 w, b 使得: 对于所有正类:w · x + b > 0 对于所有负类:w · x + b <= 0 那么这组数据是线性可分的。 AND, OR, NOT 都是线性可分的 XOR 不是线性可分的 ``` --- # 📖 第四部分:多层感知机(MLP)—— 解决非线性问题 ## 4.1 多层的思想 ### 4.1.1 为什么多层可以解决 XOR? ``` 单层感知机的局限: - 只能画一条直线(一个超平面) - 只能解决线性可分问题 多层感知机的能力: - 第一层:学多条直线(多个超平面) - 第二层:综合第一层的结果,画出更复杂的边界 XOR 的多层解决方案: 输入层 (x1, x2) | v 隐藏层:两个感知机 h1, h2 | v 输出层:综合判断 具体: h1 = AND(NOT x1, x2) <- 在区域 (0,1) 输出 1 h2 = AND(x1, NOT x2) <- 在区域 (1,0) 输出 1 y = OR(h1, h2) <- 任何一个为 1 就输出 1 ``` ### 4.1.2 多层的几何解释 ``` 单层:一条直线把空间分成两部分 多层:多条直线组合,把空间分成多个区域 XOR 的解决过程: 第一步:两个隐藏单元各自画一条直线 h1: w1=[-1,1], b=0.5 -> 识别 (0,1) 区域 h2: w2=[1,-1], b=0.5 -> 识别 (1,0) 区域 第二步:输出层综合 y = OR(h1, h2) 效果:把原本无法用一条直线分开的问题, 转换成两步:先画两条线,再综合 ``` --- ## 4.2 MLP 的网络结构 ### 4.2.1 本项目的 MLP 结构 ``` 网络结构: 784 -> 128 -> 10 输入层: - 784 个神经元(对应 28x28 图片的 784 个像素值) - 输入向量 x:形状 (batch_size, 784) 隐藏层: - 128 个神经元 - 每个神经元:z1[i] = x @ W1[:,i] + b1[i] - 激活后:a1[i] = ReLU(z1[i]) 输出层: - 10 个神经元(对应数字 0-9) - 每个神经元输出一个"分数" - 通过 Softmax 转换成概率 ``` ### 4.2.2 网络的矩阵表示 ``` 输入:X (batch_size, 784) 第一层(输入 -> 隐藏): Z1 = X @ W1 + b1 W1 形状: (784, 128) b1 形状: (128,) Z1 形状: (batch_size, 128) A1 = ReLU(Z1) 第二层(隐藏 -> 输出): Z2 = A1 @ W2 + b2 W2 形状: (128, 10) b2 形状: (10,) Z2 形状: (batch_size, 10) A2 = Softmax(Z2) # 最终概率输出 ``` ### 4.2.3 参数量计算 ``` 总参数量: W1: 784 × 128 = 100,352 个 b1: 128 个 W2: 128 × 10 = 1,280 个 b2: 10 个 总计:101,770 个参数 这就是为什么深度学习需要大量参数! 参数量决定了网络能学多复杂的模式。 ``` --- ## 4.3 激活函数 ### 4.3.1 为什么需要激活函数? ``` 如果网络只有线性变换: Z1 = X @ W1 + b1 Z2 = A1 @ W2 + b2 那么整个网络仍然是线性的! Z2 = (X @ W1 + b1) @ W2 + b2 = X @ (W1 @ W2) + (b1 @ W2 + b2) = X @ W + b 这等价于一个单层网络! 所以:必须加入非线性激活函数,才能让多层网络有意义! ``` ### 4.3.2 ReLU 激活函数 ``` ReLU(x) = max(0, x) = { x, if x > 0 0, if x <= 0 } 函数图像: y | / | / | / | / +----------- x 0 数学性质: - 正半轴:导数 = 1 - 负半轴:导数 = 0 - 在 x=0 处:不可导(但实际使用中可以任意选择一边) 导数(求导过程): 当 x > 0 时,ReLU(x) = x,所以 d/dx ReLU(x) = 1 当 x < 0 时,ReLU(x) = 0,所以 d/dx ReLU(x) = 0 当 x = 0 时,导数不存在,但在实践中常定义为 0 或 1 ReLU'(x) = { 1, if x > 0 { 0, if x <= 0 ``` ### 4.3.3 Softmax 激活函数 ``` Softmax 把一个向量转换成概率分布 对于向量 z = [z1, z2, ..., zk]: softmax(zi) = exp(zi) / Σj exp(zj) 特点: 1. 每个 softmax(zi) > 0(因为 exp 永远是正数) 2. Σi softmax(zi) = 1(因为分母是所有分子的和) 所以 softmax 的输出是一个合法的概率分布! 数值稳定技巧: 由于 exp 可能溢出,通常用以下技巧: softmax(zi) = exp(zi - C) / Σj exp(zj - C) 其中 C = max(z),这样所有指数都不会太大 ``` **Softmax 的求导**: ``` 设 y_i = softmax(z_i) = exp(z_i) / Σj exp(z_j) 当 i == j 时(对角元素): ∂y_i/∂z_i = [exp(z_i) * Σj exp(z_j) - exp(z_i) * exp(z_i)] / (Σj exp(z_j))^2 = exp(z_i)(Σj exp(z_j) - exp(z_i)) / (Σj exp(z_j))^2 = exp(z_i) / Σj exp(z_j) * (Σj exp(z_j) - exp(z_i)) / Σj exp(z_j) = y_i * (1 - y_i) 当 i != j 时(非对角元素): ∂y_i/∂z_j = [0 * Σj exp(z_j) - exp(z_i) * exp(z_j)] / (Σj exp(z_j))^2 = -exp(z_i) * exp(z_j) / (Σj exp(z_j))^2 = -y_i * y_j 总结: ∂y_i/∂z_j = { y_i * (1 - y_i), if i == j { -y_i * y_j, if i != j ``` --- # 📖 第五部分:前向传播 —— 详细数学推导 ## 5.1 前向传播的完整流程 ``` 输入:x (784,) 第一步:输入层 -> 隐藏层 z1 = x @ W1 + b1 a1 = ReLU(z1) 第二步:隐藏层 -> 输出层 z2 = a1 @ W2 + b2 y = softmax(z2) ``` ### 5.1.1 第一步:输入层 -> 隐藏层 ``` 给定: x: (784,) # 一张图片的像素值 W1: (784, 128) # 权重矩阵 b1: (128,) # 偏置向量 计算 z1: z1[i] = Σj x[j] * W1[j,i] + b1[i] z1 的形状: (128,) 直观理解:每个隐藏神经元 i 接收所有输入的加权求和 计算 a1(ReLU 激活): a1[i] = max(0, z1[i]) a1 的形状: (128,) 批量计算(batch_size = N): X: (N, 784) Z1 = X @ W1 + b1: (N, 128) A1 = ReLU(Z1): (N, 128) ``` ### 5.1.2 第二步:隐藏层 -> 输出层 ``` 给定: a1: (128,) # 隐藏层激活 W2: (128, 10) # 权重矩阵 b2: (10,) # 偏置向量 计算 z2: z2[i] = Σj a1[j] * W2 a1[j] * W2[j,i] + b2[i] z2 的形状: (10,) 直观理解:每个输出神经元 i 综合所有隐藏层的激活 计算 y(Softmax): y[i] = softmax(z2)[i] = exp(z2[i]) / Σj exp(z2[j]) y 的形状: (10,) # 10 个数字的概率分布 批量计算(batch_size = N): A1: (N, 128) Z2 = A1 @ W2 + b2: (N, 10) Y = Softmax(Z2): (N, 10) ``` --- ## 5.2 前向传播的 NumPy 实现 ```python import numpy as np class MLP: def __init__(self, input_size=784, hidden_size=128, output_size=10): # Xavier 初始化 self.W1 = np.random.randn(input_size, hidden_size) * np.sqrt(2.0 / input_size) self.b1 = np.zeros(hidden_size) self.W2 = np.random.randn(hidden_size, output_size) * np.sqrt(2.0 / hidden_size) self.b2 = np.zeros(output_size) def relu(self, x): return np.maximum(0, x) def softmax(self, x): # 数值稳定技巧 x_shifted = x - np.max(x, axis=1, keepdims=True) exp_x = np.exp(x_shifted) return exp_x / np.sum(exp_x, axis=1, keepdims=True) def forward(self, X): """ 前向传播 X: (batch_size, input_size) 输入图片 返回: (batch_size, output_size) 每个类的概率 """ # 第一层 self.z1 = X @ self.W1 + self.b1 # (batch, 784) @ (784, 128) = (batch, 128) self.a1 = self.relu(self.z1) # (batch, 128) # 第二层 self.z2 = self.a1 @ self.W2 + self.b2 # (batch, 128) @ (128, 10) = (batch, 10) self.probs = self.softmax(self.z2) # (batch, 10) return self.probs ``` --- # 📖 第六部分:反向传播 —— 完整数学推导 > 这是神经网络最核心的部分,包含了完整的链式法则推导。 ## 6.1 损失函数 ### 6.1.1 交叉熵损失函数 ``` 对于多分类问题,我们使用交叉熵损失(Cross-Entropy Loss) 设: y_true: 真实标签的 One-Hot 编码 (10,) # 比如数字 5:[0,0,0,0,0,1,0,0,0,0] y_pred: 网络预测的概率分布 (10,) # 比如:[0.1, 0.05, 0.02, 0.03, 0.1, 0.6, 0.02, 0.05, 0.02, 0.01] 交叉熵损失: L = -Σ y_true[i] * log(y_pred[i]) 由于 y_true 是 One-Hot 编码,只有一个元素是 1,其他都是 0 所以 L = -log(y_pred[真实类别]) 例子: 真实标签是 5,y_true[5] = 1 网络预测 y_pred[5] = 0.6 L = -log(0.6) = 0.51 网络预测 y_pred[5] = 0.9 L = -log(0.9) = 0.11 预测越准确,损失越小! ``` ### 6.1.2 交叉熵的求导 ``` 设 L = -Σ y_true[i] * log(y_pred[i]) 对 y_pred[i] 求偏导: ∂L/∂y_pred[i] = -y_true[i] / y_pred[i] 但是我们实际上是对 z2(Softmax 的输入)求导,而不是对 y_pred。 设 y_pred = softmax(z2) 我们需要求:∂L/∂z2[i] ``` --- ## 6.2 反向传播的数学推导 ### 6.2.1 输出层梯度 ``` 已知: L = -Σ y_true[j] * log(y_pred[j]) y_pred[j] = softmax(z2)[j] = exp(z2[j]) / Σk exp(z2[k]) 我们需要求:∂L/∂z2[i] 根据链式法则: ∂L/∂z2[i] = Σj ∂L/∂y_pred[j] * ∂y_pred[j]/∂z2[i] = Σj (-y_true[j] / y_pred[j]) * ∂y_pred[j]/∂z2[i] 根据 Softmax 的求导结果: ∂y_i/∂z_j = { y_i * (1 - y_i), if i == j { -y_i * y_j, if i != j 所以: ∂L/∂z2[i] = Σj (-y_true[j] / y_pred[j]) * ∂y_pred[j]/∂z2[i] = (-y_true[i] / y_pred[i]) * y_pred[i] * (1 - y_pred[i]) + Σ_{j≠i} (-y_true[j] / y_pred[j]) * (-y_pred[i] * y_pred[j]) = -y_true[i] * (1 - y_pred[i]) + Σ_{j≠i} y_true[j] * y_pred[i] = -y_true[i] + y_true[i] * y_pred[i] + Σ_{j≠i} y_true[j] * y_pred[i] = -y_true[i] + Σj y_true[j] * y_pred[i] = -y_true[i] + y_pred[i] * Σj y_true[j] = -y_true[i] + y_pred[i] * 1 (因为 y_true 是 One-Hot,和为1) = y_pred[i] - y_true[i] 这就是著名的结论:Softmax + Cross-Entropy 的梯度 = y_pred - y_true! d_z2 = y_pred - y_true ``` ### 6.2.2 第二层权重梯度 ``` 现在我们已经知道 d_z2 = y_pred - y_true 需要求: ∂L/∂W2[i,j] ∂L/∂b2[i] 根据链式法则: L 是 z2 的函数,z2 是 W2, b2 的函数 z2[j] = Σk a1[k] * W2[k,j] + b2[j] ∂z2[j]/∂W2[i,j] = a1[i] # 只有 W2[i,j] 影响 z2[j],系数是 a1[i] ∂z2[j]/∂b2[j] = 1 所以: ∂L/∂W2[i,j] = Σj ∂L/∂z2[j] * ∂z2[j]/∂W2[i,j] = Σj d_z2[j] * a1[i] = a1[i] * Σj d_z2[j] = a1[i] * d_z2[j] 向量形式: d_W2 = a1.T @ d_z2 # (128, 10) ∂L/∂b2[i] = Σj ∂L/∂z2[j] * ∂z2[j]/∂b2[i] = d_z2[i] 向量形式: d_b2 = d_z2 # (10,) ``` ### 6.2.3 隐藏层梯度 ``` 现在需要把梯度反向传播到隐藏层。 我们需要求: ∂L/∂a1[i] # 损失对隐藏层激活的梯度 ∂L/∂z1[i] # 损失对 ReLU 输入的梯度 第一步:求 ∂L/∂a1 根据前向传播: z2[j] = Σk a1[k] * W2[k,j] + b2[j] 所以: ∂z2[j]/∂a1[k] = W2[k,j] ∂L/∂a1[k] = Σj ∂L/∂z2[j] * ∂z2[j]/∂a1[k] = Σj d_z2[j] * W2[k,j] = Σj W2[k,j] * d_z2[j] 向量形式: d_a1 = d_z2 @ self.W2.T # (batch, 128) 第二步:求 ∂L/∂z1 根据前向传播: a1[i] = ReLU(z1[i]) ReLU 的导数: ∂a1[i]/∂z1[i] = 1, if z1[i] > 0 = 0, if z1[i] <= 0 所以: ∂L/∂z1[i] = ∂L/∂a1[i] * ∂a1[i]/∂z1[i] = d_a1[i] * ReLU'(z1[i]) 向量形式: d_z1 = d_a1 * (z1 > 0).astype(float) ``` ### 6.2.4 第一层权重梯度 ``` 现在求损失对第一层权重的梯度。 我们需要求: ∂L/∂W1[i,j] ∂L/∂b1[i] 第一步:求 ∂L/∂z1 我们已经知道: d_z1[i] = ∂L/∂z1[i] 第二步:求 ∂L/∂W1[i,j] 根据前向传播: z1[j] = Σk x[k] * W1[k,j] + b1[j] ∂z1[j]/∂W1[i,j] = x[i] ∂L/∂W1[i,j] = Σj ∂L/∂z1[j] * ∂z1[j]/∂W1[i,j] = Σj d_z1[j] * x[i] = x[i] * Σj d_z1[j] 向量形式: d_W1 = X.T @ d_z1 # (784, 128) 第三步:求 ∂L/∂b1 ∂L/∂b1[i] = Σj ∂L/∂z1[j] * ∂z1[j]/∂b1[i] = d_z1[i] 向量形式: d_b1 = d_z1 # (128,) ``` --- ## 6.3 梯度更新 ``` 得到所有梯度后,用梯度下降更新权重: W1 = W1 - learning_rate * d_W1 b1 = b1 - learning_rate * d_b1 W2 = W2 - learning_rate * d_W2 b2 = b2 - learning_rate * d_b2 注意:这里梯度已经除以了 batch_size(因为损失是平均的) ``` --- ## 6.4 完整反向传播代码 ```python def backward(self, X, y_true): """ 反向传播:计算梯度并更新权重 参数: X: (batch_size, 784) 输入图片 y_true: (batch_size, 10) One-Hot 真实标签 """ batch_size = X.shape[0] # ===== 第1步:输出层梯度 ===== # Softmax + CrossEntropy 的组合梯度:d_z2 = y_pred - y_true d_z2 = self.probs - y_true # (batch_size, 10) # ===== 第2步:第二层权重梯度 ===== d_W2 = self.a1.T @ d_z2 # (128, 10) d_b2 = np.sum(d_z2, axis=0) # (10,) # ===== 第3步:隐藏层梯度 ===== d_a1 = d_z2 @ self.W2.T # (batch_size, 128) d_z1 = d_a1 * (self.z1 > 0).astype(float) # ReLU 导数 # ===== 第4步:第一层权重梯度 ===== d_W1 = X.T @ d_z1 # (784, 128) d_b1 = np.sum(d_z1, axis=0) # (128,) # ===== 第5步:梯度裁剪(防止梯度爆炸) ===== max_grad = 1.0 for dW in [d_W1, d_W2]: np.clip(dW, -max_grad, max_grad, out=dW) for db in [d_b1, d_b2]: np.clip(db, -max_grad, max_grad, out=db) # ===== 第6步:梯度下降更新权重 ===== self.W1 -= self.lr * d_W1 / batch_size self.b1 -= self.lr * d_b1 / batch_size self.W2 -= self.lr * d_W2 / batch_size self.b2 -= self.lr * d_b2 / batch_size ``` --- # 📖 第七部分:训练流程 ## 7.1 完整训练循环 ```python def fit(self, X_train, y_train, epochs=50, batch_size=64): """ 训练 MLP 参数: X_train: (N, 784) 训练图片 y_train: (N, 10) 训练标签(One-Hot) epochs: 训练轮数 batch_size: 每批图片数 """ N = len(X_train) num_batches = (N + batch_size - 1) // batch_size for epoch in range(epochs): # 打乱数据 indices = np.random.permutation(N) X_shuffled = X_train[indices] y_shuffled = y_train[indices] epoch_loss = 0 # 批训练 for batch_idx in range(num_batches): start = batch_idx * batch_size end = min(start + batch_size, N) X_batch = X_shuffled[start:end] y_batch = y_shuffled[start:end] # 前向传播 probs = self.forward(X_batch) # 反向传播 self.backward(X_batch, y_batch) # 计算损失 loss = self.cross_entropy_loss(probs, y_batch) epoch_loss += loss # 打印进度 if (epoch + 1) % 5 == 0: acc = self.accuracy(X_train, y_train) print(f"Epoch {epoch+1:3d}/{epochs} | Loss: {epoch_loss/num_batches:.4f} | 准确率: {acc:.4f}") def cross_entropy_loss(self, probs, y_true): """交叉熵损失""" # 取真实类别的概率 correct_probs = probs[np.arange(len(y_true)), y_true.argmax(axis=1)] # 避免 log(0) loss = -np.mean(np.log(np.clip(correct_probs, 1e-10, 1.0))) return loss def accuracy(self, X, y): """计算准确率""" if len(y.shape) > 1: y = np.argmax(y, axis=1) predictions = np.argmax(self.forward(X), axis=1) return np.mean(predictions == y) ``` --- ## 7.2 超参数详解 | 超参数 | 默认值 | 说明 | 调参建议 | |--------|--------|------|---------| | **hidden_size** | 128 | 隐藏层神经元数 | 多→能力强但慢,少→快但弱 | | **learning_rate** | 0.1 | 学习率(每步走多远) | 大→震荡,小→太慢 | | **epochs** | 50 | 训练轮数 | 多→过拟合,少→欠拟合 | | **batch_size** | 64 | 每批图片数 | 大→稳定,小→泛化好 | --- # 📖 第八部分:MNIST 数据集 ## 8.1 MNIST 数据集介绍 ``` MNIST = Modified National Institute of Standards and Technology 美国国家标准与技术研究院改良数据集 数据集信息: - 70,000 张图片(60,000 训练 + 10,000 测试) - 每张图片 28x28 像素,灰度图 - 10 类(数字 0-9) - 来自 250 个不同人的手写样本 ``` ## 8.2 数据预处理 ```python def load_and_preprocess(): """加载并预处理 MNIST 数据""" # 1. 加载数据(使用 sklearn) from sklearn.datasets import fetch_openml mnist = fetch_openml('mnist_784', version=1) X, y = mnist.data, mnist.target # 2. 转换为 float32 并归一化到 [0, 1] X = X.values.astype(np.float32) / 255.0 # 3. 标签转为 One-Hot y = y.astype(int) num_classes = 10 y_onehot = np.zeros((len(y), num_classes)) y_onehot[np.arange(len(y)), y] = 1 return X, y_onehot ``` --- # 📖 第九部分:梯度消失与梯度爆炸 ## 9.1 什么是梯度消失? ``` 在反向传播过程中,梯度可能会变得非常小(接近 0) 原因: - 链式法则的乘法效应 - 深层网络中,梯度逐层相乘 - 如果每层的梯度都小于 1,深层的梯度会指数衰减 例子: 假设每层的梯度都是 0.5 第 1 层:梯度 = 0.5 第 5 层:梯度 = 0.5^5 = 0.03 第 10 层:梯度 = 0.5^10 = 0.001 深层的权重几乎学不到东西! ``` ## 9.2 什么是梯度爆炸? ``` 与梯度消失相反,梯度可能会变得非常大 原因: - 每层的梯度大于 1 - 深层网络中,梯度指数增长 例子: 假设每层的梯度都是 2 第 1 层:梯度 = 2 第 5 层:梯度 = 2^5 = 32 第 10 层:梯度 = 2^10 = 1024 权重更新过大,网络不稳定! ``` ## 9.3 解决方案 ``` 梯度裁剪: d_W = clip(d_W, -max_grad, max_grad) 本项目使用这种方法,效果良好 ReLU 激活函数: - 正半轴梯度为 1,不会衰减 - 缓解梯度消失问题 Xavier/He 初始化: - 让梯度在传播过程中保持合理范围 - 本项目使用 Xavier 初始化 ``` --- # 📖 第十部分:总结与公式速查 ## 10.1 核心公式汇总 | 公式 | 说明 | |------|------| | `z1 = X @ W1 + b1` | 第一层线性变换 | | `a1 = ReLU(z1)` | 第一层激活 | | `z2 = a1 @ W2 + b2` | 第二层线性变换 | | `y = softmax(z2)` | Softmax 概率输出 | | `L = -Σ y_true * log(y_pred)` | 交叉熵损失 | | `d_z2 = y_pred - y_true` | Softmax+CE 梯度 | | `d_W2 = a1.T @ d_z2` | 第二层权重梯度 | | `d_z1 = d_a1 * (z1 > 0)` | ReLU 梯度 | | `d_W1 = X.T @ d_z1` | 第一层权重梯度 | | `W = W - lr * d_W` | 梯度下降更新 | ## 10.2 训练流程图 ``` 输入数据 X | v 前向传播 -> 保存中间结果 (z1, a1, z2, probs) | v 计算损失 L = -Σ y_true * log(probs) | v 反向传播(从后往前): d_z2 = probs - y_true d_W2 = a1.T @ d_z2 d_z1 = (d_z2 @ W2.T) * ReLU'(z1) d_W1 = X.T @ d_z1 | v 梯度下降更新权重 | v 重复,直到损失足够小 ``` --- # 📝 课后练习 ## 练习1:手算前向传播 已知: ``` x = [1, 0, 1] # 输入 W1 = [[1, 0], [0, 1], [1, 1]] # (3, 2) b1 = [0, 0] W2 = [[1, 2], [3, 4]] # (2, 2) b2 = [0, 0] ``` 计算: 1. z1 = x @ W1 + b1 2. a1 = ReLU(z1) 3. z2 = a1 @ W2 + b2 4. y = softmax(z2) ## 练习2:验证 XOR 的多层解决方案 用两层感知机实现 XOR: - 隐藏层:h1 = AND(NOT x1, x2), h2 = AND(x1, NOT x2) - 输出层:y = OR(h1, h2) 验证对所有 4 种输入都正确。 ## 练习3:理解梯度消失 如果每层的 ReLU 输出有一半是 0(因为输入小于 0),那么 10 层之后: - 第 10 层的梯度大约是第 1 层的多少? - 这说明什么问题? ## 练习4:调参实验 修改 config.py 中的超参数,记录变化: - hidden_size: 64 vs 256 - learning_rate: 0.01 vs 1.0 - epochs: 20 vs 100 --- > **记住**:神经网络的核心是"前向传播 + 反向传播 + 梯度下降",通过不断调整权重来最小化损失!