Pytorch深度学习指南 卷I --编程基础(A Beginner‘s Guide) 第1章 一个简单的回归

news2025/1/23 11:26:39

本章正式开始使用pytorch的接口来实现对应的numpy的学习的过程,来学习模型的实现,我们会介绍numpy是如何学习的,以及我们如何一步步的通过torch的接口来实现简单化的过程,优雅的展示我们的代码,已经我们的代码完成的事情

numpy的线性回归

在此之前,先看看现在的numpy实现的学习的过程是什么样的

#引入计算模块
import numpy as np
from sklearn.linear_model import LinearRegression

import torch
import torch.optim as optim
import torch.nn as nn
from torchviz import make_dot


用真实的数据来生成对应的分布点的数据
true_b = 1
true_w = 2
N = 100

# Data Generation
np.random.seed(42)
x = np.random.rand(N, 1)
epsilon = (.1 * np.random.randn(N, 1))
y = true_b + true_w * x + epsilon
np.rand

# Shuffles the indices
idx = np.arange(N)
np.random.shuffle(idx)

# Uses first 80 random indices for train
train_idx = idx[:int(N*.8)]
# Uses the remaining indices for validation
val_idx = idx[int(N*.8):]

# Generates train and validation sets
x_train, y_train = x[train_idx], y[train_idx]
x_val, y_val = x[val_idx], y[val_idx]



np.random.seed(42)
b = np.random.randn(1)
w = np.random.randn(1)


for _ in range(1000):
	# Step 1 - Computes our model's predicted output - forward pass
	yhat = b + w * x_train
	
	# Step 2 - Computing the loss
	# We are using ALL data points, so this is BATCH gradient
	# descent. How wrong is our model? That's the error!
	error = (yhat - y_train)
	
	# It is a regression, so it computes mean squared error (MSE)
	loss = (error ** 2).mean()
	
	
	# Step 3 - Computes gradients for both "b" and "w" parameters
	b_grad = 2 * error.mean()
	w_grad = 2 * (x_train * error).mean()
	
	
	# Sets learning rate - this is "eta" ~ the "n" like Greek letter
	lr = 0.1
	
	# Step 4 - Updates parameters using gradients and 
	# the learning rate
	b = b - lr * b_grad
	w = w - lr * w_grad


#验证,通过线性的模型直接学习
linr = LinearRegression()
linr.fit(x_train, y_train)

如上一章所说,我们的5个步骤,就是准备数据,前向传递,计算损失,计算梯度,更新参数,循环往复

pytorch的取代

张量(通常指3维)但是这里除了标量全部都是张量,为了简化。

#如下是创建张量的例子
scalar = torch.tensor(3.14159)	#张量
vector = torch.tensor([1, 2, 3]) #一维
matrix = torch.ones((2, 3), dtype=torch.float) #二维
tensor = torch.randn((2, 3, 4), dtype=torch.float)	#三维

获取到张量的shape

shape将会是我们以后将会长期用到的东西

print(tensor.size(), tensor.shape)
torch.Size([2, 3, 4]) torch.Size([2, 3, 4])

view

我们可以使用view的接口来改变一个张量的shape,注意view并不会创建新的张量。

# We get a tensor with a different shape but it still is the SAME tensor
same_matrix = matrix.view(1, 6)
# If we change one of its elements...
same_matrix[0, 1] = 2.

创建新的tensor

使用这个可以使用new_tensor和clone

# We can use "new_tensor" method to REALLY copy it into a new one
different_matrix = matrix.new_tensor(matrix.view(1, 6))
# Now, if we change one of its elements...
different_matrix[0, 1] = 3.

使用clone.detach,为什么要用detach(涉及到后面的数据存放和计算的内容)

another_matrix = matrix.view(1, 6).clone().detach()

加载数据、设备、CUDA

我们需要从numpy的数据转成tensor张量

x_train_tensor = torch.as_tensor(x_train)
x_train.dtype, x_train_tensor.dtype

注意,这里变成了torch的张量,但是这里是还是共享的原始的内存,换句话说,改了还是会一起改

(dtype('float64'), torch.float64)

cuda的数据

cuda就是要使用GPU来存储和运算的数据,我们需要手动的将其放到GPU的内存中,用于加速计算的过程

