Python打卡训练营Day41

article/2025/8/19 16:38:46

DAY 41 简单CNN

知识回顾

  1. 数据增强
  2. 卷积神经网络定义的写法
  3. batch归一化:调整一个批次的分布,常用与图像数据
  4. 特征图:只有卷积操作输出的才叫特征图
  5. 调度器:直接修改基础学习率

卷积操作常见流程如下:

1. 输入 → 卷积层 → Batch归一化层(可选) → 池化层 → 激活函数 → 下一层

  1. Flatten -> Dense (with Dropout,可选) -> Dense (Output)

这里相关的概念比较多,如果之前没有学习过计算机视觉部分,请自行上网检索视频了解下基础概念,也可以对照讲义学习下。

计算机视觉入门

作业:尝试手动修改下不同的调度器和CNN的结构,观察训练的差异。

import torch
import torch.nn as nn
import torch.optim as optim
from torchvision import datasets, transforms
from torch.utils.data import DataLoader
import matplotlib.pyplot as plt
import numpy as np# 设置中文字体支持
plt.rcParams["font.family"] = ["SimHei"]
plt.rcParams['axes.unicode_minus'] = False  # 解决负号显示问题# 检查GPU是否可用
device = torch.device("cuda" if torch.cuda.is_available() else "cpu")
print(f"使用设备: {device}")# 1. 数据预处理
# 训练集:使用多种数据增强方法提高模型泛化能力
train_transform = transforms.Compose([# 随机裁剪图像,从原图中随机截取32x32大小的区域transforms.RandomCrop(32, padding=4),# 随机水平翻转图像(概率0.5)transforms.RandomHorizontalFlip(),# 随机颜色抖动:亮度、对比度、饱和度和色调随机变化transforms.ColorJitter(brightness=0.2, contrast=0.2, saturation=0.2, hue=0.1),# 随机旋转图像(最大角度15度)transforms.RandomRotation(15),# 将PIL图像或numpy数组转换为张量transforms.ToTensor(),# 标准化处理:每个通道的均值和标准差,使数据分布更合理transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010))
])# 测试集:仅进行必要的标准化,保持数据原始特性,标准化不损失数据信息,可还原
test_transform = transforms.Compose([transforms.ToTensor(),transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010))
])# 2. 加载CIFAR-10数据集
train_dataset = datasets.CIFAR10(root='./data',train=True,download=True,transform=train_transform  # 使用增强后的预处理
)test_dataset = datasets.CIFAR10(root='./data',train=False,transform=test_transform  # 测试集不使用增强
)# 3. 创建数据加载器
batch_size = 64
train_loader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True)
test_loader = DataLoader(test_dataset, batch_size=batch_size, shuffle=False)
# 4. 定义CNN模型的定义(替代原MLP)
class CNN(nn.Module):def __init__(self):super(CNN, self).__init__()  # 继承父类初始化# ---------------------- 第一个卷积块 ----------------------# 卷积层1:输入3通道(RGB),输出32个特征图,卷积核3x3,边缘填充1像素self.conv1 = nn.Conv2d(in_channels=3,       # 输入通道数(图像的RGB通道)out_channels=32,     # 输出通道数(生成32个新特征图)kernel_size=3,       # 卷积核尺寸(3x3像素)padding=1            # 边缘填充1像素,保持输出尺寸与输入相同)# 批量归一化层:对32个输出通道进行归一化,加速训练self.bn1 = nn.BatchNorm2d(num_features=32)# ReLU激活函数:引入非线性,公式:max(0, x)self.relu1 = nn.ReLU()# 最大池化层:窗口2x2,步长2,特征图尺寸减半(32x32→16x16)self.pool1 = nn.MaxPool2d(kernel_size=2, stride=2)  # stride默认等于kernel_size# ---------------------- 第二个卷积块 ----------------------# 卷积层2:输入32通道(来自conv1的输出),输出64通道self.conv2 = nn.Conv2d(in_channels=32,      # 输入通道数(前一层的输出通道数)out_channels=64,     # 输出通道数(特征图数量翻倍)kernel_size=3,       # 卷积核尺寸不变padding=1            # 保持尺寸:16x16→16x16(卷积后)→8x8(池化后))self.bn2 = nn.BatchNorm2d(num_features=64)self.relu2 = nn.ReLU()self.pool2 = nn.MaxPool2d(kernel_size=2)  # 尺寸减半:16x16→8x8# ---------------------- 第三个卷积块 ----------------------# 卷积层3:输入64通道,输出128通道self.conv3 = nn.Conv2d(in_channels=64,      # 输入通道数(前一层的输出通道数)out_channels=128,    # 输出通道数(特征图数量再次翻倍)kernel_size=3,padding=1            # 保持尺寸:8x8→8x8(卷积后)→4x4(池化后))self.bn3 = nn.BatchNorm2d(num_features=128)self.relu3 = nn.ReLU()  # 复用激活函数对象(节省内存)self.pool3 = nn.MaxPool2d(kernel_size=2)  # 尺寸减半:8x8→4x4# ---------------------- 全连接层(分类器) ----------------------# 计算展平后的特征维度:128通道 × 4x4尺寸 = 128×16=2048维self.fc1 = nn.Linear(in_features=128 * 4 * 4,  # 输入维度(卷积层输出的特征数)out_features=512          # 输出维度(隐藏层神经元数))# Dropout层:训练时随机丢弃50%神经元,防止过拟合self.dropout = nn.Dropout(p=0.5)# 输出层:将512维特征映射到10个类别(CIFAR-10的类别数)self.fc2 = nn.Linear(in_features=512, out_features=10)def forward(self, x):# 输入尺寸:[batch_size, 3, 32, 32](batch_size=批量大小,3=通道数,32x32=图像尺寸)# ---------- 卷积块1处理 ----------x = self.conv1(x)       # 卷积后尺寸:[batch_size, 32, 32, 32](padding=1保持尺寸)x = self.bn1(x)         # 批量归一化,不改变尺寸x = self.relu1(x)       # 激活函数,不改变尺寸x = self.pool1(x)       # 池化后尺寸:[batch_size, 32, 16, 16](32→16是因为池化窗口2x2)# ---------- 卷积块2处理 ----------x = self.conv2(x)       # 卷积后尺寸:[batch_size, 64, 16, 16](padding=1保持尺寸)x = self.bn2(x)x = self.relu2(x)x = self.pool2(x)       # 池化后尺寸:[batch_size, 64, 8, 8]# ---------- 卷积块3处理 ----------x = self.conv3(x)       # 卷积后尺寸:[batch_size, 128, 8, 8](padding=1保持尺寸)x = self.bn3(x)x = self.relu3(x)x = self.pool3(x)       # 池化后尺寸:[batch_size, 128, 4, 4]# ---------- 展平与全连接层 ----------# 将多维特征图展平为一维向量:[batch_size, 128*4*4] = [batch_size, 2048]x = x.view(-1, 128 * 4 * 4)  # -1自动计算批量维度,保持批量大小不变x = self.fc1(x)           # 全连接层:2048→512,尺寸变为[batch_size, 512]x = self.relu3(x)         # 激活函数(复用relu3,与卷积块3共用)x = self.dropout(x)       # Dropout随机丢弃神经元,不改变尺寸x = self.fc2(x)           # 全连接层:512→10,尺寸变为[batch_size, 10](未激活,直接输出logits)return x  # 输出未经过Softmax的logits,适用于交叉熵损失函数# 初始化模型
model = CNN()
#model = model.to(device)  # 将模型移至GPU(如果可用)
# 5. 定义损失函数和优化器
criterion = nn.CrossEntropyLoss()  # 交叉熵损失函数,适用于多分类任务
optimizer = optim.Adam(model.parameters(), lr=0.001)  # Adam优化器,学习率0.001
# 引入学习率调度器,在训练过程中动态调整学习率--训练初期使用较大的 LR 快速降低损失,训练后期使用较小的 LR 更精细地逼近全局最优解。
# 在每个 epoch 结束后,需要手动调用调度器来更新学习率,可以在训练过程中调用 scheduler.step()
scheduler = optim.lr_scheduler.ReduceLROnPlateau(optimizer,        # 指定要控制的优化器(这里是Adam)mode='min',       # 监测的指标是"最小化"(如损失函数)patience=3,       # 如果连续3个epoch指标没有改善,才降低LRfactor=0.5        # 降低LR的比例(新LR = 旧LR × 0.5)
)
# scheduler = optim.lr_scheduler.StepLR(optimizer, step_size=5, gamma=0.1)  
# # 每5个epoch,LR = LR × 0.1  # scheduler = optim.lr_scheduler.MultiStepLR(optimizer, milestones=[10, 20, 30], gamma=0.5)  
# # 当epoch=10、20、30时,LR = LR × 0.5  # scheduler = optim.lr_scheduler.CosineAnnealingLR(optimizer, T_max=10, eta_min=0.0001)  
# # LR在[0.0001, LR_initial]之间按余弦曲线变化,周期为2×T_max  
# 5. 训练模型(记录每个 iteration 的损失)
def train(model, train_loader, test_loader, criterion, optimizer, scheduler, device, epochs):model.train()  # 设置为训练模式# 记录每个 iteration 的损失all_iter_losses = []  # 存储所有 batch 的损失iter_indices = []     # 存储 iteration 序号# 记录每个 epoch 的准确率和损失train_acc_history = []test_acc_history = []train_loss_history = []test_loss_history = []for epoch in range(epochs):running_loss = 0.0correct = 0total = 0for batch_idx, (data, target) in enumerate(train_loader):data, target = data.to(device), target.to(device)  # 移至GPUoptimizer.zero_grad()  # 梯度清零output = model(data)  # 前向传播loss = criterion(output, target)  # 计算损失loss.backward()  # 反向传播optimizer.step()  # 更新参数# 记录当前 iteration 的损失iter_loss = loss.item()all_iter_losses.append(iter_loss)iter_indices.append(epoch * len(train_loader) + batch_idx + 1)# 统计准确率和损失running_loss += iter_loss_, predicted = output.max(1)total += target.size(0)correct += predicted.eq(target).sum().item()# 每100个批次打印一次训练信息if (batch_idx + 1) % 100 == 0:print(f'Epoch: {epoch+1}/{epochs} | Batch: {batch_idx+1}/{len(train_loader)} 'f'| 单Batch损失: {iter_loss:.4f} | 累计平均损失: {running_loss/(batch_idx+1):.4f}')# 计算当前epoch的平均训练损失和准确率epoch_train_loss = running_loss / len(train_loader)epoch_train_acc = 100. * correct / totaltrain_acc_history.append(epoch_train_acc)train_loss_history.append(epoch_train_loss)# 测试阶段model.eval()  # 设置为评估模式test_loss = 0correct_test = 0total_test = 0with torch.no_grad():for data, target in test_loader:data, target = data.to(device), target.to(device)output = model(data)test_loss += criterion(output, target).item()_, predicted = output.max(1)total_test += target.size(0)correct_test += predicted.eq(target).sum().item()epoch_test_loss = test_loss / len(test_loader)epoch_test_acc = 100. * correct_test / total_testtest_acc_history.append(epoch_test_acc)test_loss_history.append(epoch_test_loss)# 更新学习率调度器scheduler.step(epoch_test_loss)print(f'Epoch {epoch+1}/{epochs} 完成 | 训练准确率: {epoch_train_acc:.2f}% | 测试准确率: {epoch_test_acc:.2f}%')# 绘制所有 iteration 的损失曲线plot_iter_losses(all_iter_losses, iter_indices)# 绘制每个 epoch 的准确率和损失曲线plot_epoch_metrics(train_acc_history, test_acc_history, train_loss_history, test_loss_history)return epoch_test_acc  # 返回最终测试准确率# 6. 绘制每个 iteration 的损失曲线
def plot_iter_losses(losses, indices):plt.figure(figsize=(10, 4))plt.plot(indices, losses, 'b-', alpha=0.7, label='Iteration Loss')plt.xlabel('Iteration(Batch序号)')plt.ylabel('损失值')plt.title('每个 Iteration 的训练损失')plt.legend()plt.grid(True)plt.tight_layout()plt.show()# 7. 绘制每个 epoch 的准确率和损失曲线
def plot_epoch_metrics(train_acc, test_acc, train_loss, test_loss):epochs = range(1, len(train_acc) + 1)plt.figure(figsize=(12, 4))# 绘制准确率曲线plt.subplot(1, 2, 1)plt.plot(epochs, train_acc, 'b-', label='训练准确率')plt.plot(epochs, test_acc, 'r-', label='测试准确率')plt.xlabel('Epoch')plt.ylabel('准确率 (%)')plt.title('训练和测试准确率')plt.legend()plt.grid(True)# 绘制损失曲线plt.subplot(1, 2, 2)plt.plot(epochs, train_loss, 'b-', label='训练损失')plt.plot(epochs, test_loss, 'r-', label='测试损失')plt.xlabel('Epoch')plt.ylabel('损失值')plt.title('训练和测试损失')plt.legend()plt.grid(True)plt.tight_layout()plt.show()# 8. 执行训练和测试
epochs = 20  # 增加训练轮次以获得更好效果
print("开始使用CNN训练模型...")
final_accuracy = train(model, train_loader, test_loader, criterion, optimizer, scheduler, device, epochs)
print(f"训练完成!最终测试准确率: {final_accuracy:.2f}%")# # 保存模型
# torch.save(model.state_dict(), 'cifar10_cnn_model.pth')
# print("模型已保存为: cifar10_cnn_model.pth")

