前言

随着人工智能的不断发展,机器学习这门技术也越来越重要,很多人都开启了学习机器学习,本文就介绍了机器学习的基础内容。来源于哔哩哔哩博主“霹雳吧啦Wz”,博主学习作为笔记记录,欢迎大家一起讨论学习交流。

一、搭建MobileNet网络

用MobileNetv2学习,轮次训练5轮次(代码),五个epoch2分钟,设备太重要了(服务器显卡P4000,网上说相当于1070)。

二、代码部分

1.module.py----定义MobileNet的网络结构

代码如下(示例):

from torch import nn
import torchdef _make_divisible(ch, divisor=8, min_ch=None):"""This function is taken from the original tf repo.It ensures that all layers have a channel number that is divisible by 8It can be seen here:https://github.com/tensorflow/models/blob/master/research/slim/nets/mobilenet/mobilenet.py"""if min_ch is None:min_ch = divisornew_ch = max(min_ch, int(ch + divisor / 2) // divisor * divisor)# Make sure that round down does not go down by more than 10%.if new_ch < 0.9 * ch:new_ch += divisorreturn new_chclass ConvBNReLU(nn.Sequential):def __init__(self, in_channel, out_channel, kernel_size=3, stride=1, groups=1):padding = (kernel_size - 1) // 2super(ConvBNReLU, self).__init__(nn.Conv2d(in_channel, out_channel, kernel_size, stride, padding, groups=groups, bias=False),nn.BatchNorm2d(out_channel),nn.ReLU6(inplace=True))class InvertedResidual(nn.Module):def __init__(self, in_channel, out_channel, stride, expand_ratio):super(InvertedResidual, self).__init__()hidden_channel = in_channel * expand_ratioself.use_shortcut = stride == 1 and in_channel == out_channellayers = []if expand_ratio != 1:# 1x1 pointwise convlayers.append(ConvBNReLU(in_channel, hidden_channel, kernel_size=1))layers.extend([# 3x3 depthwise convConvBNReLU(hidden_channel, hidden_channel, stride=stride, groups=hidden_channel),# 1x1 pointwise conv(linear)nn.Conv2d(hidden_channel, out_channel, kernel_size=1, bias=False),nn.BatchNorm2d(out_channel),])self.conv = nn.Sequential(*layers)def forward(self, x):if self.use_shortcut:return x + self.conv(x)else:return self.conv(x)class MobileNetV2(nn.Module):def __init__(self, num_classes=1000, alpha=1.0, round_nearest=8):super(MobileNetV2, self).__init__()block = InvertedResidualinput_channel = _make_divisible(32 * alpha, round_nearest)last_channel = _make_divisible(1280 * alpha, round_nearest)inverted_residual_setting = [# t, c, n, s[1, 16, 1, 1],[6, 24, 2, 2],[6, 32, 3, 2],[6, 64, 4, 2],[6, 96, 3, 1],[6, 160, 3, 2],[6, 320, 1, 1],]features = []# conv1 layerfeatures.append(ConvBNReLU(3, input_channel, stride=2))# building inverted residual residual blockesfor t, c, n, s in inverted_residual_setting:output_channel = _make_divisible(c * alpha, round_nearest)for i in range(n):stride = s if i == 0 else 1features.append(block(input_channel, output_channel, stride, expand_ratio=t))input_channel = output_channel# building last several layersfeatures.append(ConvBNReLU(input_channel, last_channel, 1))# combine feature layersself.features = nn.Sequential(*features)# building classifierself.avgpool = nn.AdaptiveAvgPool2d((1, 1))self.classifier = nn.Sequential(nn.Dropout(0.2),nn.Linear(last_channel, num_classes))# weight initializationfor m in self.modules():if isinstance(m, nn.Conv2d):nn.init.kaiming_normal_(m.weight, mode='fan_out')if m.bias is not None:nn.init.zeros_(m.bias)elif isinstance(m, nn.BatchNorm2d):nn.init.ones_(m.weight)nn.init.zeros_(m.bias)elif isinstance(m, nn.Linear):nn.init.normal_(m.weight, 0, 0.01)nn.init.zeros_(m.bias)def forward(self, x):x = self.features(x)x = self.avgpool(x)x = torch.flatten(x, 1)x = self.classifier(x)return x

2.train.py----加载数据集并进行训练,训练集计算loss,测试集计算accuracy,保存训练好的网络参数

代码如下(示例):

import os
import sys
import jsonimport torch
import torch.nn as nn
import torch.optim as optim
from torchvision import transforms, datasets
from tqdm import tqdmfrom model_v2 import MobileNetV2def main():device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")print("using {} device.".format(device))batch_size = 16epochs = 5data_transform = {"train": transforms.Compose([transforms.RandomResizedCrop(224),transforms.RandomHorizontalFlip(),transforms.ToTensor(),transforms.Normalize([0.485, 0.456, 0.406], [0.229, 0.224, 0.225])]),"val": transforms.Compose([transforms.Resize(256),transforms.CenterCrop(224),transforms.ToTensor(),transforms.Normalize([0.485, 0.456, 0.406], [0.229, 0.224, 0.225])])}data_root = os.path.abspath(os.path.join(os.getcwd(), "../.."))  # get data root pathimage_path = os.path.join(data_root, "data_set", "flower_data")  # flower data set pathassert os.path.exists(image_path), "{} path does not exist.".format(image_path)train_dataset = datasets.ImageFolder(root=os.path.join(image_path, "train"),transform=data_transform["train"])train_num = len(train_dataset)# {'daisy':0, 'dandelion':1, 'roses':2, 'sunflower':3, 'tulips':4}flower_list = train_dataset.class_to_idxcla_dict = dict((val, key) for key, val in flower_list.items())# write dict into json filejson_str = json.dumps(cla_dict, indent=4)with open('class_indices.json', 'w') as json_file:json_file.write(json_str)nw = min([os.cpu_count(), batch_size if batch_size > 1 else 0, 8])  # number of workersprint('Using {} dataloader workers every process'.format(nw))train_loader = torch.utils.data.DataLoader(train_dataset,batch_size=batch_size, shuffle=True,num_workers=nw)validate_dataset = datasets.ImageFolder(root=os.path.join(image_path, "val"),transform=data_transform["val"])val_num = len(validate_dataset)validate_loader = torch.utils.data.DataLoader(validate_dataset,batch_size=batch_size, shuffle=False,num_workers=nw)print("using {} images for training, {} images for validation.".format(train_num,val_num))# create modelnet = MobileNetV2(num_classes=5)# load pretrain weights# download url: https://download.pytorch.org/models/mobilenet_v2-b0353104.pthmodel_weight_path = "./mobilenet_v2.pth"assert os.path.exists(model_weight_path), "file {} dose not exist.".format(model_weight_path)pre_weights = torch.load(model_weight_path, map_location='cpu')# delete classifier weightspre_dict = {k: v for k, v in pre_weights.items() if net.state_dict()[k].numel() == v.numel()}missing_keys, unexpected_keys = net.load_state_dict(pre_dict, strict=False)# freeze features weightsfor param in net.features.parameters():param.requires_grad = Falsenet.to(device)# define loss functionloss_function = nn.CrossEntropyLoss()# construct an optimizerparams = [p for p in net.parameters() if p.requires_grad]optimizer = optim.Adam(params, lr=0.0001)best_acc = 0.0save_path = './MobileNetV2.pth'train_steps = len(train_loader)for epoch in range(epochs):# trainnet.train()running_loss = 0.0train_bar = tqdm(train_loader, file=sys.stdout)for step, data in enumerate(train_bar):images, labels = dataoptimizer.zero_grad()logits = net(images.to(device))loss = loss_function(logits, labels.to(device))loss.backward()optimizer.step()# print statisticsrunning_loss += loss.item()train_bar.desc = "train epoch[{}/{}] loss:{:.3f}".format(epoch + 1,epochs,loss)# validatenet.eval()acc = 0.0  # accumulate accurate number / epochwith torch.no_grad():val_bar = tqdm(validate_loader, file=sys.stdout)for val_data in val_bar:val_images, val_labels = val_dataoutputs = net(val_images.to(device))# loss = loss_function(outputs, test_labels)predict_y = torch.max(outputs, dim=1)[1]acc += torch.eq(predict_y, val_labels.to(device)).sum().item()val_bar.desc = "valid epoch[{}/{}]".format(epoch + 1,epochs)val_accurate = acc / val_numprint('[epoch %d] train_loss: %.3f  val_accuracy: %.3f' %(epoch + 1, running_loss / train_steps, val_accurate))if val_accurate > best_acc:best_acc = val_accuratetorch.save(net.state_dict(), save_path)print('Finished Training')if __name__ == '__main__':main()

和之前train.py代码有不一样:预训练模型在IMAGENET训练得到的权重最后全连接展开是1000个类别,我们数据集是用的5个类别,所以要将其delete classifier weights并使训练效果更好,进行freeze features weights。

预训练权重下载方式:在train.py代码中输入:import torchvision.models.mobilenet进行查看。官方预训练权重链接

3.predict.py——得到训练好的网络参数后,用自己找的图像进行分类测试

import os
import jsonimport torch
from PIL import Image
from torchvision import transforms
import matplotlib.pyplot as pltfrom model_v2 import MobileNetV2def main():device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")data_transform = transforms.Compose([transforms.Resize(256),transforms.CenterCrop(224),transforms.ToTensor(),transforms.Normalize([0.485, 0.456, 0.406], [0.229, 0.224, 0.225])])# load imageimg_path = "../tulip.jpg"assert os.path.exists(img_path), "file: '{}' dose not exist.".format(img_path)img = Image.open(img_path)plt.imshow(img)# [N, C, H, W]img = data_transform(img)# expand batch dimensionimg = torch.unsqueeze(img, dim=0)# read class_indictjson_path = './class_indices.json'assert os.path.exists(json_path), "file: '{}' dose not exist.".format(json_path)with open(json_path, "r") as f:class_indict = json.load(f)# create modelmodel = MobileNetV2(num_classes=5).to(device)# load model weightsmodel_weight_path = "./MobileNetV2.pth"model.load_state_dict(torch.load(model_weight_path, map_location=device))model.eval()with torch.no_grad():# predict classoutput = torch.squeeze(model(img.to(device))).cpu()predict = torch.softmax(output, dim=0)predict_cla = torch.argmax(predict).numpy()print_res = "class: {}   prob: {:.3}".format(class_indict[str(predict_cla)],predict[predict_cla].numpy())plt.title(print_res)for i in range(len(predict)):print("class: {:10}   prob: {:.3}".format(class_indict[str(i)],predict[i].numpy()))plt.show()if __name__ == '__main__':main()

三、MobileNetv1

VGG16权重文件大概有490MB,Resnet152权重文件大概有644MB,对于移动设备和嵌入式不大现实。

MobileNet就是一个轻量级网络:精确度降低非常少,可以忽略的前提,模型的参数可以减少原来的32倍。也是牛逼,自我感觉,就是深度学习一直都是在刷榜单,我们要top最好的精确度,看谁是老大,但是MobileNet感觉是在首先在参数上进行创新,而且不影响精确度,就是走了一个极少数当时不热的方向,结果证明是可行的。(通过DW卷积实现减少模型参数)

总结:MoileNet就是轻量级网络 (通过DW卷积实现减少模型参数)

传统卷积:卷积核的深度=输入特征矩阵的深度;输出矩阵的通道数=卷积核的个数

DW卷积:卷积核的深度=1;输入矩阵的通道数=卷积核的个数=输出特征矩阵的通道数(就是不改变输入和输出的通道数)

总结:DW卷积是卷积核的深度=1(1个卷积核只负责一个输入图像的深度);输入矩阵的通道数=卷积核的个数=输出特征矩阵的通道数(就是不改变输入和输出的通道数)

深度可分离卷积就是由DW卷积和PW卷积组成,其中PW卷积就是普通卷积,只不过卷积核大小是=1.(所以也满足普通卷积的特点:卷积核的深度与输入特征的深度一样,卷积核的个数与输出特征矩阵的通道数一样)

通常DW卷积是和PW卷积一起使用的,使用这种组合可以对比于普通的卷积,大大减少参数的参与。

普通卷积:卷积核大小(DK DK)x输入特征矩阵深度(M)x输出特征矩阵深度(N)x输入特征矩阵大小(DF DF)

DW+PW卷积:卷积核大小(DK)x输入特征矩阵深度(1)x输出特征矩阵深度(M)x输入特征矩阵大小(DF DF)+卷积核大小(1X1)x输入特征矩阵深度(M)x输出特征矩阵深度(N)x输入特征矩阵大小(DF DF)

总结:理论上普通卷积计算量是DW+PW的8-9倍。

普通卷积Conv 步距 3x3x3x32 前两个3是卷积核的尺寸,第三个3是输入图像的深度,第四个3是卷积核的个数。DW卷积深度是1.

a是卷积核个数的倍率,β是分辨率参数。

四、MobileNetv2

优点:倒残差结构inverted residuals+linear bottlenecks,这个两个优点就是MobileNetV2的论文题目。

ResNet残差结构是先是1x1降维,然后是3x3卷积处理,最后1x1卷积升维。(“V”,两头大,中间小)

Inverted Residual Block是先1x1升维,然后是3x3卷积DW,最后是1x1卷积降维。(^,两头小,中间大)

激活函数也不一样,一个是Relu,一个是Relu6

Relu6L:当输入小于0时候,置为0;当输入大于0小于6时候,相当于y=x,大于6时候,置为0。

Relu激活函数对低维度特征信息造成大量损失。

MobileNetv2先1x1升维卷积(Relu6),然后是3x3卷积DW(Relu6),最后是1x1卷积降维(Linear线性激活)。(^,两头小,中间大)

t是扩展因子,前一个卷积得到的输出矩阵的深度是后一个卷积的输入矩阵的深度,DW卷积不改变输入和输出的通道数。

当步距=1时候且输入特征矩阵与输出矩阵的shape相同时候才有shortcut连接。

t扩展因子,c是输出特征矩阵的深度channel,n是bottleneck(倒残差结构)重复次数。s是步距(一个block由一系列bottleneck组成):比如bottleneck,n=2时候,bottleneck重复两遍,对他而言,第一层bottleneck的步距为2,第二层为1.

当步距=1时候且输入特征矩阵与输出矩阵的shape相同时候才有shortcut连接。上图的block,他采用了三个bottleneck结构,stride=1,按照论文应该有捷径分支,但实际上是没有的。因为输入特征矩阵深度是=64,而输出特征矩阵深度是=96,深度不一样,不可以直接相加,即没有捷径分支。

当步距=1时候且输入特征矩阵与输出矩阵的shape相同时候才有shortcut连接。上图的block,他采用了三个bottleneck结构,stride=1,按照论文应该有捷径分支,但实际上是没有的。因为输入特征矩阵深度是=64,而输出特征矩阵深度是=96,深度不一样,不可以直接相加,即没有捷径分支。

对于第二层bottleneck而言,步距=1,输入特征矩阵深度=上一层输出特征矩阵深度96,输出特征矩阵=96,宽高也不发生变化,可以有捷径分支。

性能对比在CPU上,移动设备实时处理。感觉就是也有点神奇。

五、MobileNetv3

MobileNet网络,更新Block(bneck)

MobileNetV3在Image Net上比MobileNetV2性能提高20%,在延时上比MoblieNetV2延时少6.6%。

总结:MobileNetV3在精确度和延时上比较好。

MobileNetV2就是先是1x1卷积升维(BN+Relu6),然后再3x3卷积DW卷积(BN+Relu6),最后1x1卷积降维(Relu6)。

1.增加了SE模块(注意力机制):针对特征矩阵的每一个channel进行池化处理,channel是多少,一维向量就有多少个元素。再通过两个全连接层得到输出向量。第一个全连接层节点数目是通道数的1/4,第二个全连接层节点数目是和通道数保持一致。最后输出的是和每一个通道数有关系,赋予不同的权重,即可实现SE模块的功能。


1.SE模块(注意力机制)

2.NL非线性激活函数,最后的1x1的卷积层没有经过激活函数处理

重新设计耗时层结构:

1.减少第一个卷积层的个数,由32变为16准确度差不多。但是参数减少和时间减少。

2.Efficient Last Stage减少运算参数。

SWISH函数计算求导不方便,h-swish函数比较方便。可以用h-sigmoid函数替代sigmoid函数,两个曲线相似。

输入的shape out=输出特征矩阵Chanel ;bneck3x3对应的是Dwise卷积的大小 ;exp size代表第一个升维的1x1卷积将维度升到多少维度。

第七课:MobileNetv1、MobileNetv2、MobileNetv3学习相关推荐

  1. 投资学习网课笔记(part7)--基金第七课

    学习笔记,仅供参考 文章目录 基金第七课 宽基指数和行业指数 市值加权型和策略加权型 常见的市值加权型指数 常见的策略加权型指数 中证系列指数基金 创业板指数基金 红利类指数基金 基本面类指数基金 价 ...

  2. Python学习,第七课(灵活使用Frame,让布局更舒适)

    Python学习第七课(让界面布局舒适,是一个长久的工作) 一入布局深似海,加一减一都很难 基础知识 尝试布局 尝试好布局,感受下元素带来的不同 改造主程序,细节还是要优化 细节优化,细枝末节的参数 ...

  3. Python学习第七课-元组字典集合

    Python学习第七课-元组&字典&集合 一.元组 1.1元组简介 1.2元组的拆包 1.3字符串拆包 1.4列表的拆包 1.5拆包练习 二.字典的简介 2.1介绍 2.2操作 三.字 ...

  4. 第七课 大数据技术之Fink1.13的实战学习-Fink CEP

    第七课 大数据技术之Fink1.13的实战学习-Fink CEP 文章目录 第七课 大数据技术之Fink1.13的实战学习-Fink CEP 第一节 Fink CEP介绍 1.1 Flink CEP背 ...

  5. 2011斯坦福大学iOS应用开发教程学习笔记(第七课)Pad和iPhone的通用程序

    第七课的主要内容:iPad 和iPhone的通用程序 这节课主要讲如何在一个程序里适配iPad 和iPhone. 内容简介 1.UIToolbar上面放的都是UIBarButtonItem 可以参考i ...

  6. 斯坦福大学iOS应用开发教程学习笔记(第七课)Pad和iPhone的通用程序

    第七课的主要内容:iPad 和iPhone的通用程序 这节课主要讲如何在一个程序里适配iPad 和iPhone. 内容简介 1.UIToolbar上面放的都是UIBarButtonItem 可以参考i ...

  7. 第七课.简单的图像分类(一)

    第七课目录 图像分类基础 卷积神经网络 Pooling layer BatchNormalization BatchNormalization与归一化 torch.nn.BatchNorm2d MNI ...

  8. Coursera公开课笔记: 斯坦福大学机器学习第七课“正则化(Regularization)”

     Coursera公开课笔记: 斯坦福大学机器学习第七课"正则化(Regularization)" +13投票 斯坦福大学机器学习第七课"正则化"学习笔记, ...

  9. 【Cocos游戏实战】功夫小子第七课之游戏主功能场景逻辑功能和暂停功能场景的分析和实现...

    CSDN的markdown编辑器是吃屎了么! !.什么玩意.!写了一半写不了东西还全没了,搞个毛线! 本节课的视频教程地址是:第七课在此  假设本教程有帮助到您,希望您能点击进去观看一下,并且如今注冊 ...

  10. 《迅雷链精品课》第七课:以太坊数据存储分析

    上一节课我们学习了比特币的区块链数据存储,接着前一篇的内容,我们继续了解以太坊的相关内容.业界一直把以太坊认为是区块链发展进程中2.0的代表,因为它在比特币的基础上增加了图灵完备的智能合约,扩展了区块 ...

最新文章

  1. Latex中设置字体颜色
  2. 为什么你用matplotlib savefig保存下来的图像是一个空白图片?
  3. python3 %%time 表示执行单元格时间 时间指的是CPU时间
  4. javascript --- ES6模块与CommonJS模块的差异
  5. 广州技术沙龙第 2、3 期参会者twitter、blog、兴趣大合集
  6. oracle 10g db_file,oracle 10g的db_file_multiblock_read_count参数
  7. Mr.J-- jQuery学习笔记(十八)--自定义动画
  8. html5 mp4转换ogv格式,如何把OGV转换为MP4?用它,轻松转换视频文件!
  9. UIAlertController (UIActionSheet, UIAlertView is deprecated in iOS 8.)
  10. 20200819:力扣202周周赛题解记录
  11. Hyperledger Fabric1.4超详细环境搭建 搭建不出来本博主倒立写代码
  12. android studio for android learning (十四) android的数据的存储sharedPreferences
  13. Android注册时总是出现验证码不正确问题的解决
  14. 猪齿鱼_01_环境搭建(一)_微服务支撑组件部署(源码形式)
  15. 问:女孩学编程好就业吗?
  16. 练手小项目——canvas放大镜效果 放大图片
  17. 移动域全链路可观测架构和关键技术
  18. BZOJ4706 B君的多边形 (超级卡特兰数/施罗德数)
  19. 素数统计 平移区间筛质数(1e9)
  20. python程序语言和机器人控制-利用树莓派+Python语言制作一部机器人就是如此简单!...

热门文章

  1. 转行学习java自学好还是去培训学校学习好?系统学习路线推荐
  2. 杀戮尖塔java启动_slay the spire杀戮尖塔无法启动
  3. 〖Python APP 自动化测试实战篇⑩〗- app自动化总结与展望
  4. Markdown字体大小颜色、大小、背景色 (二)
  5. 数据科学竞赛经验分享:你从未见过的究极进化秘笈
  6. exynos 4412
  7. 4月第1周榜单丨飞瓜数据B站UP主排行榜(哔哩哔哩平台)发布!
  8. excel 公式标多级目录序号
  9. 六个参数帮你全面了解固态硬盘
  10. 51nod1423 最大二“货”问题