#判断是否有GPU的数据可以使用的接口,如果存在那么设置device为GPU,否则CPU
device = 'cuda' if torch.cuda.is_available() else 'cpu'
#可以获取GPU的数量
n_cudas = torch.cudsa.device_count()
for i in range(n_cuda):
	print(torch.cuda.get_device_naem(i))

我的机器上的输出的结果

NVIDIA GeForce RTX 2060

将数据发送到GPU上

gpu_tensor = torch.as_tensor(x_train).to(device)
gpu_tensor[0]

输出

torch.cuda.FloatTensor

如果发送到了GPU上的数据,需要重新变成numpy的数组,需要使用CPU的变量

back_to_numpy = x_train_tensor.cpu().numpy()

创建参数,并且需要梯度计算

为什么要使用torch,很多的时候在于可以自行进行梯度计算,也可以使用到很多pytorch的使用的接口和用法

# 创建一个随机1以内的参数,并且需要梯度,类型是float
b = torch.randn(1, requires_grad=True, dtype=torch.float)
w = torch.randn(1, requires_grad=True, dtype=torch.float)
print(b, w)

创建数据并且发送到GPU,但是需要注意的是,这样会丢失梯度,因为这个是CPU的数据需要梯度,发送到GPU后又是新的数据,

b = torch.randn(1, requires_grad=True, dtype=torch.float).to(device)
w = torch.randn(1, requires_grad=True, dtype=torch.float).to(device)

好的方法是直接在GPU上创建变量

b = torch.randn(1, requires_grad=True, dtype=torch.float,device = device)
w = torch.randn(1, requires_grad=True, dtype=torch.float),device = device)

Autograd

自动求解梯度

backward

我们可以通过backward直接计算和进行backward的实现,注意b,w是我们创建的参数(需要梯度的那种)

# Step 1 - Computes our model's predicted output - forward pass
yhat = b + w * x_train_tensor

# Step 2 - Computes the loss
# We are using ALL data points, so this is BATCH gradient descent
# How wrong is our model? That's the error! 
error = (yhat - y_train_tensor)
# It is a regression, so it computes mean squared error (MSE)
loss = (error ** 2).mean()

# Step 3 - Computes gradients for both "b" and "w" parameters
# No more manual computation of gradients! 
# b_grad = 2 * error.mean()
# w_grad = 2 * (x_tensor * error).mean()
loss.backward()

需要说明的是,所有的参与到b,w的计算的都是需要梯度的参数,例如这里面的yhat,error,loss,都是通过w,b的计算得来的,我们都认为是传递了梯度的计算特性

需要特别注意的是,这里的梯度是累计的,因为为了后续的小批量的情况,所以每次更新完毕以后需要手动设置gard_zero_()函数

b.grad.zero_(), w.grad.zero_()

还需要特别注意的是,我们更新参数的时候,是不能直接更新的,需要使用

    with torch.no_grad():
        b -= lr * b.grad
        w -= lr * w.grad

这在停一下,除了前面的变量的配置不一样的地方,我们这里已经在改造我们的numpy代码了

	# 原来的numpy的梯度的计算需要手动计算
	# Step 3 - Computes gradients for both "b" and "w" parameters
	b_grad = 2 * error.mean()
	w_grad = 2 * (x_train * error).mean()
	
	#但是这里已经可以使用自动计算的方法
	loss.backward()
	#可以直接读取当前的梯度的值
	b.grad
	w.grad

动态计算图

可以使用动态计算图,直接看到当前的所有的梯度的变量的相互之间的关系,这里大家可以自己看,我就不放出来了

优化器

我们之前都要手动的执行backward,然后获取b,w的grad,然后手动的进行更新使用了learning rate,最后还需要zero_grad。我们可以通过优化器的方式将变量一开始就加入到优化器中

	optimizer = optim.SGD([b,w],lr = lr)
	#	....执行学习的步骤
	loss.backward()
	
	optimizer.step()
	#一次性更新所有的参数的变量
	
	optimizer.zero_grad()
	#一次性zero所有的变量的值

损失函数

实际上,我们的损失函数也有torch的封装,可以直接使用已经配置好的损失函数
loss_fn = nn.MSELoss(reduction = ‘mean’)