一般来说,在一个训练良好的深度学习模型中,训练集上的准确率应该等于或略高于测试集上的准确率。如果测试集上的准确率显著优于训练集,这通常是一个异常情况,可能暗示着以下几种潜在的问题或特殊情况:

正则化层(最常见原因):

许多深度学习模型包含在训练和测试阶段表现不同的层,最典型的就是:

  • Dropout 层: 在训练阶段,Dropout 层会随机关闭一部分神经元,引入噪声,降低模型在训练集上的表现(因为它每次都在一个“残缺”的网络上学习)。但在测试阶段,Dropout 层是关闭的,所有神经元都参与计算,模型使用其全部学习到的能力进行预测。因此,测试时 Dropout 的关闭可能会导致模型在测试集上表现略好于训练集。
  • Batch Normalization(批归一化)层: 在训练阶段,Batch Normalization 使用当前批次数据的均值和方差进行归一化。但在测试阶段,它使用在训练过程中累积的全局均值和方差(移动平均)。这种行为上的差异也可能导致在某些情况下测试集准确率略高于训练集。
  • 解决方法: 在评估模型性能时,确保正确切换模型的模式。例如,在 PyTorch 中使用 model.eval() 模式进行评估,而在 TensorFlow/Keras 中,确保在评估函数中设置 training=False。在评估训练集准确率时,也应切换到 eval() 模式,以进行公平比较。如果这是原因,切换模式后,训练集准确率通常会提高,并接近或略高于测试集准确率。
  • 数据泄露(Data Leakage):

  • 这是机器学习中最严重的问题之一。如果测试集中的信息在无意中“泄露”到了训练过程中,模型就会在测试集上表现得异常好,因为它已经“见过”这部分数据了。常见的数据泄露形式包括:

  • 特征工程在划分数据集之前完成: 例如,计算了整个数据集的均值、标准差来标准化数据,然后再进行训练/测试集划分。这样测试集的信息就通过这些统计量进入了训练集。

  • 不小心将测试数据混入训练数据: 复制粘贴错误,或者数据集划分逻辑有误。
  • 使用“未来”信息进行预测: 在时间序列数据中,如果使用了未来的数据来训练模型,然后在测试集上预测,也会出现数据泄露。
  • 解决方法: 严格遵循数据处理的最佳实践,即先将数据集划分为训练集、验证集和测试集,然后只在训练集上进行特征工程和数据预处理,并将这些处理规则应用到验证集和测试集。

