当前位置: 首页 > news >正文

使用Pytorch从零开始构建Conditional PixelCNN

条件 PixelCNN

PixelCNN 是 PixelRNN 的卷积版本,它将图像中的像素视为一个序列,并在看到前面的像素后预测每个像素(定义如上和左,尽管这是任意的)。PixelRNN 是图像联合先验分布的自回归模型:
p ( x ) = p ( x 0 ) ∏ p ( x i ∣ x 0 , ⋯ , x i − 1 ) p(x) = p(x_0 ) ∏ p(x_i | x_0, \cdots,x_{i-1} ) p(x)=p(x0)p(xix0,,xi1)
PixelRNN 的训练速度很慢,因为循环无法并行化——即使是小图像也有数百或数千个像素,这对于 RNN 来说是一个相对较长的序列。用掩码卷积替换循环,使卷积滤波器仅看到上方和左侧的像素,从而实现更快的训练(图来自条件 PixelCNN 论文)。
在这里插入图片描述

然而,值得注意的是,最初的 PixelCNN 实现产生的结果比 PixelRNN 更差。在后续论文(使用 PixelCNN 解码器生成条件图像)中推测,结果降级的一个可能原因是 PixelCNN 中的 ReLU 激活与 LSTM 中的门控连接相比相对简单。Conditional PixelCNN 论文随后用门控激活取代了 ReLU:
y = t a n h ( W f ∗ x ) • σ ( W g ∗ x ) y = tanh (W f * x) • σ(W g * x) y=tanh(Wfx)σ(Wgx)
后续论文中提供的另一个可能的原因是,堆叠掩模卷积滤波器会导致盲点,无法捕获预测像素之上的所有像素(论文中的图):
在这里插入图片描述

PixelCNN 与 GAN

PixelCNN 和 GAN 是目前用于生成图像的两种深度学习模型。GAN 最近受到了很多关注,但在很多方面我发现它们的流行是没有根据的。

目前尚不清楚 GAN 实际上试图优化什么目标,因为训练目标的最小值(即愚弄鉴别器)将导致生成器重新创建所有训练图像和/或生成不一定类似于自然图像的对抗性示例。这反映在训练 GAN 的众所周知的困难以及无数的对其进行正则化的技巧上。让两个网络相互对抗以产生训练信号的想法很有趣,并且已经产生了许多好的论文(尤其是 CycleGAN),但我仍然不相信它们除了在社交媒体上发布华丽的帖子之外还有其他用途。

另一方面,PixelCNN 有很好的概率基础。这使得它们不仅可以通过对分布进行采样(从左到右,从上到下,遵循自回归定义)来生成图像,而且还意味着它们可以用于其他任务。例如:作为预筛选网络来检测域外或对抗性示例;用于检测训练集中的异常值;或估计测试中的不确定性。我将在下一篇文章中详细介绍其中一些扩展。

我很想知道是否有人尝试过将 PixelCNN 和 GAN 结合起来。也许 PixelCNN 可以用作解码器的前级或最后阶段(以一些更高级别的学习表示为条件),以避免 GAN 的一些训练困难。

实现

我的实现使用门控块,但为了快速实现,我决定放弃针对盲点问题的双流解决方案(将滤波器分为水平和垂直组件)。有代码可用于解决 Tensorflow 中的盲点问题,并且在 PyTorch 中重写它相当简单。这样,掩蔽就很简单:当前像素下方和右侧的所有内容在滤波器中都被清零,并且在第一层中,当前像素也在滤波器中设置为零。