损失函数直接得到我们的yhat的结果和y_lable之间的损失的值

	#原文中的损失函数的部分
	error = (yhat - y_train_tensor)
	loss = (error**2).mean()

	#取代后
	loss_fn = nn.MSELoss(reduction='mean')
	loss = loss_fn(y_hat,y_train_tensor)
	
	此时的loss计算出来的结果和之前的是一模一样的

需要注意的是,如果此时需要回归的numpy,需要执行detach()操作,表示Loss函数不再参与grad的计算

模型

我们已经有了优化器(用于更新参数),损失函数(用于生成损失值),我们还可以定义自己的module模型(显然吗,我们还需要构建很多的我们自己的东西)

我们使用model函数来用现有的模型针对输入得到我们的输出函数,model函数对比forward函数,还会 前向和反向的钩子函数
我们声明一个继承与Module的自己的类

class ManualLinearRegression(nn.Module):
    def __init__(self):
        super().__init__()
        # To make "b" and "w" real parameters of the model,
        # we need to wrap them with nn.Parameter
        self.b = nn.Parameter(torch.randn(1,
                                          requires_grad=True, 
                                          dtype=torch.float))
        self.w = nn.Parameter(torch.randn(1, 
                                          requires_grad=True,
                                          dtype=torch.float))
        
    def forward(self, x):
        # Computes the outputs / predictions
        return self.b + self.w * x

通过parameters的函数,我们可以得到一个module类的目前包含的参数

dummpy = ManualLinearRegression()
list(dummy.parameters)
[Parameter containing:
 tensor([0.3367], requires_grad=True), Parameter containing:
 tensor([0.1288], requires_grad=True)]

也可以state_dict来获取所有的参数的当前值,和parameters的区别在于state_dict通常用于加载和保存模型,而前面的通常用于展示优化器的包含的变量
注意,数据和模型需要在同一个设备

dummy = ManualLinearRegression().to(device)

阶段代码

经过我们的使用自己的类以后的代码可以优化如下

# Greek letter
lr = 0.1

# Step 0 - Initializes parameters "b" and "w" randomly
torch.manual_seed(42)
# Now we can create a model and send it at once to the device
model = ManualLinearRegression().to(device)

# Defines a SGD optimizer to update the parameters 
# (now retrieved directly from the model)
optimizer = optim.SGD(model.parameters(), lr=lr)

# Defines a MSE loss function
loss_fn = nn.MSELoss(reduction='mean')

# Defines number of epochs
n_epochs = 1000

for epoch in range(n_epochs):
    model.train() # What is this?!?

    # Step 1 - Computes model's predicted output - forward pass
    # No more manual prediction,直接使
    yhat = model(x_train_tensor)
    # 一定注意这里使用的是model而不是forward
    
    # Step 2 - Computes the loss
    loss = loss_fn(yhat, y_train_tensor)

    # Step 3 - Computes gradients for both "b" and "w" parameters
    loss.backward()
    
    # Step 4 - Updates parameters using gradients and
    # the learning rate
    optimizer.step()
    optimizer.zero_grad()
    
# We can also inspect its parameters using its state_dict
print(model.state_dict())

model.train()

这个是配置了训练模式,训练模式可以有很多内容,例如我们常见的dropout的优化模式来减少过拟合的问题

嵌套模型

我们可以使用了已经有的模型来作为嵌套的模型

首先我们使用nn自带的Linear来取代我们手写的线性的Module,如下是一个具有一个权重w,和一个bias的线性模型,我们完全可以用这个来取代我们之前的手写的类

linear = nn.Linear(1,1)

当然我们可以嵌套这个到我们自己的类,这个和之前的基本上是完全等价的

class MyLinearRegression(nn.Module):
    def __init__(self):
        super().__init__()
        # Instead of our custom parameters, we use a Linear model
        # with single input and single output
        self.linear = nn.Linear(1, 1)
                
    def forward(self, x):
        # Now it only takes a call
        self.linear(x)

序列模型

一个好的深度学习的模型,显然不只有一层,通常都会有很多隐藏层,将所有的封装子在一起,我们可以认为是一个序列模型

# Building the model from the figure above
model = nn.Sequential(nn.Linear(3, 5), nn.Linear(5, 1)).to(device)

model.state_dict()

