从零开始学习深度学习库-6:集成新的自动微分模块和MNIST数字分类器

本文主要是介绍从零开始学习深度学习库-6:集成新的自动微分模块和MNIST数字分类器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上一篇文章中,我们完成了自动微分模块的代码。深度学习库依赖于自动微分模块来处理模型训练期间的反向传播过程。然而,我们的库目前还是“手工”计算权重导数。现在我们拥有了自己的自动微分模块,接下来让我们的库使用它来执行反向传播吧!

此外,我们还将构建一个数字分类器来测试一切是否正常工作。

使用自动微分模块并非必要,不使用这个模块也没事,用原本的方法也能很好地工作。

然而,当我们开始在库中实现更复杂的层和激活函数时,硬编码导数计算可能会变得难以理解。

自动微分模块为我们提供了一个抽象层,帮助我们计算导数,这样我们就无需手动完成这一过程了。

让我们开始创建名为 nn.py 的文件。这个文件将作为您需要的所有神经网络组件的中心存储库,包括不同类型的层、激活函数以及构建和操作神经网络所需的潜在其他实用程序。

import autodiff as ad
import numpy as np
import loss 
import optimnp.random.seed(345)class Layer:def __init__(self):passclass Linear(Layer):def __init__(self, units):self.units = unitsself.w = Noneself.b = Nonedef __call__(self, x):if self.w is None:self.w = ad.Tensor(np.random.uniform(size=(x.shape[-1], self.units), low=-1/np.sqrt(x.shape[-1]), high=1/np.sqrt(x.shape[-1])))self.b = ad.Tensor(np.zeros((1, self.units)))return x @ self.w + self.b

到目前为止,一切都很简单。当这个类的实例以函数形式调用时,__call__ 方法只执行前向传播。如果是首次调用,它还将初始化层的参数。

权重和偏置现在是 Tensor类的实例,这意味着它们将在操作开始时成为计算图的一部分。这也意味着我们的自动微分模块能够计算它们的导数。

请注意,我们不再需要像以前那样的backward方法。自动微分模块将为我们计算导数!
激活函数

class Sigmoid:def __call__(self, x):return 1 / (1 + np.e ** (-1 * x))class Softmax:def __call__(self, x):e_x = np.e ** (x - np.max(x.value))s_x = (e_x) / ad.reduce_sum(e_x, axis=1, keepdims=True)return s_xclass Tanh:def __call__(self, x):return (2 / (1 + np.e ** (-2 * x))) - 1

Model class

class Model:def __init__(self, layers):self.layers = layersdef __call__(self, x):output = xfor layer in self.layers:output = layer(output)return outputdef train(self, x, y, epochs=10, loss_fn = loss.MSE, optimizer=optim.SGD(lr=0.1), batch_size=32):for epoch in range(epochs):_loss = 0print (f"EPOCH", epoch + 1)for batch in tqdm(range(0, len(x), batch_size)):output = self(x[batch:batch+batch_size])l = loss_fn(output, y[batch:batch+batch_size])optimizer(self, l)_loss += lprint ("LOSS", _loss.value)

模型类的结构与之前相似,但现在可以对数据集进行批量训练。

与一次性使用整个数据集相比,批量训练使模型能更好地理解其处理的数据。
loss.py
loss.py 文件将包含我们在库中实现的各种损失函数。

import autodiff as addef MSE(pred, real):loss = ad.reduce_mean((pred - real)**2)return lossdef CategoricalCrossentropy(pred, real):loss = -1 * ad.reduce_mean(real * ad.log(pred))return loss

同样,与之前一样,只是没有了 backward 方法。

关于新的自动微分功能:在我们继续讨论优化器之前,您可能已经注意到代码现在使用了自动微分模块中的一些新功能,

以下是这些新功能:

def reduce_sum(tensor, axis = None, keepdims=False):var = Tensor(np.sum(tensor.value, axis = axis, keepdims=keepdims))var.dependencies.append(tensor)var.grads.append(np.ones(tensor.value.shape))return vardef reduce_mean(tensor, axis = None, keepdims=False):return reduce_sum(tensor, axis, keepdims) / tensor.value.sizedef log(tensor):var = Tensor(np.log(tensor.value))var.dependencies.append(tensor)var.grads.append(1 / tensor.value)return var

