【发布时间】:2019-11-28 20:16:20
【问题描述】:
我创建了一个简单的 6 层 DCGAN,并在 CelebA 数据集(其中一部分包含 30K 图像)上对其进行了训练。
我注意到我的网络生成的图像看起来很暗,随着网络训练的更多,明亮的颜色会逐渐变成暗淡的颜色!
这里有一些例子:
这就是 CelebA 图像的样子(用于训练的真实图像):
这些是生成的,数字表示 epoch 数(他们最终训练了 30 个 epoch):
造成这种现象的原因是什么?
我尝试了所有关于GANs 的一般技巧,例如在-1 和1 之间重新缩放输入图像,或者在Discriminator 的第一层和@的最后一层不使用BatchNorm 987654335@ 或
在Discriminator 中使用LeakyReLU(0.2),在Generator 中使用ReLU。但我不知道为什么图像会这么暗/暗!
这是由于训练图像较少造成的吗?
还是网络问题造成的?如果是这样,这些缺陷的根源是什么?
以下是这些网络的实施方式:
def conv_batch(in_dim, out_dim, kernel_size, stride, padding, batch_norm=True):
layers = nn.ModuleList()
conv = nn.Conv2d(in_dim, out_dim, kernel_size, stride, padding, bias=False)
layers.append(conv)
if batch_norm:
layers.append(nn.BatchNorm2d(out_dim))
return nn.Sequential(*layers)
class Discriminator(nn.Module):
def __init__(self, conv_dim=32, act = nn.ReLU()):
super().__init__()
self.conv_dim = conv_dim
self.act = act
self.conv1 = conv_batch(3, conv_dim, 4, 2, 1, False)
self.conv2 = conv_batch(conv_dim, conv_dim*2, 4, 2, 1)
self.conv3 = conv_batch(conv_dim*2, conv_dim*4, 4, 2, 1)
self.conv4 = conv_batch(conv_dim*4, conv_dim*8, 4, 1, 1)
self.conv5 = conv_batch(conv_dim*8, conv_dim*10, 4, 2, 1)
self.conv6 = conv_batch(conv_dim*10, conv_dim*10, 3, 1, 1)
self.drp = nn.Dropout(0.5)
self.fc = nn.Linear(conv_dim*10*3*3, 1)
def forward(self, input):
batch = input.size(0)
output = self.act(self.conv1(input))
output = self.act(self.conv2(output))
output = self.act(self.conv3(output))
output = self.act(self.conv4(output))
output = self.act(self.conv5(output))
output = self.act(self.conv6(output))
output = output.view(batch, self.fc.in_features)
output = self.fc(output)
output = self.drp(output)
return output
def deconv_convtranspose(in_dim, out_dim, kernel_size, stride, padding, batchnorm=True):
layers = []
deconv = nn.ConvTranspose2d(in_dim, out_dim, kernel_size = kernel_size, stride=stride, padding=padding)
layers.append(deconv)
if batchnorm:
layers.append(nn.BatchNorm2d(out_dim))
return nn.Sequential(*layers)
class Generator(nn.Module):
def __init__(self, z_size=100, conv_dim=32):
super().__init__()
self.conv_dim = conv_dim
# make the 1d input into a 3d output of shape (conv_dim*4, 4, 4 )
self.fc = nn.Linear(z_size, conv_dim*4*4*4)#4x4
# conv and deconv layer work on 3d volumes, so we now only need to pass the number of fmaps and the
# input volume size (its h,w which is 4x4!)
self.drp = nn.Dropout(0.5)
self.deconv1 = deconv_convtranspose(conv_dim*4, conv_dim*3, kernel_size =4, stride=2, padding=1)
self.deconv2 = deconv_convtranspose(conv_dim*3, conv_dim*2, kernel_size =4, stride=2, padding=1)
self.deconv3 = deconv_convtranspose(conv_dim*2, conv_dim, kernel_size =4, stride=2, padding=1)
self.deconv4 = deconv_convtranspose(conv_dim, conv_dim, kernel_size =3, stride=2, padding=1)
self.deconv5 = deconv_convtranspose(conv_dim, 3, kernel_size =4, stride=1, padding=1, batchnorm=False)
def forward(self, input):
output = self.fc(input)
output = self.drp(output)
output = output.view(-1, self.conv_dim*4, 4, 4)
output = F.relu(self.deconv1(output))
output = F.relu(self.deconv2(output))
output = F.relu(self.deconv3(output))
output = F.relu(self.deconv4(output))
# we create the image using tanh!
output = F.tanh(self.deconv5(output))
return output
# testing nets
dd = Discriminator()
zd = np.random.rand(2,3,64,64)
zd = torch.from_numpy(zd).float()
# print(dd)
print(dd(zd).shape)
gg = Generator()
z = np.random.uniform(-1,1,size=(2,100))
z = torch.from_numpy(z).float()
print(gg(z).shape)
【问题讨论】:
-
你试过长时间训练它吗?喜欢 100 或 500 个 epoch 吗?可能只是网络还没有收敛。
-
另外,我肯定不会在鉴别器的第一层使用
BatchNorm。如果你没有提到它,这就是我的建议。 -
我已经训练了 60 个 epoch,结果是一样的,generator loss 会飙升!所以我将它减少到 30 个 epoch。我不使用
BatchNorm作为第一个conv层。 -
您找到解决此问题的方法了吗?
-
一种可能的理论:生成器“发现”了通过使图像更暗(对比度更低、远离数据集均值等),鉴别器无法区分真假示例跨度>
标签: python pytorch generative-adversarial-network