注意这里创建了一个序列类,序列类里面有两层,第一层是一个35的输出,第二层是一个51的模型,或者我们可以通过添加层的方式来增加带名字的层

# Building the model from the figure above
model = nn.Sequential()
model.add_module('layer1', nn.Linear(3, 5))
model.add_module('layer2', nn.Linear(5, 1))
model.to(device)

这里其实已经涉及到了深度学习的部分,pytorch的层只是很多例如
卷积层、池化层、填充层、非线性激活层、归一化层、循环层、transformer层、线性层、丢弃层、稀疏层、视觉层、数据平移(多GPU),展平层

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2280869.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Springboot自动配置的原理

先拿redis来举个例子 第一步导入依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> 第二步配置 spring: redis: database:host:127.0.0.1 port…

【动态规划】落花人独立,微雨燕双飞 - 8. 01背包问题

本篇博客给大家带来的是01背包问题之动态规划解法技巧. &#x1f40e;文章专栏: 动态规划 &#x1f680;若有问题 评论区见 ❤ 欢迎大家点赞 评论 收藏 分享 如果你不知道分享给谁,那就分享给薯条. 你们的支持是我不断创作的动力 . 王子,公主请阅&#x1f680; 要开心要快乐顺便…

记一次虚机上传过慢问题排查

最近线上虚机有个特殊的用户反馈&#xff0c;用户反馈虚机从A服务器下载文件特别慢&#xff0c;于是scp A服务器数据到本地client&#xff0c;发现 只有几十K的流量。 当时第一反应怀疑是虚机负载压力比较大&#xff0c;但是查看虚机IO以及负载都很低。。。。 然后tcpdump抓包发…

web服务器 网站部署的架构

WEB服务器工作原理 Web web是WWW(World Wide Web)的简称&#xff0c;基本原理是&#xff1a;请求(客户端)与响应(服务器端)原理&#xff0c;由遍布在互联网中的Web服务器和安装了Web浏览器的计算机组成 客户端发出请求的方式&#xff1a;地址栏请求、超链接请求、表单请求 …

数据结构——实验七·排序

嗨~~欢迎来到Tubishu的博客&#x1f338;如果你也是一名在校大学生&#xff0c;正在寻找各种编程资源&#xff0c;那么你就来对地方啦&#x1f31f; Tubishu是一名计算机本科生&#xff0c;会不定期整理和分享学习中的优质资源&#xff0c;希望能为你的编程之路添砖加瓦⭐&…

Windows系统提示RunDLL PcaWallpaperAppDetect错误修复方法

最近&#xff0c;Win11 24H2预览版和Win10 LTSC 2025功能更新偶尔会触发RunDLL错误弹窗 具体表现为 //英文提示 Error in C:\WINDOWS\system32\PcaSvc.dll Missing entry: PcaWallpaperAppDetect//中文提示 C:\WINDOWS\system32\PcaSvc.dll出错 丢失条目:PcaWallpaperAppDe…

计算机组成原理——数据表示(二)

当生活的压力和困惑缠绕在身边&#xff0c;我们往往需要振奋精神&#xff0c;勇往直前。无论在何种困境中&#xff0c;我们都要保持积极的态度和坚定的信念。将悲观的情绪抛之脑后&#xff0c;展现出坚强的意志力和无尽的活力。振奋精神意味着我们要战胜自己内心的负面情绪&…

人源化抗体的改造方式及其优势【卡梅德生物】

随着生物制药行业的迅速发展&#xff0c;抗体药物已经成为治疗多种疾病&#xff08;尤其是癌症、免疫性疾病等&#xff09;的重要手段。抗体人源化改造技术作为抗体药物研发的关键技术之一&#xff0c;在提高药物疗效和降低免疫原性方面发挥了至关重要的作用。 1. 人源化抗体的…

【Linux】深刻理解动静态库

1.什么是库 库是写好的现有的&#xff0c;成熟的&#xff0c;可以复⽤的代码。现实中每个程序都要依赖很多基础的底层库&#xff0c;不可能每个⼈的代码都从零开始&#xff0c;因此库的存在意义⾮同寻常。本质上来说库是⼀种可执⾏代码的⼆进制形式&#xff0c;可以被操作系统载…

【java数据结构】其他非基于比较排序