optim.py
文件将包含我们在这个库中实现的不同优化器。
SGD

from nn import Layerclass SGD:def __init__(self, lr):self.lr = lrdef delta(self, param):return param.gradient * self.lrdef __call__(self, model, loss):loss.get_gradients()for layer in model.layers:if isinstance(layer, Layer):layer.update(self)

Momentum

class Momentum:def __init__(self, lr = 0.01, beta=0.9):self.lr = lrself.beta = betaself.averages = {}def momentum_average(self, prev, grad):return (self.beta * prev) + (self.lr * grad)def delta(self, param):param_id = param.idif param_id not in self.averages:self.averages[param_id] = 0self.averages[param_id] = self.momentum_average(self.averages[param_id], param.gradient)return self.averages[param_id]def __call__(self, model, loss):loss.get_gradients()for layer in model.layers:if isinstance(layer, Layer):layer.update(self)

RMSProp

class RMSProp:def __init__(self, lr = 0.01, beta=0.9, epsilon=10**-10):self.lr = lrself.beta = betaself.epsilon = epsilonself.averages = {}def rms_average(self, prev, grad):return self.beta * prev + (1 - self.beta) * (grad ** 2)def delta(self, param):param_id = param.idif param_id not in self.averages:self.averages[param_id] = 0self.averages[param_id] = self.rms_average(self.averages[param_id], param.gradient)return (self.lr / (self.averages[param_id] + self.epsilon) ** 0.5) * param.gradientdef __call__(self, model, loss):loss.get_gradients()for layer in model.layers:if isinstance(layer, Layer):layer.update(self)

Adam

class Adam:def __init__(self, lr = 0.01, beta1=0.9, beta2=0.999, epsilon=10**-8):self.lr = lrself.beta1 = beta1self.beta2 = beta2self.epsilon = epsilonself.averages = {}self.averages2 = {}def rms_average(self, prev, grad):return (self.beta2 * prev) + (1 - self.beta2) * (grad ** 2)def momentum_average(self, prev, grad):return (self.beta1 * prev) + ((1 - self.beta1) * grad)def delta(self, param):param_id = param.idif param_id not in self.averages:self.averages[param_id] = 0self.averages2[param_id] = 0self.averages[param_id] = self.momentum_average(self.averages[param_id], param.gradient)self.averages2[param_id] = self.rms_average(self.averages2[param_id], param.gradient)adjust1 = self.averages[param_id] / (1 - self.beta1)adjust2 = self.averages2[param_id] / (1 - self.beta2)return self.lr * (adjust1 / (adjust2 ** 0.5 + self.epsilon))def __call__(self, model, loss):loss.get_gradients()for layer in model.layers:            if isinstance(layer, Layer):layer.update(self)

call

def __call__(self, model, loss):loss.get_gradients()for layer in model.layers:            if isinstance(layer, Layer):layer.update(self)

当一个优化器类的实例被调用时,它会接受它的训练模型和损失值。
loss.get_gradients()

在这里,我们利用了我们的自动微分模块,

如果您还记得,get_gradients 方法是 Tensor 类的一部分,它计算涉及这个张量计算的所有变量的导数。

这意味着网络中的所有权重和偏置现在都已计算出其导数,这些导数都存储在它们的梯度属性中。

for layer in model.layers:            if isinstance(layer, Layer):layer.update(self)

现在导数已经计算完毕,优化器将遍历网络的每一层,并通过调用层的更新方法来更新其参数,将自身作为参数传递给它。

我们线性层类中的更新方法如下:

#nn.py
class Linear(Layer):...def update(self, optim):self.w.value -= optim.delta(self.w)self.b.value -= optim.delta(self.b)self.w.grads = []self.w.dependencies = []self.b.grads = []self.b.dependencies = []

这个方法接收一个优化器的实例,并根据优化器计算出的delta值更新层的参数。

self.w.value -= optim.delta(self.w)
self.b.value -= optim.delta(self.b)

delta 方法是优化器类中的一个函数。它接收一个张量,并利用其导数来确定这个张量应该调整的量。

delta 方法的具体实现可能会根据使用的优化器而有所不同。

让我们来看一下其中一个 delta 方法的实现。