浙大疏锦行-CSDN博客


http://www.hkcw.cn/article/bLSpKCdZsw.shtml

相关文章

樊振东为何加盟德国乒乓球俱乐部 欧洲冒险新挑战

樊振东在社交媒体上发布了一张观看欧冠比赛的照片,引发了球迷对他欧洲之行目的的猜测。紧接着,德国乒乓球甲级联赛FC萨尔布吕肯俱乐部宣布,奥运冠军樊振东将加盟球队。俱乐部官方公告表示,前世界排名第一、国际乒坛巨星之一的樊振东将在新赛季代表球队出战德国乒乓球甲级联…

被认定正当防卫两年后申请128万国赔 正义迟来引发热议

近日,“海南七旬老人救孙砍伤酒后闹事者”事件有了新动态,令人揪心。2025年5月19日,杨成杰在被认定正当防卫的两周年之际提交国家赔偿申请,两天后海南省第二中级人民法院迅速受理,引发社会各界广泛关注与热议。回溯到2017年8月31日,这一天改变了杨成杰的命运。同村的杨某…

一德克士店长用消费券套现13.7万 骗取补贴被判刑

在消费刺激政策不断发力的背景下,政府发放消费券旨在拉动内需、惠及民生。然而,这一暖心之举却被人钻了空子。5月29日下午,上海市松江区人民法院公开审理了一起政府消费券补贴诈骗案,为那些妄图钻空子的人敲响警钟。案件主角是德克士门店的一位店长夏某。2024年,上海推出了…