【java数据结构】其他非基于比较排序 一、计数排序二、基数排序三、桶排序 博客最后附有整篇博客的全部代码&#xff01;&#xff01;&#xff01; 一、计数排序 场景&#xff1a;集中在某个范围内的一组数据 思路&#xff1a; 找到这组序列的最大值和最小值&#xff0c;通过…

博客之星2024年度总评选——我的年度创作回顾与总结

2024年&#xff0c;是我在CSDN博客上持续耕耘、不断成长的一年。在此&#xff0c;与大家分享一下我的年度创作回顾与总结。 一、创作成长与突破 在人工智能领域&#xff0c;技术迭代迅速&#xff0c;知识更新频繁。为了保持自己的竞争力&#xff0c;在今年&#xff0c;我始终…

ChromeOS 132 版本更新

ChromeOS 132 版本更新 1. 企业定制化 Chrome Web Store 管理员现在可以使用新设置定制 Chrome Web Store 以适应他们管理的用户&#xff0c;包括以下功能&#xff1a; 添加公司标志添加首页横幅和自定义公告策划扩展集合实施基于类别的控制 这些设置可以通过管理员控制台进…

Golang Gin系列-5:数据模型和数据库

在这篇Gin教程的博客中&#xff0c;我们将探索如何将模型和数据库与Gin框架无缝集成&#xff0c;使你能够构建健壮且可扩展的web应用程序。通过利用流行的库并遵循最佳实践&#xff0c;你将学习如何定义模型、建立数据库连接、执行CRUD操作以及确保基于gin的项目中的数据完整性…

计算机毕业设计hadoop+spark股票基金推荐系统 股票基金预测系统 股票基金可视化系统 股票基金数据分析 股票基金大数据 股票基金爬虫

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

设计模式的艺术-代理模式

结构性模式的名称、定义、学习难度和使用频率如下表所示&#xff1a; 1.如何理解代理模式 代理模式&#xff08;Proxy Pattern&#xff09;&#xff1a;给某一个对象提供一个代理&#xff0c;并由代理对象控制对原对象的引用。代理模式是一种对象结构型模式。 代理模式类型较多…

Spring Boot整合Thymeleaf、JDBC Template与MyBatis配置详解

本文将详细介绍如何在Spring Boot项目中整合Thymeleaf模板引擎、JDBC Template和MyBatis&#xff0c;涵盖YAML配置、依赖版本匹配、项目结构设计及代码示例。 一、版本兼容性说明 Spring Boot版本与Java版本对应关系 Spring Boot 2.x&#xff1a;支持Java 8、11&#xff08;推…

【博客之星】2024年度创作成长总结 - 面朝大海 ,春暖花开!

没关系的&#xff0c;大家都会做错选择&#xff0c;会 莫名其妙掉眼泪&#xff0c;走在路上会突然崩溃&#xff0c; 但这并不影响我们去看看晚霞&#xff0c; 再次爱上这个世界。 面朝大海 &#xff0c;春暖花开! about meReviewLife about me 现在我是一名24级计算机类的…

StyleMaster: Stylize Your Video with Artistic Generation and Translation 论文解读

目录 一、概述 二、相关工作 1、图像风格化 2、视频风格化 三、StyleMaster 1、创建对比数据集 2、提取全局描述子 3、局部描述和全局描述结合 4、时间和风格质量的运动适配器 5、Gray Tile ControlNet 四、实验 一、概述 Our StyleMaster demonstrates superior vi…

c++进阶---c++三大特性之一---多态

多态的简单介绍&#xff1a;是一种动态的访问函数&#xff0c;比如&#xff1a;你定义了一个一个人类和一个学生类&#xff0c;当你传入的是学生类的时候&#xff0c;你需要有购物优惠&#xff0c;这种情境下用多态就很适用。 1.简单的多态使用&#xff1a; 1.1构造多态的条件…

安卓程序作为web服务端的技术实现(二):Room 实现数据存储

已经实现web服务器安卓程序作为web服务端的技术实现&#xff1a;AndServer 实现登录权限拦截-CSDN博客 现在需要和正常web项目类似&#xff0c;那么就需要操作数据库 一般web项目都是选择较为重型的数据库如MySQL&#xff0c;SQL server等 这里是安卓项目&#xff0c;我目前…