【复现】尝试使用numpy对卷积神经网络中各经典结构进行改写复现

简介: 【复现】尝试使用numpy对卷积神经网络中各经典结构进行改写复现

前言

  numpy作为Python中最常用的科学计算库之一,也被广泛应用于卷积神经网络中的各个组件。本篇博客将介绍如何使用numpy完成卷积神经网络中的各个组件,包括卷积层、池化层、全连接层等,帮助读者更好地理解卷积神经网络的实现原理,同时也能够更加熟练地使用numpy进行深度学习相关的编程。

卷积层

  使用numpy写出卷积层函数需要考虑多个方面,包括输入数据的格式、卷积核的设置、卷积的步长和填充、矩阵乘法的实现以及梯度的计算等

  1. 输入数据的维度和格式:卷积层的输入数据通常是一个四维数组,分别表示样本数、通道数、高度和宽度。因此,在编写卷积层函数时,需要考虑输入数据的维度和格式,以保证函数能够正确地处理数据。
  2. 卷积核的维度和格式:卷积核也是一个四维数组,分别表示输入通道数、输出通道数、卷积核高度和卷积核宽度。在编写卷积层函数时,需要考虑卷积核的维度和格式,以保证函数能够正确地卷积输入数据。
  3. 卷积的步长和填充:卷积层通常会用步长和填充来控制输出数据的尺寸。在编写卷积层函数时,需要考虑步长和填充的设置,以保证函数能够正确地卷积输入数据并输出正确的尺寸。
  4. 矩阵乘法的实现:卷积操作可以看作是一种矩阵乘法操作,因此在编写卷积层函数时,需要考虑如何使用numpy实现高效的矩阵乘法操作,以提高函数的运行效率。
  5. 梯度的计算:在卷积神经网络中,梯度的计算是非常重要的,它可以用来更新卷积核和输入数据的参数。在编写卷积层函数时,需要考虑如何计算梯度,并将梯度传递给前一层的神经元,以实现反向传播算法。

参数说明:

x: 输入数据,shape为(batch_size, in_channels, height, width)

w: 卷积核,shape为(out_channels, in_channels, kernel_height, kernel_width)

b: 偏置,shape为(out_channels,)

stride: 步长,默认为1

padding: 填充,默认为0\

out: 卷积结果,shape为(batch_size, out_channels, out_height, out_width)

ini

复制代码

import numpy as np
    def conv2d(x, w, b, stride=1, padding=0):
        batch_size, in_channels, height, width = x.shape
        out_channels, _, kernel_height, kernel_width = w.shape
        # 计算输出特征图的尺寸
        out_height = (height + 2 * padding - kernel_height) // stride + 1
        out_width = (width + 2 * padding - kernel_width) // stride + 1
        # 对输入数据进行填充
        x_pad = np.pad(x, ((0, 0), (0, 0), (padding, padding), (padding, padding)), mode='constant')
        # 初始化输出特征图
        out = np.zeros((batch_size, out_channels, out_height, out_width))
        # 进行卷积操作
        for i in range(out_height):
            for j in range(out_width):
                for k in range(out_channels):
                    out[:, k, i, j] = np.sum(x_pad[:, :, i*stride:i*stride+kernel_height, j*stride:j*stride+kernel_width] * w[k, :, :, :], axis=(1, 2, 3)) + b[k]
        return out

使用示例

ini

复制代码

# 构造输入数据
x = np.random.randn(1, 3, 640, 640)
# 构造卷积核和偏置
w = np.random.randn(64, 3, 3, 3)
b = np.random.randn(64)
# 进行卷积操作
out = conv2d(x, w, b, stride=1, padding=1)
# 打印输出特征图的尺寸
print(out.shape)