class MaskedConv(nn.Conv2d):def __init__(self,mask_type,in_channels,out_channels,kernel_size,stride=1):"""mask_type: 'A' for first layer of network, 'B' for all others"""super(MaskedConv,self).__init__(in_channels,out_channels,kernel_size,stride,padding=kernel_size//2)assert mask_type in ('A','B')mask = torch.ones(1,1,kernel_size,kernel_size)mask[:,:,kernel_size//2,kernel_size//2+(mask_type=='B'):] = 0mask[:,:,kernel_size//2+1:] = 0self.register_buffer('mask',mask)def forward(self,x):self.weight.data *= self.maskreturn super(MaskedConv,self).forward(x)

门控 ResNet 块的实现稍微复杂一些:PixelCNN 在网络的两半之间有快捷连接,就像 U-Net 一样;PyTorch 允许模块的前向方法仅在输入是变量时才接受多个输入;由于网络前半部分的特征图不是变量,因此它们必须与其他输入(前一层的特征)连接起来。使用条件向量可以避免这种情况,因为它是一个变量(在本例中为类标签)。

class GatedRes(nn.Module):def __init__(self,in_channels,out_channels,n_classes,kernel_size=3,stride=1,aux_channels=0):super(GatedRes,self).__init__()self.conv = MaskedConv('B',in_channels,2*out_channels,kernel_size,stride)self.y_embed = nn.Linear(n_classes,2*out_channels)self.out_channels = out_channelsif aux_channels!=2*out_channels and aux_channels!=0:self.aux_shortcut = nn.Sequential(nn.Conv2d(aux_channels,2*out_channels,1),nn.BatchNorm2d(2*out_channels,momentum=0.1))if in_channels!=out_channels:self.shortcut = nn.Sequential(nn.Conv2d(in_channels,out_channels,1),nn.BatchNorm2d(out_channels,momentum=0.1))self.batchnorm = nn.BatchNorm2d(out_channels,momentum=0.1)def forward(self,x,y):# check for aux input from first half of net stacked into xif x.dim()==5:x,aux = torch.split(x,1,dim=0)x = torch.squeeze(x,0)aux = torch.squeeze(x,0)else:aux = Nonex1 = self.conv(x)y = torch.unsqueeze(torch.unsqueeze(self.y_embed(y),-1),-1)if aux is not None:if hasattr(self,'aux_shortcut'):aux = self.aux_shortcut(aux)x1 = (x1+aux)/2# split for gate (note: pytorch dims are [n,c,h,w])xf,xg = torch.split(x1,self.out_channels,dim=1)yf,yg = torch.split(y,self.out_channels,dim=1)f = torch.tanh(xf+yf)g = torch.sigmoid(xg+yg)if hasattr(self,'shortcut'):x = self.shortcut(x)return x+self.batchnorm(g*f)

我不确定在阅读原始论文时将批量归一化放在哪里,所以我将它放在我认为有意义的地方:在添加剩余连接之前。

实现这两个类后,整个网络就相对容易了。PyTorch 方案将所有内容定义为 的子类nn.Module,初始化所有层/操作/等。在构造函数中,然后在forward方法中将它们连接在一起可能会很混乱。如果您有大量快捷连接并且想要使用任意深度的循环对模型进行编码,则尤其如此。

注意:为了能够保存/恢复模型,您必须将图层存储在一个ModuleList而不是常规列表中。不过,附加和索引此列表在其他方面是相同的。

class PixelCNN(nn.Module):def __init__(self,in_channels,n_classes,n_features,n_layers,n_bins,dropout=0.5):super(PixelCNN,self).__init__()self.layers = nn.ModuleList()self.n_layers = n_layers# Up passself.input_batchnorm = nn.BatchNorm2d(in_channels,momentum=0.1)for l in range(n_layers):if l==0:  # start with normal convblock = nn.Sequential(MaskedConv('A',in_channels+1,n_features,kernel_size=7),nn.BatchNorm2d(n_features,momentum=0.1),nn.ReLU())else:block = GatedRes(n_features, n_features, n_classes)self.layers.append(block)# Down passfor _ in range(n_layers):block = GatedRes(n_features, n_features,n_classes,aux_channels=n_features)self.layers.append(block)# Last layer: project to n_bins (output is [-1, n_bins, h, w])self.layers.append(nn.Sequential(nn.Dropout2d(dropout),nn.Conv2d(n_features,n_bins,1),nn.LogSoftmax(dim=1)))def forward(self,x,y):# Add channel of ones so network can tell where padding isx = nn.functional.pad(x,(0,0,0,0,0,1,0,0),mode='constant',value=1)# Up passfeatures = []i = -1for _ in range(self.n_layers):i += 1if i>0:x = self.layers[i](x,y)else:x = self.layers[i](x)features.append(x)# Down passfor _ in range(self.n_layers):i += 1x = self.layers[i](torch.stack((x,features.pop())),y)# Last layeri += 1x = self.layers[i](x)assert i==len(self.layers)-1assert len(features)==0return x

MNIST 实际上是黑白的,因此我将标签离散为仅 4 个灰度级,以便计算交叉熵损失。在自然图像上,输出级别的数量显然需要更高。网络中的所有层都有 200 个特征。对于数据增强,我使用了 +/-5 度的随机旋转和最近邻采样。对于训练,我使用 Adam,学习率为 10 -4,dropout 率为 0.9。

更高的特征数量(比 MNIST 所需的特征数量更多)和更高的 dropout 是训练时间与正则化之间的权衡。这是一个在论文中很少提及的技巧,但有助于避免过度拟合——我只在一篇关于视频中动作识别训练的论文中看到过它,其中由于高维度与当前数据集大小,过度拟合是一个问题可用的。

我有一个 GTX1070 GPU,所以我没有运行任何类型的超参数优化:猜测合理的超参数并使模型工作的能力很大程度上说明了 Adam + 批量归一化 + dropout 的稳健性。学习率肯定可以更高,但这会产生更有趣的 GIF。

结果

在这里插入图片描述

上面的 gif 显示了整个训练过程中每个epochs后生成的一批 50 张图像(每类 5 个示例),从看似随机的涂鸦到类似于实际数字的东西。这是最佳epochs的结果:
在这里插入图片描述
这项工作的动机是看看条件 PixelCNN 是否也可以在类之间生成合理的示例。这是通过调节软标签而不是单热编码标签来完成的。

让我们尝试一下我所期望的容易混淆的数字对:(1,7), (3,8), (4,9), (5,6)
在这里插入图片描述
生成的类间示例并不像正常示例那样真实。模型可能需要一些额外的训练信号(例如来自分类器网络的教师强制)才能沿着图像流形进行插值。这有点令人失望,因为我曾希望生成类间示例可能允许使用学习的混合形式(而不是平均图像)。显然,进一步测试这个想法将需要更多的 GPU 来生成批量输入,所以无论如何,它目前超出了我的范围。

本文的完整代码可在Github代码库中查看。

本博文译自 jrbtaylor 的博客。

相关文章:

  • C#异常处理-throw语句
  • 软著项目推荐 深度学习 python opencv 火焰检测识别 火灾检测
  • C++值常用集合算法
  • 简易键值对文本解析
  • LINUX入门篇【10】---进程篇【2】---进程状态
  • TCP/IP协议、三次握手、四次挥手
  • <JavaEE> 什么是线程(Thread)?进程和线程有什么区别?
  • 性能优化中使用Profiler进行内存泄露的排查及解决方式
  • Linux 中的 ls 命令使用教程
  • anacoda 在 git 远程仓库的时候遇到 SSL 无法访问和字节缓冲太小和换源以及无法连接到官网和字节写入冲突
  • Linux——使用命令查看文件和文件夹数量
  • react项目自行配置热更新
  • 1-1、汇编语言概述
  • 【Python】巧用tkinter设计秒表计时器
  • linux下的工具---vim
  • Angular 响应式表单之下拉框
  • golang中接口赋值与方法集
  • Idea+maven+scala构建包并在spark on yarn 运行
  • JavaScript学习总结——原型
  • java小心机(3)| 浅析finalize()
  • log4j2输出到kafka
  • PHP面试之三:MySQL数据库
  • Solarized Scheme
  • spark本地环境的搭建到运行第一个spark程序
  • SpiderData 2019年2月13日 DApp数据排行榜
  • Vue.js 移动端适配之 vw 解决方案
  • webpack4 一点通
  • 爱情 北京女病人
  • 使用common-codec进行md5加密
  • 译米田引理
  • 原生JS动态加载JS、CSS文件及代码脚本
  • #图像处理
  • (10)Linux冯诺依曼结构操作系统的再次理解
  • (2)关于RabbitMq 的 Topic Exchange 主题交换机
  • (2009.11版)《网络管理员考试 考前冲刺预测卷及考点解析》复习重点
  • (70min)字节暑假实习二面(已挂)
  • (C语言)fread与fwrite详解
  • (C语言)求出1,2,5三个数不同个数组合为100的组合个数
  • (附表设计)不是我吹!超级全面的权限系统设计方案面世了
  • (机器学习-深度学习快速入门)第三章机器学习-第二节:机器学习模型之线性回归
  • (淘宝无限适配)手机端rem布局详解(转载非原创)
  • (详细版)Vary: Scaling up the Vision Vocabulary for Large Vision-Language Models
  • (一)appium-desktop定位元素原理
  • ./include/caffe/util/cudnn.hpp: In function ‘const char* cudnnGetErrorString(cudnnStatus_t)’: ./incl
  • .net core 微服务_.NET Core 3.0中用 Code-First 方式创建 gRPC 服务与客户端
  • .NET中使用Protobuffer 实现序列化和反序列化
  • .one4-V-XXXXXXXX勒索病毒数据怎么处理|数据解密恢复
  • @NoArgsConstructor和@AllArgsConstructor,@Builder
  • @vue/cli脚手架
  • [ web基础篇 ] Burp Suite 爆破 Basic 认证密码
  • [22]. 括号生成
  • [Android]Tool-Systrace
  • [ASP]青辰网络考试管理系统NES X3.5
  • [BT]BUUCTF刷题第9天(3.27)
  • [IE编程] 如何编程清除IE缓存