🎯 目标:建立Python编程能力和数学理论基础,为后续深度学习和大模型学习打下根基。 📋 前置要求:无,零基础可开始


本阶段知识依赖图

flowchart TD
    A[Python基础] --> B[数学基础]
    B --> C[机器学习入门]
    A --> J["Jupyter Notebook<br/>贯穿始终的工具"]
    B --> B1[微积分] --> B1a[梯度下降]
    B --> B2[线性代数] --> B2a[矩阵运算]
    B --> B3[概率论] --> B3a[最大似然估计]
    C --> C1[线性回归]
    C --> C2[逻辑回归]
    C --> C3[正则化]

模块一:Python编程基础

为什么Python是AI工程师的第一语言?

Python之所以成为AI/ML领域的首选语言,核心原因有三:

  1. 语法简洁:接近伪代码,让你专注于算法逻辑而非语法细节
  2. 生态丰富:NumPy、Pandas、PyTorch、HuggingFace等核心库全部基于Python
  3. 社区庞大:遇到问题几乎都能找到解答

1.1 环境搭建

核心任务

安装Anaconda + VS Code,配置好开发环境。

什么是Anaconda?

Anaconda是一个Python发行版,它不仅仅是Python解释器,还包含:

  • conda:包管理器(类似pip,但能管理Python版本和虚拟环境)
  • 预装库:NumPy、Pandas、Matplotlib等常用科学计算库
  • Jupyter Notebook:交互式编程环境

什么是虚拟环境?为什么需要它?

虚拟环境是Python的"沙盒",每个项目可以有独立的依赖包版本:

生活类比:想象你有两间厨房。A厨房需要烤箱,B厨房需要空气炸锅。如果只有一间厨房,两个设备可能冲突(插座不够、位置不够)。虚拟环境就像给每个项目分配一间独立的厨房,互不干扰。

# 创建虚拟环境(创建一间新"厨房")
conda create -n llm_study python=3.11

# 激活虚拟环境(进入这间厨房)
conda activate llm_study

# 安装常用包(放入你需要的"厨具")
pip install numpy pandas matplotlib jupyter

为什么需要虚拟环境? 假设项目A需要PyTorch 2.0,项目B需要PyTorch 1.13,没有虚拟环境就会冲突——就像两个人同时要往一个水杯里倒不同品牌的果汁。

VS Code配置

  • 安装Python扩展(让VS Code"看得懂"Python代码)
  • 安装Jupyter扩展(在VS Code中直接运行Jupyter Notebook)
  • 配置conda环境作为解释器(告诉VS Code"用哪个虚拟环境")

1.2 Python基础语法

数据类型——AI中最常用的5种

类比理解:把AI模型想象成一个工厂,数据类型就是不同形状的原材料:

# 1. 整数和浮点数——模型参数、损失值、学习率都是数值
#    类比:工厂里的"数量"和"尺寸"——最基本的数据
learning_rate = 0.001        # 浮点数(小数)
epoch = 100                   # 整数

# 2. 字符串——文本数据的基本单位
#    类比:工厂里的"标签"和"说明书"
text = "大模型改变了世界"     # NLP中处理的就是字符串

# 3. 列表——最常用的数据结构,对应向量/张量的概念
#    类比:工厂里的"传送带"——有序排列的一组数据
weights = [0.1, 0.2, 0.3, 0.4]  # 类似一个一维张量(向量)
batch_data = [[1, 2], [3, 4], [5, 6]]  # 类似一个二维张量(矩阵)
#    ↑ 这就是深度学习中"一个batch的数据"的雏形

# 4. 字典——存储模型配置、JSON数据
#    类比:工厂里的"配置表"——用名字查找对应的值
model_config = {
    "hidden_size": 768,       # 隐藏层维度
    "num_layers": 12,         # Transformer层数
    "vocab_size": 30522       # 词表大小
}

# 5. 布尔值——控制流程
#    类比:工厂里的"开关"——控制程序走哪条路
is_training = True
if is_training:
    print("训练模式:启用Dropout")  # 训练时随机丢弃一些神经元
else:
    print("推理模式:关闭Dropout")  # 推理时所有神经元都要工作

控制流——程序的"大脑"

# for循环——遍历数据集的核心方式
# 类比:工厂的"流水线"——一个一个地处理数据
for batch in dataloader:       # 这就是训练循环的基本形式
    loss = model(batch)        # 模型处理这个batch
    loss.backward()            # 计算梯度(后面会详细讲)

# if-else——条件判断
# 类比:工厂的"分拣机"——根据条件走不同的分支
if loss < 0.01:
    print("模型已收敛")        # 损失足够小,可以停止训练了
elif loss < 0.1:
    print("继续训练")          # 还需要继续优化
else:
    print("模型可能有问题")    # 损失太大,可能需要检查数据或模型

# 列表推导式——Python特有的简洁写法,AI代码中非常常见
squares = [x**2 for x in range(10)]  # [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
# 一行代码 = 一个for循环 + 一个操作,是Python优雅的体现

1.3 函数与面向对象

函数——代码复用的基本单位

类比:函数就像一台"机器"——你给它输入,它给你输出。

# 函数在AI中的典型应用:定义损失函数
def cross_entropy_loss(predictions, targets):
    """
    交叉熵损失函数——分类任务中最常用的损失函数
    
    类比:这是一个"评分器"
    - predictions:模型的预测结果(模型认为答案是什么)
    - targets:真实标签(正确答案是什么)
    - 返回值:预测和正确答案之间的"差距"(越小越好)
    """
    loss = -sum(t * np.log(p) for t, p in zip(targets, predictions))
    return loss / len(targets)

# 参数默认值——配置超参数的常用方式
# 超参数 = 人为设定的参数(不是模型自己学到的)
def train_model(model, lr=0.001, epochs=10, batch_size=32):
    """
    lr: 学习率——每次更新参数的"步子大小"
    epochs: 训练轮数——把整个数据集看几遍
    batch_size: 批次大小——每次处理多少个样本
    """
    for epoch in range(epochs):
        # 训练逻辑
        pass

面向对象——理解PyTorch模型的基础

类比:如果函数是一台"机器",那类就是"工厂"——它包含多台机器(方法)和原材料(属性)。