池化层

  使用numpy写出池化层函数需要考虑多个方面,包括输入数据的格式、池化的类型和大小、池化的步长和填充、矩阵乘法的实现以及梯度的计算。

  1. 输入数据的维度和格式:池化层的输入数据通常是一个四维数组,分别表示样本数、通道数、高度和宽度。因此,在编写池化层函数时,需要考虑输入数据的维度和格式,以保证函数能够正确地处理数据。
  2. 池化的类型和大小:池化层通常有两种类型,即最大池化和平均池化,同时还需要设置池化的大小。在编写池化层函数时,需要考虑池化的类型和大小的设置,以保证函数能够正确地池化输入数据。
  3. 池化的步长和填充:池化层通常会用步长和填充来控制输出数据的尺寸。在编写池化层函数时,需要考虑步长和填充的设置,以保证函数能够正确地池化输入数据并输出正确的尺寸。
  4. 矩阵乘法的实现:池化操作可以看作是一种矩阵乘法操作,因此在编写池化层函数时,需要考虑如何使用numpy实现高效的矩阵乘法操作,以提高函数的运行效率。
  5. 梯度的计算:在卷积神经网络中,梯度的计算是非常重要的,它可以用来更新输入数据的参数。在编写池化层函数时,需要考虑如何计算梯度,并将梯度传递给前一层的神经元,以实现反向传播算法。

ini

复制代码

import numpy as np
    def max_pooling(input_data, pool_size, strides):
        batch_size, in_channels, in_height, in_width = input_data.shape
        pool_height, pool_width = pool_size
        stride_height, stride_width = strides
        out_height = int((in_height - pool_height) / stride_height + 1)
        out_width = int((in_width - pool_width) / stride_width + 1)
        output_data = np.zeros((batch_size, in_channels, out_height, out_width))
        for i in range(out_height):
            for j in range(out_width):
                output_data[:, :, i, j] = np.max(
                    input_data[:, :, i * stride_height:i * stride_height + pool_height,
                               j * stride_width:j * stride_width + pool_width], axis=(2, 3))
        return output_data

scss

复制代码

input_data = np.random.randn(1, 3, 640, 640)
    pool_size = (2, 2)
    strides = (2, 2)
    output_data = max_pooling(input_data, pool_size, strides)
    print("input_data:\n", input_data.shape)
    print("output_data:\n", output_data.shape)

image.png

  这个函数接受三个参数:输入数据、池化窗口大小和步幅。 输入数据的形状为(batch_size, in_channels, in_height, in_width),其中batch_size表示批次大小,in_channels表示输入数据的通道数,in_heightin_width分别表示输入数据的高度和宽度。 池化窗口大小为(pool_height, pool_width),步幅为(stride_height, stride_width)。这个函数通过双重循环遍历每个池化窗口,然后在这个窗口内取最大值作为输出值。

Dropout层

Dropout层具有一定的随机性:

Dropout层的主要作用是随机删除一部分神经元,因此在实现时需要使用随机数生成器产生一个0-1之间的随机数矩阵,然后根据设定的概率阈值(如0.5)来判断哪些神经元需要被删除。

训练和测试模式:

在训练和测试两个阶段,Dropout层的行为是不同的。在训练阶段,Dropout层会删除一部分神经元;而在测试阶段,Dropout层不会删除神经元,而是将所有神经元的权重乘以概率阈值(如0.5),以保持期望输出不变。因此,在实现时需要添加一个参数来表示当前是训练还是测试模式,并针对不同模式进行不同的计算。

归一化:

在删除一部分神经元后,Dropout层会使得剩余神经元的输出值变大,因此需要对输出值进行归一化。一种简单的方法是将输出值除以概率阈值(如0.5),以保持期望输出不变。

反向传播:

在反向传播过程中,Dropout层需要将输出值乘以一个掩码矩阵(即随机删除的神经元对应位置为0,未删除的神经元对应位置为1),以保留未删除神经元的梯度信息。因此,在实现时需要注意掩码矩阵的生成和使用。

参数管理:

Dropout层没有可训练参数,因此在实现时不需要考虑参数初始化和更新的问题。