class RMSProp:...def rms_average(self, prev, grad):return self.beta * prev + (1 - self.beta) * (grad ** 2)def delta(self, param):param_id = param.idif param_id not in self.averages:self.averages[param_id] = 0self.averages[param_id] = self.rms_average(self.averages[param_id], param.gradient)return (self.lr / (self.averages[param_id] + self.epsilon) ** 0.5) * param.gradient...
param_id = param.idif param_id not in self.averages:self.averages[param_id] = 0

请记住,大多数优化器会跟踪每个参数梯度的某种平均值,以帮助定位全局最小值。

这就是为什么我们为每个张量分配了一个ID,以便优化器能够跟踪它们的梯度平均值。

self.averages[param_id] = self.rms_average(self.averages[param_id], param.gradient)return (self.lr / (self.averages[param_id] + self.epsilon) ** 0.5) * param.gradient

如有必要,会重新计算参数的梯度平均值(请注意,SGD 不维持平均值)。

然后,该方法计算参数应调整的幅度,并返回此值。

探索其他优化器,以帮助您了解它们的工作原理。

MNIST 数字分类器

为了验证我们所有新更改的功能是否符合预期,让我们构建一个神经网络来分类手写数字图像。

from sklearn.datasets import load_digits
import numpy as np
import nn
import optim
import loss
from autodiff import *
from matplotlib import pyplot as plt

数据集准备:

def one_hot(n, max):arr = [0] * maxarr[n] = 1return arrmnist = load_digits()
images = np.array([image.flatten() for image in mnist.images])
targets = np.array([one_hot(n, 10) for n in mnist.target])

MNIST 数据集包含作为 2D 数组表示的图像。然而,由于我们的库目前不支持接受 2D 输入的层,我们需要将这些数组展平成 1D 向量。

one_hot 函数接收一个数字,并为其返回一个长度由数据集中的最大值确定的 one-hot 编码数组。

one_hot(3, 10) => [0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0]

建立模型:

model = nn.Model([nn.Linear(64),nn.Tanh(),nn.Linear(32),nn.Sigmoid(),nn.Linear(10),nn.Softmax()
])

这是一个简单的前馈网络,使用 softmax 函数来输出概率分布。

这个分布指定了在给定输入(本例中为图像)的情况下,每个类别(本例中为每个数字)为真的概率。

训练模型:

model.train(images[:1000], targets[:1000], epochs=50, loss_fn=loss.CategoricalCrossentropy, optimizer=optim.RMSProp(0.001), batch_size=128)

我们只需要这一行代码就可以训练我们的模型。

我决定使用数据集中的前1000张图像来训练模型(总共约有1700张图像)。

随意尝试不同的训练配置,看看模型的反应如何。您可以尝试更改优化器、损失函数或学习率,看看这些更改如何影响训练。

测试模型:

images = images[1000:]
np.random.shuffle(images)for image in images:plt.imshow(image.reshape((8, 8)), cmap='gray')plt.show()pred = np.argmax(model(np.array([image])).value, axis=1)print (pred)

在这里,我们将模型未训练的图像随机打乱顺序。

然后我们逐一查看每张图像,显示它,并让我们的模型预测每张图像所表示的数字。
在这里插入图片描述

这篇关于从零开始学习深度学习库-6:集成新的自动微分模块和MNIST数字分类器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1101244

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

从去中心化到智能化:Web3如何与AI共同塑造数字生态

在数字时代的演进中,Web3和人工智能(AI)正成为塑造未来互联网的两大核心力量。Web3的去中心化理念与AI的智能化技术,正相互交织,共同推动数字生态的变革。本文将探讨Web3与AI的融合如何改变数字世界,并展望这一新兴组合如何重塑我们的在线体验。 Web3的去中心化愿景 Web3代表了互联网的第三代发展,它基于去中心化的区块链技术,旨在创建一个开放、透明且用户主导的数字生态。不同于传统

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

usaco 1.2 Name That Number(数字字母转化)

巧妙的利用code[b[0]-'A'] 将字符ABC...Z转换为数字 需要注意的是重新开一个数组 c [ ] 存储字符串 应人为的在末尾附上 ‘ \ 0 ’ 详见代码: /*ID: who jayLANG: C++TASK: namenum*/#include<stdio.h>#include<string.h>int main(){FILE *fin = fopen (

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss