鞋子,靴子,拖鞋傻傻分不清楚 pytorch实现分类 入门小案例
创始人
2024-04-06 20:41:49
0

鞋子,靴子,拖鞋傻傻分不清楚 pytorch入门

  • 前言
  • 方法
    • 网络
    • 优化器
    • 损失函数
    • 总体方法
  • 代码实现
    • 图片加字
    • 神经网络
  • 总结

前言

从入学到现在已经两个多月了,看了一个多月的论文不知道学到了啥
在这里插入图片描述
正好最近看了看pytorch的入门,像休息休息,就想着写个分类玩玩吧,但不知道写啥,突然见看到一个数据集网站,有一个鞋子的数据集
在这里插入图片描述
这对我这种非常like鞋的人来说很有吸引力,那来整个鞋子分类吧。

在这里插入图片描述

方法

网络

这里我们选用的网络是DenseNet,相比于普通的CNN来说,Densenet可以使用各层提取的特征,从而避免特征的丢失,同样,我们再进行分类也希望尽可能多的特征为我们所用,Densenet的网络结构如下
在这里插入图片描述

优化器

使用的是随机梯度下降优化器,其中学习率设置为0.001,动量为0.5
具体的内容可以参考该博客详解随机梯度下降法(Stochastic Gradient Descent,SGD)

损失函数

使用的是交叉熵损失函数

这里多说几句,最开始我看到这个损失的时候一脸懵逼,咋着,一个数还能和一个向量比较一下子
在这里插入图片描述
直到后面我看到b站的讲解,好吧,可能!

在这里插入图片描述
这就是b站里给到的公式

这里的x是指图像真实的类别,class是指图像在该类别的得分,x[j]是指所有类别在预测后的得分

以我们要介绍的网络为例,因为要判断的只有三类,鞋子,靴子和凉鞋,则我们最终的输出维度是3,即最终会输出一个向量,这个向量有三个值,分别代表分为鞋子,靴子和凉鞋的概率得分,越大就表示图片属于这一类的可能性越大。

例如我们输出的结果为[0.1 , 0.2, 0.8]

假如我们要输入的图片类别是第2类(从0开始算),那么带入上面公式就是

-2*0.8+log(e(0.1)+e(0.2)+e(0.3))

则当上面公式越小时,越接近真实结果

总体方法

这样就很简单了,就是利用我们现有的1.5万照片去训练该网络(这里做的比较糙,没有设置验证集和测试集),采用随机梯度下降的方式进行训练,每次训练的图片数量为10,所有图片被作为输入训练一次后为一个epoch,总共训练50个epoch,训练结束后就是我们需要的模型了。

代码实现

图片加字

在判断类别后在图片上加上类别,方便看

from PIL import ImageFont, ImageDraw, Image
import numpy as np
import cv2def settags(info, img):url = imgimg_cv = cv2.imread(img)img = Image.fromarray(img_cv)font1 = ImageFont.truetype("./simsun.ttc", 100)draw = ImageDraw.Draw(img)draw.text((10, 10), info, font=font1, fill=(0, 0, 255))img1 = np.array(img)cv2.imwrite('r' + url, img1)

神经网络

卷积层


class ConvLayer(torch.nn.Module):def __init__(self, in_channels, out_channels, kernel_size, stride, is_last=False):super(ConvLayer, self).__init__()reflection_padding = int(np.floor(kernel_size / 2))self.reflection_pad = nn.ReflectionPad2d(reflection_padding)self.conv2d = nn.Conv2d(in_channels, out_channels, kernel_size, stride)self.dropout = nn.Dropout2d(p=0.5)self.is_last = is_lastdef forward(self, x):# 图片进行填充 保证输出都是相同大小,从而才能使后面的层也使用前面的特征out = self.reflection_pad(x)out = self.conv2d(out)if self.is_last is False:out = F.leaky_relu(out, inplace=True)return out

单个densenet

# Dense convolution unit
class DenseConv2d(torch.nn.Module):def __init__(self, in_channels, out_channels, kernel_size, stride, is_Last=False):super(DenseConv2d, self).__init__()self.dense_conv = ConvLayer(in_channels, out_channels, kernel_size, stride, is_Last)self.is_last = is_Lastdef forward(self, x):out = self.dense_conv(x)if self.is_last == False:# 按第二个维度进行拼接    为了实现densenetout = torch.cat([x, out], 1)return out

所有DenseNet

out_channels_def = 16denseblock = []# densenetdenseblock += [DenseConv2d(in_channels, out_channels_def, kernel_size, stride),DenseConv2d(in_channels + out_channels_def, out_channels_def, kernel_size, stride),DenseConv2d(in_channels + out_channels_def * 2, out_channels_def, kernel_size, stride),DenseConv2d(in_channels + out_channels_def * 3, out_channels_def, kernel_size, stride),DenseConv2d(in_channels + out_channels_def * 4, out_channels_def, kernel_size, stride),DenseConv2d(in_channels + out_channels_def * 5, out_channels_def, kernel_size, stride),DenseConv2d(in_channels + out_channels_def * 6, out_channels_def, kernel_size, stride, True)]self.denseblock = nn.Sequential(*denseblock)

训练

def train():net = Net(3, 3, 1)net.cuda()net.train()# 训练cirterion = nn.CrossEntropyLoss()optimizer = optim.SGD(net.parameters(), lr=0.001, momentum=0.5)for epoch in range(50):running_loss = 0.0for i, data in enumerate(train_loader, 0):inputs, labels = datainputs, labels = Variable(inputs), Variable(labels)inputs = torch.tensor(inputs)labels = torch.tensor(labels)inputs = inputs.cuda()labels = labels.cuda()optimizer.zero_grad()  # 优化器清零outputs = net(inputs)loss = cirterion(outputs, labels)loss.backward()optimizer.step()  # 优化running_loss += loss.item()if i % 5 == 0:print('[%d %5d] loss: %.3f' % (epoch + 1, i + 1, running_loss / 200))running_loss = 0.0# 每个epoch存储一个模型torch.save(net, 'shoenet' + epoch.__str__() + '.pth')torch.save(net, 'shoenet.pth')

总结

第一次写神经网络的代码,花了好久才搞出来,虽然很多代码都是照葫芦画瓢,但对我这种小白来说确实蛮难的,但总归是搞出来了,来看下结果把。
在这里插入图片描述
可以看到哈,分类还是ok 的,那就到这了,溜了溜了

相关内容

热门资讯

银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
AWSElasticBeans... 在Dockerfile中手动配置nginx反向代理。例如,在Dockerfile中添加以下代码:FR...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
月入8000+的steam搬砖... 大家好,我是阿阳 今天要给大家介绍的是 steam 游戏搬砖项目,目前...
​ToDesk 远程工具安装及... 目录 前言 ToDesk 优势 ToDesk 下载安装 ToDesk 功能展示 文件传输 设备链接 ...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
AWS管理控制台菜单和权限 要在AWS管理控制台中创建菜单和权限,您可以使用AWS Identity and Access Ma...