ini

复制代码

import numpy as np
    def dropout(x, p):
        # 其中,x是输入的矩阵,p是保留概率
        mask = np.random.binomial(1, 1-p, size=x.shape) / (1-p)
        return x * mask
    # 生成一个的矩阵
    x = np.random.rand(1, 3, 640, 640)
    # 使用dropout函数保留概率为0.5
    y = dropout(x, 0.5)

scss

复制代码

# 输出x和y的值
    print("x:")
    print(x.shape)
    print("y:")
    print(y.shape)

image.png

全连接层

输入输出维度:

全连接层的输入和输出都是二维矩阵,其中输入矩阵的第一维表示样本数,第二维表示特征数;输出矩阵的第一维表示样本数,第二维表示输出节点数。在实现时需要确保输入输出矩阵的维度正确。

权重和偏置:

全连接层的核心是权重矩阵和偏置向量。在实现时需要使用随机数生成器初始化权重矩阵和偏置向量,并在反向传播过程中更新它们的值。

前向传播:

全连接层的前向传播过程是将输入矩阵和权重矩阵相乘,再加上偏置向量,最后通过激活函数得到输出矩阵。在实现时需要注意矩阵乘法的维度匹配和激活函数的选择。

反向传播:

全连接层的反向传播过程是根据损失函数对权重矩阵和偏置向量进行梯度更新。在实现时需要注意梯度的计算和矩阵乘法的转置。同时,为了避免梯度消失或爆炸,可以使用梯度裁剪等方法进行优化。

批量处理:

在实际使用中,一般会采用批量处理的方式进行训练。因此,在实现时需要注意批量大小的设置,以及在前向传播和反向传播过程中对批量数据进行处理。

参数管理:

全连接层的参数包括权重矩阵和偏置向量,需要进行初始化和更新。在实现时需要注意参数的维度和存储方式。

python

复制代码

import numpy as np
    class FullyConnectedLayer:
        def __init__(self, input_size, output_size):
            self.W = np.random.randn(input_size, output_size) * 0.01
            self.b = np.zeros((1, output_size))
            self.input = None
        def forward(self, x):
            self.input = x
            output = np.dot(x, self.W) + self.b
            return output
        def backward(self, grad_output, learning_rate):
            grad_input = np.dot(grad_output, self.W.T)
            grad_W = np.dot(self.input.T, grad_output)
            grad_b = np.sum(grad_output, axis=0, keepdims=True)
            self.W -= learning_rate * grad_W
            self.b -= learning_rate * grad_b
            return grad_input

ini

复制代码

# 生成随机输入
x = np.random.randn(10, 5)
# 创建全连接层
fc_layer = FullyConnectedLayer(5, 3)
# 进行前向传播
output = fc_layer.forward(x)
# 打印输出的形状
print("output.shape:", output.shape)
# 进行反向传播
grad_output = np.random.randn(10, 3)
grad_input = fc_layer.backward(grad_output, 0.01)
# 打印反向传播的输出形状
print("grad_input.shape", grad_input.shape)

image.png