余华说50%以上鸡汤语都不是他说的 伪语录泛滥引发关注

余华打假网络鸡汤,称超过50%的语录都不是他说的。网友调侃建议他开直播,为写作文积累素材。近年来,余华频繁登上热搜,两年间共上四十多次热搜,成为文坛网红。他的走红原因在于幽默感和反差萌。尽管作品如《活着》、《许三观卖血记》等题材沉重严肃,但他在采访中却极其风趣…

童年照 雷军晒“三条杠”旧照

6月1日儿童节,雷军发文祝所有小朋友都有一个快乐的童年。随后,他晒出了自己的童年照,并配文“三条杠”。照片中的小雷军手捧着花,花上写着“光荣”二字,胳膊上佩戴着三条杠的标志。网友热议纷纷:“这孩子以后有出息”“我掐指一算,此人日后必成大器”“祝小雷同学儿童节…

媒体人辣评《歌手》第三期 选曲策略成胜负关键

随着《歌手2025》第二期节目的热度未减,第三期的网传歌单已在各大社交平台引发热议。这份尚未得到官方确认的歌单不仅展示了歌手们挑战自我的勇气,还暗示了激烈的淘汰危机。单依纯可能在本期节目中挑战林忆莲的高难度作品《归零》或《纤维》,这两首歌曲对气息控制和情感表达…