# PyTorch中的模型就是一个类!
# 理解类是理解nn.Module的基础

class SimpleModel:
    def __init__(self, input_size, output_size):
        # __init__ 方法:初始化模型参数(类似工厂的"建厂"过程)
        # 这里定义工厂里有哪些"机器"和"原材料"
        self.weights = [0.1] * input_size  # 权重(模型要学习的参数)
        self.bias = 0                       # 偏置(另一个要学习的参数)
    
    def forward(self, x):
        # forward 方法:定义前向传播(数据如何流过工厂)
        # 这就是工厂的"生产流程"
        result = sum(w * xi for w, xi in zip(self.weights, x)) + self.bias
        return result

# 使用
model = SimpleModel(input_size=3, output_size=1)
output = model.forward([1.0, 2.0, 3.0])

关键理解:PyTorch中的nn.Module就是基于这个原理。你定义的每个神经网络都是一个类:

  • __init__中定义网络有哪些层(“工厂里有哪些机器”)
  • forward中定义数据如何流过这些层(“原材料如何经过每台机器”)
# PyTorch中的写法(对比上面的SimpleModel)
import torch.nn as nn

class SimpleModel(nn.Module):
    def __init__(self, input_size, output_size):
        super().__init__()  # 调用父类的初始化
        self.linear = nn.Linear(input_size, output_size)  # 定义一个线性层
    
    def forward(self, x):
        return self.linear(x)  # 数据流过线性层

1.4 Jupyter Notebook

为什么AI工程师离不开Jupyter?