BN层

  1. 计算均值和方差:BN层的主要作用是对每个特征进行均值和方差归一化,以使得每个特征在不同样本中具有相似的分布。在实现时需要计算每个特征的均值和方差,可以使用numpy的mean和var函数进行计算。
  2. 归一化:根据计算得到的均值和方差,可以对每个特征进行归一化处理,以使得每个特征的均值为0,方差为1。在实现时需要注意除以方差时需要加上一个很小的数(如1e-8)以避免除以0的情况。
  3. 缩放和平移:为了保留每个特征的表达能力,BN层还需要对归一化后的结果进行缩放和平移。具体来说,对于每个特征,需要学习一个缩放因子和一个平移因子,以使得归一化后的结果可以任意平移和缩放。在实现时需要使用随机数生成器初始化缩放和平移因子,并在反向传播过程中更新它们的值。
  4. 反向传播:在反向传播过程中,BN层需要计算梯度,并根据梯度更新缩放和平移因子。在实现时需要注意梯度的计算和矩阵乘法的转置。同时,为了避免梯度消失或爆炸,可以使用梯度裁剪等方法进行优化。
  5. 训练和测试模式:在训练和测试两个阶段,BN层的行为是不同的。在训练阶段,BN层会根据当前批量数据计算均值和方差,并使用它们对每个特征进行归一化;而在测试阶段,BN层需要使用之前计算得到的整个数据集的均值和方差对每个特征进行归一化。因此,在实现时需要添加一个参数来表示当前是训练还是测试模式,并针对不同模式进行不同的计算。
  6. 参数管理:BN层的参数包括缩放因子和平移因子,需要进行初始化和更新。在实现时需要注意参数的维度和存储方式。

ini

复制代码

import numpy as np
    import matplotlib.pyplot as plt
    def batch_norm(x, gamma, beta, eps=1e-5):
        N, C, H, W = x.shape
        mu = np.mean(x, axis=(0, 2, 3), keepdims=True)
        var = np.var(x, axis=(0, 2, 3), keepdims=True)
        x_norm = (x - mu) / np.sqrt(var + eps)
        out = gamma.reshape(1, C, 1, 1) * x_norm + beta.reshape(1, C, 1, 1)
        cache = (x, x_norm, mu, var, gamma, beta, eps)
        return out, cache

ini

复制代码

# 随机生成一个四维数据集
    np.random.seed(0)
    x = np.random.randn(1, 3, 640, 640)
    # 对数据进行BN层处理
    gamma = np.ones(3)
    beta = np.zeros(3)
    x_norm, _ = batch_norm(x, gamma, beta)
    # 绘制归一化前后的数据分布图
    fig, axs = plt.subplots(2, 1, figsize=(8, 6))
    axs[0].hist(np.sum(x, axis=(1, 2, 3)), bins=50)
    axs[0].set_title("Before BN")
    axs[1].hist(np.sum(x_norm, axis=(1, 2, 3)), bins=50)
    axs[1].set_title("After BN")
    plt.show()

image.png