陈小春演唱会首场郑伊健做特别嘉宾 古惑仔兄弟同框引“爷青回”

5月31日,陈小春在上海举办生旦净末丑巡演首站,郑伊健作为嘉宾惊喜登场。两人合唱了《一起冲一起闯》,古惑仔兄弟再次同框让不少观众感叹“爷青回”。郑伊健还清唱了一段《友情岁月》,感动得陈小春落泪。1996年,陈小春与郑伊健因电影《古惑仔之人在江湖》结缘,二人饰演的“…

专家解读马克龙为何到亚太来演讲 暗流涌动的外交棋局

马克龙访问印尼的行程表面上看似寻常,实际上却暗藏玄机。他此行不仅仅是进行简单的贸易和安全合作,背后可能还有更深层次的战略考量。法国总统出现在东南亚大国,是否仅仅是为了军火交易?还是说这背后隐藏着更多复杂的政治意图?法国与印尼的互动远不止表面的贸易和联合军演…

小车被两辆大货车挤扁 司机爬出 安全问题不容忽视

5月29日,浙江温州赤溪隧道附近发生了一起交通事故,迅速在网上引起关注。根据行车记录仪的画面,下午三点左右,事故发生在一个红绿灯路口,交警已在现场处理,周围聚集了一些司机和行人。事故现场紧张,小车后部严重变形,几乎失去了原来的轮廓。一辆货车以相当快的速度追尾了…

SOC-ESP32S3部分:21-非易失性存储库