类比:如果普通的Python脚本是"写一封信然后寄出去",那Jupyter Notebook就是"面对面聊天"——你可以看到对方(代码)的即时反应。

  1. 交互式编程:可以逐个cell运行代码,实时看到结果(不用等整个程序跑完)
  2. 可视化集成:图表直接嵌入在代码旁边(就像在笔记本上画图一样自然)
  3. 文档化:Markdown和代码混排,方便记录实验过程(“实验笔记"和"实验数据"放在一起)
  4. 快速原型:不需要创建完整项目文件,快速测试想法(“先试试这个想法行不行”)

核心操作

  • Shift+Enter:运行当前cell(执行当前这段代码)
  • Esc+A:在上方插入新cell(在上面加一段新代码或笔记)
  • Esc+B:在下方插入新cell
  • Esc+M:转为Markdown cell(写文字笔记)
  • Esc+DD:删除当前cell

1.5 文件操作与常用库

NumPy基础——一切数值计算的基石

为什么需要NumPy? Python原生的列表运算很慢。NumPy用C语言实现了底层运算,速度比纯Python快10-100倍。在深度学习中,我们每秒要进行数百万次矩阵运算,没有NumPy根本不可能。

import numpy as np

# 创建数组(张量的前身)
# 类比:向量 = 一条线上的点,矩阵 = 一个表格,3D张量 = 一摞表格
arr = np.array([1, 2, 3, 4, 5])          # 一维数组 ≈ 向量(1×5)
matrix = np.array([[1, 2], [3, 4]])       # 二维数组 ≈ 矩阵(2×2)
tensor_3d = np.ones((2, 3, 4))           # 三维数组 ≈ 3D张量(2×3×4)

# 矩阵运算——神经网络前向传播的核心
# 这是整个深度学习最基础的运算,务必理解!
A = np.array([[1, 2], [3, 4]])  # 2×2 矩阵
B = np.array([[5, 6], [7, 8]])  # 2×2 矩阵
C = A @ B          # 矩阵乘法,等价于np.dot(A, B)
# C[0][0] = 1*5 + 2*7 = 19
# C[0][1] = 1*6 + 2*8 = 22
# C[1][0] = 3*5 + 4*7 = 43
# C[1][1] = 3*6 + 4*8 = 50
# 结果:[[19, 22], [43, 50]]

# 这就是神经网络中 "output = input @ weight + bias" 的基本操作
# 每一层神经网络的本质就是:输入 × 权重矩阵 + 偏置

# 广播机制——理解这个,就理解了PyTorch张量运算的一大半
# 类比:你有3个苹果,你朋友也有3个苹果,你们把苹果合在一起
#       不需要写循环,NumPy自动帮你"一对一对"地加
x = np.array([[1, 2, 3]])       # shape: (1, 3)
bias = np.array([[10, 20, 30]]) # shape: (1, 3)
result = x + bias               # 自动广播,shape: (1, 3)
# result = [[11, 22, 33]]
# 广播规则:如果两个数组的shape不同,NumPy会自动"扩展"较小的数组

Pandas基础——数据处理的瑞士军刀

类比:如果NumPy是"计算器”,那Pandas就是"Excel"——它擅长处理表格数据(有列名、有索引的数据)。

import pandas as pd

# 读取CSV数据(数据集通常就是CSV格式,就像Excel表格)
df = pd.read_csv("data.csv")

# 基本操作
df.head()           # 查看前5行("先看一眼数据长什么样")
df.describe()       # 统计描述("数据的平均值、最大值、最小值是多少")
df['column']        # 选择列("只看这一列")
df[df['age'] > 18]  # 筛选("只要年龄大于18的")

模块二:微积分基础

为什么AI需要微积分?

一句话回答:神经网络的训练过程就是"求导+更新"的循环。

更详细的解释:想象你在一座山上,想要找到山谷的最低点(最优解)。你需要知道两件事:

  1. 哪个方向是下坡?(导数告诉你方向)
  2. 应该走多远?(学习率决定步幅)

当你听到"反向传播算法"时,本质就是链式法则求导——计算每个参数对最终误差的"贡献度"。当你听到"梯度下降"时,本质就是沿着导数方向更新参数——让模型的预测越来越准。

2.1 导数与求导法则

导数的直觉理解

核心类比:速度计

想象你开车:

  • 你的位置随时间变化(比如从0公里开到了100公里)
  • 速度就是位置对时间的导数——它告诉你"位置变化有多快"

同理,在神经网络中:

  • 损失函数随参数变化(参数调整后,误差会变大或变小)

  • 导数就是损失对参数的"变化率"——它告诉你"参数应该往哪个方向调,调多少"

  • 导数 > 0:函数在递增 → 参数应该减小(往左走,减少误差)

  • 导数 < 0:函数在递减 → 参数应该增大(往右走,减少误差)

  • 导数 = 0:到达极值点 → 可能是最优解(误差最小的地方)

更形象的类比:山谷寻宝

想象损失函数是一座山的地形图:

  • 山的高度 = 当前误差(越高误差越大)
  • 你的位置 = 当前参数值
  • 你的目标 = 找到山谷最低点(最小误差)

导数就像一个"坡度指示器":

  • 它告诉你脚下哪个方向最陡(梯度方向)
  • 它告诉你坡有多陡(梯度大小)

你每走一步都看一眼指示器,然后往最陡的下坡方向走——这就是梯度下降!

核心求导公式(必须记住)

  1. 常数求导:$(c)' = 0$ ——常数不变化,导数为0
  2. 幂函数:$(x^n)' = nx^{n-1}$ ——$x^2$ 的导数是 $2x$,$x^3$ 的导数是 $3x^2$
  3. 指数函数:$(e^x)' = e^x$ ——$e^x$ 是唯一"导数等于自身"的函数!
  4. 对数函数:$(\ln x)' = 1/x$ ——对数函数的导数是倒数
  5. 三角函数:$(\sin x)' = \cos x$ ——正弦的导数是余弦

为什么要记住这些? 因为神经网络的激活函数(Sigmoid、ReLU、Tanh)都是由这些基本函数组合而成的。记住基本公式,就能用链式法则推导出任何复杂函数的导数。

链式法则——反向传播的数学基础

核心公式:如果 $y = f(g(x))$,则 $\frac{dy}{dx} = f'(g(x)) \cdot g'(x)$

类比:工厂流水线的责任追溯

想象一个工厂流水线:原材料 $x$ → 工序1($g(x) = z$,加工成半成品) → 工序2($f(z) = y$,加工成成品) → 最终产品 $y$

现在最终产品有缺陷(损失L很大),你想知道是谁的责任:

  • 工序2的责任:$\frac{\partial L}{\partial z}$(成品的问题有多少是工序2造成的)
  • 工序1的责任:$\frac{\partial L}{\partial x} = \frac{\partial L}{\partial z} \times \frac{\partial z}{\partial x}$(把工序2的责任"传回"工序1)

这就是链式法则——把最终的误差责任一层一层往回追溯!

AI中的例子——单个神经元

网络结构:输入 $x$ → 线性变换 $z = wx + b$ → 激活 $a = \sigma(z)$ → 损失 $L$

求 $\frac{\partial L}{\partial w}$(用于更新 $w$):

  • 步骤1:L对a的导数(损失对激活值的敏感度):$\frac{\partial L}{\partial a} = 2(a - y)$(如果L是均方误差)
  • 步骤2:a对z的导数(激活函数的导数):$\frac{\partial a}{\partial z} = \sigma'(z) = a(1-a)$
  • 步骤3:z对w的导数(线性变换的导数):$\frac{\partial z}{\partial w} = x$

链式法则组合:

$$ \frac{\partial L}{\partial w} = \frac{\partial L}{\partial a} \cdot \frac{\partial a}{\partial z} \cdot \frac{\partial z}{\partial w} = 2(a-y) \cdot a(1-a) \cdot x $$

这就是反向传播的核心!每一步都很简单,但组合起来就能计算复杂网络的梯度。

扩展到多变量:当网络有多个参数时,每个参数都有自己的"责任链":

$$ \begin{aligned} \frac{\partial L}{\partial w_1} &= \frac{\partial L}{\partial z} \cdot \frac{\partial z}{\partial w_1} \\ \frac{\partial L}{\partial w_2} &= \frac{\partial L}{\partial z} \cdot \frac{\partial z}{\partial w_2} \\ \frac{\partial L}{\partial b} &= \frac{\partial L}{\partial z} \cdot \frac{\partial z}{\partial b} \end{aligned} $$
  • $w_1$ 的梯度:$\frac{\partial L}{\partial w_1}$
  • $w_2$ 的梯度:$\frac{\partial L}{\partial w_2}$
  • $b$ 的梯度:$\frac{\partial L}{\partial b}$

所有参数同时更新,这就是梯度下降的并行性。

2.2 激活函数求导

Sigmoid函数及其导数——完整推导

什么是Sigmoid? 它把任意实数"压缩"到0和1之间,就像一个"开关"——输入很大时输出接近1(开),输入很小时输出接近0(关)。

$$ \sigma(x) = \frac{1}{1 + e^{-x}} $$

Sigmoid导数的完整推导(这是面试常考题):

已知:$\sigma(x) = \frac{1}{1 + e^{-x}}$

设 $u = 1 + e^{-x}$,则 $\sigma(x) = \frac{1}{u} = u^{-1}$

根据链式法则:

$$ \begin{aligned} \sigma'(x) &= \frac{d}{dx} [u^{-1}] \\ &= -u^{-2} \cdot \frac{du}{dx} \\ &= -\frac{1}{u^2} \cdot \frac{d}{dx}[1 + e^{-x}] \\ &= -\frac{1}{u^2} \cdot (-e^{-x}) \\ &= \frac{e^{-x}}{(1 + e^{-x})^2} \end{aligned} $$

现在我们把它写成用 $\sigma(x)$ 表示的形式:

$$ \begin{aligned} \sigma'(x) &= \frac{e^{-x}}{(1 + e^{-x})^2} \\ &= \frac{1 + e^{-x} - 1}{(1 + e^{-x})^2} \quad \text{(把分子拆开)} \\ &= \frac{1 + e^{-x}}{(1 + e^{-x})^2} - \frac{1}{(1 + e^{-x})^2} \\ &= \frac{1}{1 + e^{-x}} - \frac{1}{(1 + e^{-x})^2} \\ &= \sigma(x) - \sigma(x)^2 \\ &= \sigma(x) \cdot (1 - \sigma(x)) \end{aligned} $$

最终结果:

$$ \sigma'(x) = \sigma(x) \cdot (1 - \sigma(x)) $$

这个结果为什么重要?

  • 导数可以直接用函数值计算——不需要重新算指数函数!
  • 只要知道 $\sigma(x)$ 的值,就能直接得到导数
  • 这在反向传播中节省了大量计算

Sigmoid的问题——梯度消失

  • 当 $x$ 很大时:$\sigma(x) \approx 1$,$\sigma'(x) = 1 \cdot (1-1) = 0$ → 梯度几乎为0
  • 当 $x$ 很小时:$\sigma(x) \approx 0$,$\sigma'(x) = 0 \cdot (1-0) = 0$ → 梯度几乎为0

这意味着:当输入值太大或太小时,Sigmoid的梯度接近0,参数几乎不更新,网络"学不动了"。这就是"梯度消失"问题,是深度学习早期的一大难题,后来ReLU激活函数的出现解决了这个问题。

Softmax函数

什么是Softmax? 它把一组任意实数转换为概率分布(所有值>0,且和为1)。

类比:想象一个班级的考试成绩是[85, 90, 75],Softmax把它们转换为"每个学生成绩的相对概率"——成绩越高的学生,概率越大。

Softmax 公式:$\text{softmax}(x_i) = e^{x_i} / \sum e^{x_j}$

示例,输入:[2.0, 1.0, 0.1]

$$ \begin{aligned} e^{2.0} &= 7.39, \quad e^{1.0} = 2.72, \quad e^{0.1} = 1.11 \\ \text{总和} &= 7.39 + 2.72 + 1.11 = 11.22 \\ \text{Softmax} &= [7.39/11.22, \; 2.72/11.22, \; 1.11/11.22] \\ &= [0.659, \; 0.242, \; 0.099] \\ &\approx [65.9\%, \; 24.2\%, \; 9.9\%] \end{aligned} $$

三个概率之和 = 100%

为什么要用 $e^x$ 而不是直接用 $x$?

  1. $e^x > 0$ 保证所有输出都是正数(概率不能为负)
  2. $e^x$ 是单调递增的(输入越大,输出越大——保持大小关系)
  3. $e^x$ 的导数是它自己(计算方便)

数值稳定性问题:如果输入值很大,比如 [1000, 1001, 1002],$e^{1000}$ 是一个巨大的数,会溢出!

解决方案:减去最大值。例如 $\text{softmax}([1000, 1001, 1002]) = \text{softmax}([-2, -1, 0])$,结果完全一样但不会溢出。PyTorch和所有深度学习框架都自动做了这个优化。

应用场景:分类任务的最后一层,输出每个类别的概率。

2.3 线性代数

向量——数据的基本表示

类比:向量就像一个"坐标"——它用一组数字来描述一个点的位置。

在AI中,一个数据样本通常表示为一个向量:

  • 一张 $28 \times 28$ 的灰度图片 → 展平为784维向量(每个像素是一个维度)
  • 一个句子的词嵌入 → 每个词是一个300维向量(用300个数字描述一个词的"含义")

为什么用向量?因为计算机只能处理数字。把现实世界的东西(图片、文字、声音)转换为向量,就是"表示学习"的核心任务。

矩阵乘法——神经网络的核心运算

矩阵乘法到底在做什么?

直觉1:矩阵是一组"变换规则"

想象你有一个2D图形(比如一个三角形)。矩阵乘法就是对这个图形做"变换"——旋转、缩放、剪切。

不同的矩阵 = 不同的变换。神经网络的权重矩阵 = 一种"语义变换"——把输入空间映射到输出空间。

直觉2:矩阵乘法是"批量线性组合"

$$ \begin{bmatrix} \text{output}_1 \\ \text{output}_2 \end{bmatrix} = \begin{bmatrix} w_{11} & w_{12} & w_{13} \\ w_{21} & w_{22} & w_{23} \end{bmatrix} \begin{bmatrix} \text{input}_1 \\ \text{input}_2 \\ \text{input}_3 \end{bmatrix} = \begin{bmatrix} w_{11}\cdot\text{input}_1 + w_{12}\cdot\text{input}_2 + w_{13}\cdot\text{input}_3 \\ w_{21}\cdot\text{input}_1 + w_{22}\cdot\text{input}_2 + w_{23}\cdot\text{input}_3 \end{bmatrix} $$

每个输出 = 所有输入的加权求和。权重矩阵中的每个元素 $w_{ij}$ 表示"第j个输入对第i个输出的影响程度"。这就是神经网络前向传播的本质!

实际例子:神经网络的前向传播本质上就是一连串的矩阵乘法:

  • 输入 $x$(1×784)× 权重 $W$(784×512)+ 偏置 $b$(1×512)= 隐藏层 $h$(1×512)
  • 隐藏层 $h$(1×512)× 权重 $W$(512×10)+ 偏置 $b$(1×10)= 输出 $y$(1×10)

784维输入 → 512维隐藏层 → 10维输出(10个类别的概率)。每一层都在做:$\text{output} = \text{input} \cdot W + b$

为什么GPU擅长深度学习?

矩阵乘法的本质是"大量独立的乘加运算":$C[i][j] = \sum A[i][k] \cdot B[k][j]$

这些乘加运算是完全独立的——C[0][0]和C[1][1]的计算互不影响。

  • CPU:少量核心(比如8个),每个核心很强,但只能同时算8个
  • GPU:大量核心(比如几千个),每个核心较弱,但能同时算几千个

矩阵乘法正好适合GPU的"人海战术"!这就是为什么深度学习离不开GPU。

特征值与特征向量——理解PCA降维

什么是特征值和特征向量?

类比:镜子的方向

想象你拿着一面镜子(矩阵A),向它照一束光(向量v):

  • 大多数方向的光:照进去后方向会改变
  • 特殊方向的光:照进去后方向不变,只是被拉伸或压缩

这个"特殊方向"就是特征向量,拉伸/压缩的比例就是特征值。

数学定义:对于矩阵A,如果 $A\mathbf{v} = \lambda\mathbf{v}$,则:

  • $\mathbf{v}$ 是特征向量(方向不变的"特殊方向")
  • $\lambda$ 是特征值(拉伸/压缩的比例)

示例:

$$ A = \begin{bmatrix} 2 & 1 \\ 1 & 2 \end{bmatrix} $$
  • $\mathbf{v} = \begin{bmatrix} 1 \\ 1 \end{bmatrix}$ 是一个特征向量,因为 $A\mathbf{v} = \begin{bmatrix} 3 \\ 3 \end{bmatrix} = 3\mathbf{v}$,特征值 $\lambda = 3$
  • $\mathbf{v} = \begin{bmatrix} 1 \\ -1 \end{bmatrix}$ 是另一个特征向量,因为 $A\mathbf{v} = \begin{bmatrix} 1 \\ -1 \end{bmatrix} = 1\mathbf{v}$,特征值 $\lambda = 1$

为什么特征值/特征向量在AI中重要?

PCA降维的原理:

  1. 计算数据的协方差矩阵
  2. 求协方差矩阵的特征值和特征向量
  3. 特征值大的特征向量 = 数据变化最大的方向(“最重要的方向”)
  4. 只保留前k个最重要的方向,丢弃其他方向 → 数据从高维降到k维,同时保留了最多的信息

类比:你有一张3D的照片,PCA告诉你"最有信息量的拍摄角度"是什么。只从这个角度拍一张2D照片,虽然维度降低了,但信息损失最少。

SVD奇异值分解——矩阵的"DNA"

任意矩阵 A 可以分解为三个矩阵的乘积:$A = U \cdot \Sigma \cdot V^T$

其中:

  • $U$:左奇异矩阵(“行空间的特征方向”)
  • $\Sigma$:奇异值矩阵(对角矩阵,每个对角元素是一个奇异值)
  • $V$:右奇异矩阵(“列空间的特征方向”)

SVD的直觉——信息压缩

类比:一张高清照片有1000×1000像素(100万个数字)。SVD告诉你:这100万个数字中,真正"重要"的信息可能只需要100个奇异值就能表达。

保留最大的k个奇异值,丢弃其他的:

  • 用更少的数据表示原始信息(压缩)
  • 保留了最重要的"模式"(去噪)

这就是为什么SVD被称为矩阵的"DNA"——它揭示了矩阵中最本质的结构。

SVD在AI中的应用

  • 数据压缩:保留最大的k个奇异值,用更少的数据表示原始信息
  • 推荐系统:协同过滤的核心算法(Netflix推荐电影用的就是SVD)
  • PCA实现:SVD是PCA的数值稳定实现方式
  • 词嵌入:早期的LSA(潜在语义分析)就是用SVD做的

2.4 概率论基础

条件概率与贝叶斯定理

条件概率:在已知B发生的条件下,A发生的概率

$$ P(A|B) = \frac{P(A \cap B)}{P(B)} $$

类比:在一个班里,$P(\text{及格}) = \text{及格人数}/\text{总人数}$。$P(\text{及格}|\text{男同学}) = \text{及格的男同学}/\text{所有男同学}$。条件概率就是"缩小范围后的概率"。

贝叶斯定理——从结果推原因

$$ P(A|B) = \frac{P(B|A) \cdot P(A)}{P(B)} $$

类比:你看到地上有水(结果B),想判断是"下雨了"还是"洒水车经过"(原因A)

  • $P(\text{下雨}|\text{有水}) = P(\text{有水}|\text{下雨}) \cdot P(\text{下雨}) / P(\text{有水})$
  • 如果今天是晴天,$P(\text{下雨})$ 很小,所以"下雨"的可能性不大
  • 如果洒水车经常经过,$P(\text{洒水车})$ 很大,所以"洒水车"的可能性更大

贝叶斯定理 = 用"先验知识"($P(A)$)+ “新证据”($P(B|A)$)更新你的判断($P(A|B)$)

最大似然估计(MLE)——损失函数的理论基础

核心思想:选择让观测数据出现概率最大的参数。

类比:侦探推理

场景:一个硬币被抛了10次,结果是7次正面、3次反面。问题:这枚硬币正面朝上的概率p是多少?

最大似然估计的回答:选择让"7正3反"这个结果最可能出现的p值。

似然函数:$L(p) = C(10,7) \cdot p^7 \cdot (1-p)^3$

我们要求让 $L(p)$ 最大的 $p$。

取对数(方便计算):$\log L = 7\log(p) + 3\log(1-p) + \text{常数}$

对 $p$ 求导并令其为 0:$\frac{7}{p} - \frac{3}{1-p} = 0$,解得 $p = \frac{7}{10} = 0.7$

结论:最大似然估计认为 $p = 0.7$,也就是正面概率是70%。这很直觉——10次中7次正面,所以概率大概是70%。

MLE的完整数学推导(一维高斯分布)

假设数据服从正态分布 $N(\mu, \sigma^2)$,观测到 $n$ 个数据点:$x_1, x_2, \ldots, x_n$

似然函数(所有数据点概率的乘积):

$$ L(\mu, \sigma^2) = \prod_i P(x_i | \mu, \sigma^2) = \prod_i \frac{1}{\sqrt{2\pi\sigma^2}} \cdot \exp\left(-\frac{(x_i-\mu)^2}{2\sigma^2}\right) $$

取对数(把乘法变加法,方便求导):

$$ \log L = \sum_i \left[ -\frac{1}{2}\log(2\pi\sigma^2) - \frac{(x_i-\mu)^2}{2\sigma^2} \right] $$

对 $\mu$ 求导并令其为0:

$$ \frac{\partial \log L}{\partial \mu} = \sum_i \frac{x_i - \mu}{\sigma^2} = 0 \implies \sum_i x_i - n\mu = 0 \implies \mu^* = \frac{1}{n}\sum_i x_i = \text{样本均值} $$

结论:高斯分布的MLE估计 = 样本均值。这很直觉——“最可能的中心位置"就是所有数据点的平均值。

关键联系——为什么交叉熵损失函数是合理的

交叉熵损失 = 负对数似然。当你最小化交叉熵时,你实际上在最大化似然——让模型的参数使得观测到的训练数据出现的概率最大。

这就是为什么交叉熵是分类任务的标准损失函数——它有坚实的概率论基础。


模块三:概率论与最优化

3.1 概率论进阶

期望与方差

期望——数据的"重心”

$$ E[X] = \sum x \cdot P(x) $$

类比:期望就像一根尺子上挂了一堆不同重量的砝码。砝码的位置 = $x$(取值),砝码的重量 = $P(x)$(概率)。期望 = 这根尺子平衡时的支点位置(重心)。

示例:骰子的期望 $= 1 \times \frac{1}{6} + 2 \times \frac{1}{6} + \cdots + 6 \times \frac{1}{6} = 3.5$。虽然骰子不会掷出3.5,但这是"平均值"的理论位置。

方差——数据的"分散程度"

$$ \text{Var}(X) = E[(X-\mu)^2] = E[X^2] - (E[X])^2 $$

类比:两组学生的平均分都是80分

  • A组:[79, 80, 81] → 方差很小(成绩很集中)
  • B组:[50, 80, 110] → 方差很大(成绩很分散)
  • 虽然平均分一样,但方差告诉我们"波动有多大"

在AI中的应用

  • BatchNorm:对每个mini-batch计算均值和方差,然后标准化数据
    • 为什么?因为如果每层的输入分布都在变化(“内部协变量偏移”),网络很难学习
    • BatchNorm让每层的输入分布稳定下来,加速训练
  • 权重初始化:初始化权重时需要控制方差
    • 方差太大 → 信号爆炸(输出值越来越大)
    • 方差太小 → 信号消失(输出值越来越小)
    • 合适的方差 → 信号稳定传播

协方差——两个变量的"关系"

$$ \text{Cov}(X,Y) = E[(X-\mu_x)(Y-\mu_y)] $$

协方差衡量两个变量的变化趋势:

  • 正值:X增大时Y也增大(正相关,比如身高和体重)
  • 负值:X增大时Y减小(负相关,比如温度和羽绒服销量)
  • 零:X和Y没有线性关系(不一定独立!)

3.2 梯度下降——最重要的优化算法

梯度的直觉

类比:蒙眼下山

想象你被蒙上眼睛,站在一座山上,想要找到山谷的最低点:

  • 你唯一能做的就是用脚感受脚下的坡度
  • 每一步都往最陡的下坡方向
  • 走的步幅由学习率控制

梯度就是那个"最陡的方向":

  • 在一维中:梯度 = 导数(一个数字)
  • 在多维中:梯度 = 所有偏导数组成的向量(指向最陡上升方向)

梯度方向:函数值增长最快的方向(上坡方向)。梯度下降:沿着梯度的反方向走(下坡方向)。

$$ \theta_{\text{new}} = \theta_{\text{old}} - \eta \cdot \nabla L(\theta) $$

其中:

  • $\theta$:模型参数(你所在的位置)
  • $\eta$:学习率(步幅大小)
  • $\nabla L(\theta)$:损失函数对参数的梯度(坡度指示器)
  • 减号:因为我们想减少损失,所以往梯度的反方向走

学习率——步幅的艺术

  • 学习率太大 → 步子太大,可能跨过最低点,来回震荡。就像下山时跑得太快,冲过了山谷,跑到对面山上去了。
  • 学习率太小 → 步子太小,收敛极慢。就像下山时每次只挪一厘米,要走到天黑。
  • 学习率刚好 → 稳定收敛到最优解。就像以合适的速度下山,稳步到达谷底。

常见策略:

  • 从0.001开始(一个比较安全的起点)
  • 观察loss曲线:如果震荡就减小,如果下降太慢就增大
  • 高级策略:学习率调度(先大后小,就像下山时先大步走,接近谷底时小步挪)

三种梯度下降方法——详细对比

方法描述类比优缺点
1. 批量梯度下降(BGD)每次用全部数据计算梯度你有一亿个数据点,每走一步都要看一亿个"坡度指示器"取平均优点:梯度方向最准确,一定能收敛到局部最优;缺点:数据量大时极慢
2. 随机梯度下降(SGD)每次只用1个样本计算梯度你只看一个"坡度指示器"就决定方向优点:快;缺点:方向不稳定
3. 小批量梯度下降(MBGD)每次用一小批数据(如32/64/128个样本)计算梯度你听32个人的建议取平均,既不太慢也不太偏优点:兼顾速度和稳定性;实际训练中最常用!

对线性回归应用梯度下降——完整推导

线性回归模型:$\hat{y} = wx + b$

损失函数(均方误差):$L = \frac{1}{n} \sum_i (y_i - \hat{y}_i)^2 = \frac{1}{n} \sum_i (y_i - wx_i - b)^2$

目标:找到 $w$ 和 $b$,使 $L$ 最小。

对 $w$ 求偏导:

$$ \begin{aligned} \frac{\partial L}{\partial w} &= \frac{1}{n} \sum_i 2(y_i - wx_i - b)(-x_i) \\ &= -\frac{2}{n} \sum_i x_i(y_i - wx_i - b) \end{aligned} $$

对 $b$ 求偏导:

$$ \begin{aligned} \frac{\partial L}{\partial b} &= \frac{1}{n} \sum_i 2(y_i - wx_i - b)(-1) \\ &= -\frac{2}{n} \sum_i (y_i - wx_i - b) \end{aligned} $$

更新规则:

$$ \begin{aligned} w_{\text{new}} &= w_{\text{old}} - \eta \cdot \frac{\partial L}{\partial w} \\ b_{\text{new}} &= b_{\text{old}} - \eta \cdot \frac{\partial L}{\partial b} \end{aligned} $$

重复这个过程直到收敛(损失不再明显下降)。

3.3 凸优化

什么是凸函数?

类比:碗和碗碎片

  • 凸函数像一个碗——你把弹珠放进去,它总会滚到最低点。不管弹珠从哪个位置开始,最终都会到达同一个最低点(全局最优)。
  • 非凸函数像一堆碗碎片——弹珠可能卡在某个凹陷处(局部最优),而不是到达真正的最低点(全局最优)。

凸优化重要因为:凸函数的局部最小值 = 全局最小值 → 梯度下降一定能找到最优解,不用担心"卡在局部最优"。

凸函数的数学定义: 对于任意两点 $x_1, x_2$ 和任意 $t \in [0,1]$:

$$ f(t \cdot x_1 + (1-t) \cdot x_2) \leq t \cdot f(x_1) + (1-t) \cdot f(x_2) $$

直觉:函数图像上任意两点的连线,都在函数图像上方(或重合)。

判断方法:

  • 一维:二阶导数 $f''(x) \geq 0$(开口向上)
  • 多维:Hessian矩阵半正定

为什么凸优化在AI中重要?

  • 线性回归的损失函数(均方误差)是凸的 → 有唯一最优解
  • 逻辑回归的损失函数(交叉熵)是凸的 → 有唯一最优解
  • 神经网络的损失函数是非凸的 → 只能找到局部最优

好消息:实践中,深度学习的非凸损失函数的局部最优通常也足够好。坏消息:理论上不能保证找到全局最优。实际做法:用SGD + 各种优化技巧(动量、Adam等),通常能找到好的解。

拉格朗日乘子法与KKT条件

问题:如何在有约束的情况下求最优解?

类比:在围栏内找最低点

  • 无约束优化:在整个山上找最低点(随便走)
  • 有约束优化:只能在围栏内找最低点(不能走出围栏)

拉格朗日乘子法的核心思想:把"有约束"问题转换为"无约束"问题。在目标函数上加一个"惩罚项"——如果你违反约束,惩罚项就会变大。

数学形式:

  • 原始问题:minimize $f(x)$,subject to $g(x) = 0$
  • 构造拉格朗日函数:$\mathcal{L}(x, \lambda) = f(x) + \lambda \cdot g(x)$
    • $f(x)$:我们要最小化的目标
    • $\lambda \cdot g(x)$:约束的"惩罚"
    • $\lambda$(拉格朗日乘子):惩罚的"力度"

分别对 $x$ 和 $\lambda$ 求导并令其为0:

$$ \frac{\partial \mathcal{L}}{\partial x} = 0 \quad \text{(最优解的条件)} \qquad \frac{\partial \mathcal{L}}{\partial \lambda} = 0 \quad \text{(约束条件,即 } g(x) = 0 \text{)} $$

在AI中的应用:SVM的对偶问题就是用拉格朗日乘子法推导的。


模块四:机器学习基础

4.1 线性回归

线性回归的直觉

类比:在散点图上画一条"最好"的线

给你一堆数据点(比如房屋面积和房价),你想找到一条直线来描述它们的关系:

$$ y = wx + b $$

目标:找到 $w$(斜率)和 $b$(截距),使得这条线"最贴近"所有数据点。“最贴近” = 预测值和真实值的差距最小。

正规方程——直接求解

思路:既然我们想要"最小化误差",那就直接对误差函数求导,令导数为0,解方程。

矩阵形式:$y = Xw + e$($e$ 是误差)。损失函数:$L = \|e\|^2 = \|y - Xw\|^2$

展开:

$$ L = (y - Xw)^T (y - Xw) = y^Ty - y^TXw - w^TX^Ty + w^TX^TXw = y^Ty - 2w^TX^Ty + w^TX^TXw $$

对 $w$ 求导并令其为0:

$$ \frac{\partial L}{\partial w} = -2X^Ty + 2X^TXw = 0 \implies X^TXw = X^Ty \implies w = (X^TX)^{-1} \cdot X^Ty $$

这就是正规方程!直接一步算出最优解。

优点:一步到位,不需要迭代

缺点

  • 需要计算矩阵逆 $(X^TX)^{-1}$,当特征维度很高时(如10000+),计算量巨大($O(n^3)$)
  • 当 $X^TX$ 不可逆时(比如特征之间有线性关系),无法求解

什么时候用正规方程,什么时候用梯度下降?

  • 特征数 < 10000:正规方程通常更快
  • 特征数 > 10000:梯度下降通常更合适
  • 数据量极大:梯度下降(可以每次只用一小批数据)

最小二乘法

$$ L(w,b) = \frac{1}{n} \sum (y_i - wx_i - b)^2 $$

“最小二乘"的名字来源:最小化"误差的平方和”。“二乘” = 平方(误差的平方),“最小” = 让这个平方和最小。

对L求导令其为0,就是正规方程。几何意义:找到一条线,使所有数据点到这条线的"垂直距离的平方和"最小。

4.2 正则化

为什么需要正则化?

类比:考试作弊 vs 真正学会

过拟合就像"死记硬背":

  • 训练数据(练习题)上表现很好——因为把答案都背下来了
  • 测试数据(考试题)上表现很差——因为没见过的题就不会了

正则化就像"惩罚死记硬背":

  • 不仅要求模型在训练数据上表现好
  • 还要求模型"尽量简单"(不要死记硬背)
  • 简单的模型泛化能力更好(理解了原理就能做新题)

L1正则化(Lasso)

L1正则化的损失函数:$L_{\text{total}} = L_{\text{original}} + \lambda \cdot \sum |w_i|$

L1的效果:使部分权重变为恰好0,实现特征选择。

为什么L1能使权重变为0?——几何直觉

想象一个2D空间($w_1, w_2$):

  • 损失函数的等高线是椭圆形的
  • L1的约束区域是菱形($|w_1| + |w_2| \leq c$)

椭圆和菱形最容易在"角点"相交。角点恰好在坐标轴上($w_1 = 0$ 或 $w_2 = 0$)→ 最优解中某些权重恰好为0 → 特征选择。

类比:L1像一把"剪刀",会直接把不重要的特征"剪掉"(权重变为0)。

L2正则化(Ridge)

L2正则化的损失函数:$L_{\text{total}} = L_{\text{original}} + \lambda \cdot \sum w_i^2$

L2的效果:使所有权重趋近于0但不等于0,防止某个特征权重过大。

为什么L2不会使权重变为0?——几何直觉

想象一个2D空间($w_1, w_2$):

  • 损失函数的等高线是椭圆形的
  • L2的约束区域是圆形($w_1^2 + w_2^2 \leq c$)

椭圆和圆形的交点通常不在坐标轴上 → 权重趋近于0但不会恰好为0。

类比:L2像一个"压缩弹簧",把所有权重往小的方向压,但不会压到0。

L1 vs L2 对比

特性L1 (Lasso)L2 (Ridge)
效果特征选择(稀疏解)权重衰减(平滑解)
权重会变为0?会(部分权重=0)不会(权重趋近0但不等于0)
适用场景特征很多,需要筛选特征都有用,防止过拟合
几何形状菱形约束圆形约束
导数sign(w)(在0处不可导)2w(处处可导,优化更方便)

正则化的本质——奥卡姆剃刀

不加正则化:模型只追求"拟合训练数据"(最小化训练误差)

加正则化:模型同时追求"拟合训练数据" + “保持简单”(最小化训练误差 + 惩罚复杂度)

这就是奥卡姆剃刀原则:在同样能解释数据的模型中,选择最简单的那个。

为什么简单更好?

  • 简单的模型更不容易"死记硬背"训练数据
  • 简单的模型对新数据的泛化能力更强
  • 简单的模型更容易理解和调试

4.3 逻辑回归

逻辑回归不是回归,是分类!

名字的由来:逻辑回归使用了"逻辑函数"(即Sigmoid函数),所以叫"逻辑回归"。但它实际上是一个分类算法,不是回归算法。

  • 线性回归:输出连续值(如房价=150万),$y = wx + b$,输出范围是 $(-\infty, +\infty)$
  • 逻辑回归:输出概率(如垃圾邮件概率=0.95),$y = \sigma(wx + b)$,输出范围是 $(0, 1)$,表示概率

核心区别:在输出层加了一个Sigmoid函数。

Sigmoid函数——将实数映射为概率

$$ \sigma(z) = \frac{1}{1 + e^{-z}} $$$$ \sigma(0) = 0.5, \quad \sigma(5) \approx 0.993, \quad \sigma(-5) \approx 0.007 $$

分类规则:$\sigma(z) \geq 0.5$ → 预测为正类(类别1);$\sigma(z) < 0.5$ → 预测为负类(类别0)

交叉熵损失函数——完整推导

为什么用交叉熵而不是均方误差?

这个问题在面试中经常被问到,让我们从头推导。

推导起点:最大似然估计

假设:

  • 样本i的真实标签:$y_i \in \{0, 1\}$
  • 模型预测样本i为正类的概率:$p_i = \sigma(w \cdot x_i + b)$

对于单个样本 $i$:

  • 若 $y_i = 1$:$P(y_i | x_i) = p_i$(预测为正类的概率)
  • 若 $y_i = 0$:$P(y_i | x_i) = 1 - p_i$(预测为负类的概率)

合并为一个公式:

$$ P(y_i | x_i) = p_i^{y_i} \cdot (1 - p_i)^{(1 - y_i)} $$

(当 $y_i = 1$ 时就是 $p_i$,当 $y_i = 0$ 时就是 $1 - p_i$)

整个数据集的似然:

$$ L = \prod_i P(y_i | x_i) = \prod_i \left[ p_i^{y_i} \cdot (1 - p_i)^{(1 - y_i)} \right] $$

取对数(连乘变连加,方便计算):

$$ \log L = \sum_i \left[ y_i \cdot \log(p_i) + (1 - y_i) \cdot \log(1 - p_i) \right] $$

最大似然 = 最大化 $\log L$;最小化负对数似然 = 最小化 $-\log L$

交叉熵损失:

$$ L_{CE} = -\frac{1}{n} \sum_i \left[ y_i \cdot \log(p_i) + (1 - y_i) \cdot \log(1 - p_i) \right] $$

这就是交叉熵损失函数的由来——它不是凭空设计的,而是从最大似然估计自然推导出来的!

为什么不用均方误差(MSE)?

MSE损失:$L_{MSE} = \frac{1}{n} \sum_i (y_i - p_i)^2$

问题:当Sigmoid的输入 $z$ 很大或很小时,Sigmoid的梯度接近0。MSE的梯度中包含 $\sigma'(z)$,所以MSE的梯度也会接近0 → 参数更新极慢 → 训练不动!

交叉熵的梯度:$\frac{\partial L_{CE}}{\partial w} = \frac{1}{n} \sum_i (p_i - y_i) \cdot x_i$

这个梯度中不包含 $\sigma'(z)$!→ 梯度大小只和预测误差 $(p_i - y_i)$ 成正比 → 预测越差,梯度越大,更新越快 → 训练更高效!

这就是为什么分类任务用交叉熵而不是MSE——交叉熵的梯度形式更简洁,避免了Sigmoid梯度消失的问题。

Softmax回归(多分类)

  • 二分类:Sigmoid → 输出1个概率 $p$,另一个类别的概率是 $1-p$
  • 多分类:Softmax → 输出 $k$ 个概率($k$ 为类别数),且概率之和为1
$$ \text{softmax}(z_i) = \frac{e^{z_i}}{\sum_j e^{z_j}} $$

示例:3个类别(猫、狗、鸟),模型输出 [2.0, 1.0, 0.1]。softmax = [0.659, 0.242, 0.099] → 65.9%概率是猫,24.2%概率是狗,9.9%概率是鸟。

Softmax回归的损失函数也是交叉熵,只是推广到多分类:

$$ L = -\frac{1}{n} \sum_i \sum_k y_{ik} \cdot \log(p_{ik}) $$

其中 $y_{ik}$ 是one-hot编码(只有正确类别的位置是1,其余是0)。

4.4 阶段总结

本阶段核心公式速查表

#名称公式
1梯度下降$\theta = \theta - \eta \cdot \nabla L$
2线性回归$y = Xw + b$
3正规方程$w = (X^T X)^{-1} \cdot X^T y$
4Sigmoid$\sigma(z) = \frac{1}{1 + e^{-z}}$
5Sigmoid 导数$\sigma'(z) = \sigma(z) \cdot (1 - \sigma(z))$
6交叉熵$L = -[y \cdot \log(p) + (1-y) \cdot \log(1-p)]$
7Softmax$\text{softmax}(z_i) = e^{z_i} / \sum e^{z_j}$
8L2 正则化$L_{\text{total}} = L + \lambda \cdot \sum w^2$
9贝叶斯$P(A
10最大似然$\theta^* = \arg\max \sum \log P(x_i
11链式法则$\frac{dy}{dx} = \frac{dy}{du} \cdot \frac{du}{dx}$
12SVD$A = U \cdot \Sigma \cdot V^T$

📚 推荐补充资源

知识点推荐资源说明
Python《Python编程:从入门到实践》零基础友好
NumPyNumPy官方Quickstart快速上手
微积分3Blue1Brown《微积分的本质》可视化理解,B站有中文字幕
线性代数3Blue1Brown《线性代数的本质》神级教程,必看
概率论可汗学院概率论课程零基础友好
机器学习吴恩达《机器学习》课程经典入门课程