相关文章
|
5月前
|
机器学习/深度学习 PyTorch TensorFlow
卷积神经网络深度解析:从基础原理到实战应用的完整指南
蒋星熠Jaxonic,深度学习探索者。深耕TensorFlow与PyTorch,分享框架对比、性能优化与实战经验,助力技术进阶。
|
6月前
|
机器学习/深度学习 人工智能 算法
卷积神经网络深度解析:从基础原理到实战应用的完整指南
蒋星熠Jaxonic带你深入卷积神经网络(CNN)核心技术,从生物启发到数学原理,详解ResNet、注意力机制与模型优化,探索视觉智能的演进之路。
586 11
|
9月前
|
机器学习/深度学习 算法 数据安全/隐私保护
基于PSO粒子群优化TCN-LSTM时间卷积神经网络时间序列预测算法matlab仿真
本内容展示了一种基于粒子群优化(PSO)与时间卷积神经网络(TCN)的时间序列预测方法。通过 MATLAB2022a 实现,完整程序运行无水印,核心代码附详细中文注释及操作视频。算法利用 PSO 优化 TCN 的超参数(如卷积核大小、层数等),提升非线性时间序列预测性能。TCN 结构包含因果卷积层与残差连接,结合 LSTM 构建混合模型,经多次迭代选择最优超参数,最终实现更准确可靠的预测效果,适用于金融、气象等领域。
|
6月前
|
机器学习/深度学习 传感器 数据采集
【故障识别】基于CNN-SVM卷积神经网络结合支持向量机的数据分类预测研究(Matlab代码实现)
【故障识别】基于CNN-SVM卷积神经网络结合支持向量机的数据分类预测研究(Matlab代码实现)
426 0
|
8月前
|
机器学习/深度学习 人工智能 PyTorch
零基础入门CNN:聚AI卷积神经网络核心原理与工业级实战指南
卷积神经网络(CNN)通过局部感知和权值共享两大特性,成为计算机视觉的核心技术。本文详解CNN的卷积操作、架构设计、超参数调优及感受野计算,结合代码示例展示其在图像分类、目标检测等领域的应用价值。
469 7
|
10月前
|
机器学习/深度学习 算法 数据安全/隐私保护
基于PSO粒子群优化TCN时间卷积神经网络时间序列预测算法matlab仿真
本内容介绍了一种基于PSO(粒子群优化)改进TCN(时间卷积神经网络)的时间序列预测方法。使用Matlab2022a运行,完整程序无水印,附带核心代码中文注释及操作视频。TCN通过因果卷积层与残差连接处理序列数据,PSO优化其卷积核权重等参数以降低预测误差。算法中,粒子根据个体与全局最优位置更新速度和位置,逐步逼近最佳参数组合,提升预测性能。
|
9月前
|
机器学习/深度学习 数据采集 监控
基于CNN卷积神经网络和GEI步态能量提取的步态识别算法matlab仿真,对比不同角度下的步态识别性能
本项目基于CNN卷积神经网络与GEI步态能量提取技术,实现高效步态识别。算法使用不同角度(0°、45°、90°)的步态数据库进行训练与测试,评估模型在多角度下的识别性能。核心流程包括步态图像采集、GEI特征提取、数据预处理及CNN模型训练与评估。通过ReLU等激活函数引入非线性,提升模型表达能力。项目代码兼容Matlab2022a/2024b,提供完整中文注释与操作视频,助力研究与应用开发。
|
9月前
|
机器学习/深度学习 算法 数据安全/隐私保护
基于WOA鲸鱼优化的TCN-GRU时间卷积神经网络时间序列预测算法matlab仿真
本内容包含时间序列预测算法的相关资料,涵盖以下几个方面:1. 算法运行效果预览(无水印);2. 运行环境为Matlab 2022a/2024b;3. 提供部分核心程序,完整版含中文注释及操作视频;4. 理论概述:结合时间卷积神经网络(TCN)与鲸鱼优化算法(WOA),优化TCN超参数以提升非线性时间序列预测性能。通过因果卷积层与残差连接构建TCN模型,并用WOA调整卷积核大小、层数等参数,实现精准预测。适用于金融、气象等领域决策支持。
|
9月前
|
机器学习/深度学习 数据采集 并行计算
基于WOA鲸鱼优化的TCN时间卷积神经网络时间序列预测算法matlab仿真
本内容介绍了一种基于TCN(Temporal Convolutional Network)与WOA(Whale Optimization Algorithm)的时间序列预测算法。TCN通过扩张卷积捕捉时间序列长距离依赖关系,结合批归一化和激活函数提取特征;WOA用于优化TCN网络参数,提高预测精度。算法流程包括数据归一化、种群初始化、适应度计算及参数更新等步骤。程序基于Matlab2022a/2024b开发,完整版含详细中文注释与操作视频,运行效果无水印展示。适用于函数优化、机器学习调参及工程设计等领域复杂任务。
|
9月前
|
机器学习/深度学习 算法 数据安全/隐私保护
基于PSO粒子群优化TCN-GRU时间卷积神经网络时间序列预测算法matlab仿真
本内容涵盖基于粒子群优化(PSO)与时间卷积神经网络(TCN)的时间序列预测算法。完整程序运行效果无水印,适用于Matlab2022a版本。核心代码配有详细中文注释及操作视频。理论部分阐述了传统方法(如ARIMA)在非线性预测中的局限性,以及TCN结合PSO优化超参数的优势。模型由因果卷积层和残差连接组成,通过迭代训练与评估选择最优超参数,最终实现高精度预测,广泛应用于金融、气象等领域。