飞书文档https://x509p6c8to.feishu.cn/wiki/QB0Zw7GLeio4l4kyaWQcuQT3nZS 非易失性存储 (NVS) 库主要用于在 flash 中存储键值格式的数据。 它允许我们在芯片的闪存中存储和读取数据,即使在断电后,这些数据也不会丢失。 NVS 是 ESP32 flash&#xff…

Agentic RAG 的技术演进详解

一、从被动检索到主动决策:Agentic RAG的核心突破 在人工智能领域,检索增强生成(RAG)技术的诞生标志着大语言模型(LLM)从“内部知识闭环”迈向“外部数据互联”的关键一步。传统RAG通过检索外部文档为生成…

2025-05-31 Python深度学习9——网络模型的加载与保存

文章目录 1 使用现有网络2 修改网络结构2.1 添加新层2.2 替换现有层 3 保存网络模型3.1 完整保存3.2 参数保存(推荐) 4 加载网络模型4.1 加载完整模型文件4.2 加载参数文件 5 Checkpoint5.1 保存 Checkpoint5.2 加载 Checkpoint 本文环境: Py…

智能测试新范式:GenAI 与 Playwright MCP 如何重塑 QA 流程

文章简介 在敏捷开发背景下,传统测试自动化面临动态 UI 适配难、脚本维护成本高等挑战。本文深度解析 ** 生成式 AI(GenAI)与Playwright MCP(模型上下文协议)** 的协同机制,展示如何通过自然语言驱动测试创…

孙颖莎全红婵们等比例长大 体育名将童心未泯

显示图片孙颖莎全红婵们等比例长大-今日头条-手机光明网今天是六一儿童节,祝大家儿童节快乐!来看看体育圈那些等比例长大的名将们吧!2025-06-01 10:05:46责任编辑:zx0176

赛龙舟快得千篇一律沉得五花八门 各地龙舟赛“名场面”

端午节到了,又是一年一度观看“水上F1”的时刻。作为端午节的传统节目,全国各地的龙舟大赛已经激烈打响,各地画风确实有所不同。快得千篇一律,沉得五花八门。有人划龙舟是为了纪念屈原,有人则是去救屈原,还有人是屈原救我。广东佛山的叠滘龙舟队是龙舟界公认最具实力的队…

AI预测3D新模型百十个定位预测+胆码预测+去和尾2025年5月31日第94弹

从今天开始,咱们还是暂时基于旧的模型进行预测,好了,废话不多说,按照老办法,重点8-9码定位,配合三胆下1或下2,杀1-2个和尾,再杀4-5个和值,可以做到100-300注左右。 (1)定…

HackMyVM-First

信息搜集 主机发现 ┌──(kali㉿kali)-[~] └─$ nmap -sn 192.168.43.0/24 Starting Nmap 7.95 ( https://nmap.org ) at 2025-05-31 06:13 EDT Nmap scan report for 192.168.43.1 Host is up (0.0080s latency). MAC Address: C6:45:66:05:91:88 (Unknown) …

【赵渝强老师】数据库不适合Docker容器化部署的原因

在Docker的容器中可以部署运行一个MySQL数据库,并通过数据卷将运行在容器中的MySQL数据库的数据进行持久化。如果这时候运行MySQL的容器被销毁了,数据也将会发生丢失。因此在Docker中部署数据库服务时,一定要考虑数据持久化的问题。但数据库并…

“婚内强奸”被羁押285天当事人发声 申请33万赔偿

5月30日,尹某向河南省濮阳县人民检察院递交了《国家赔偿申请书》,申请国家赔偿33万元,并要求追责相关办案人员。濮阳县人民检察院接受了相关的申请材料。尹某表示,当时以强奸罪对他提起公诉,量刑为4年,但一年前决定不起诉,他被错误羁押了285天,希望获得应有的国家赔偿。…

美计划大幅增加对台军售,外交部回应 坚决反对售台武器

中国外交部发言人林剑在例行记者会上表示,中方坚决反对美国向中国台湾地区出售武器,敦促美方恪守一个中国原则和中美三个联合公报,特别是“八一七”公报的规定,停止售台武器,停止制造台海局势紧张因素。有记者在会上提问,据报道,美国特朗普政府正计划扩大对台军售规模,…