目录
DCGAN理论讲解
DCGAN的改进:
DCGAN的设计技巧
DCGAN纯代码实现
导入库
导入数据和归一化
定义生成器
定义鉴别器
初始化和 模型训练
运行结果
DCGAN也叫深度卷积生成对抗网络,DCGAN就是将CNN与GAN结合在一起,生成模型和判别模型都运用了深度卷积神经网络的生成对抗网络。
DCGAN将GAN与CNN相结合,奠定了之后几乎所有GAN的基本网络架构。DCGAN极大地提升了原始GAN训练的稳定性以及生成结果的质量
DCGAN主要是在网络架构上改进了原始的GAN,DCGAN的生成器与判别器都利用CNN架构替换了原始GAN的全连接网络,主要改进之处有如下几个方面,
(1)DCGAN的生成器和判别器都舍弃了CNN的池化层,判别器保留CNN的整体架构,生成器则是将卷积层替换成了反卷积层。
(2)在判别器和生成器中使用了BatchNormalization(BN)层,这里有助于处理初始化不良导致的训练问题,加速模型训练提升训练的稳定性。要注意,在生成器的输出层和判别器的输入层不使用BN层。
(3)在生成器中除输出层使用Tanh()激活函数,其余层全部使用Relu激活函数,在判别器中,除输出层外所有层都使用LeakyRelu激活函数,防止梯度稀疏
自己画的,凑合着看吧/(*/ω\*)捂脸/
一,取消所有pooling层,G网络中使用转置卷积进行上采样,D网络中加入stride的卷积(为防止梯度稀疏)代替pooling
二,去掉FC层(全连接),使网络变成全卷积网络
三,G网络中使用Relu作为激活函数,最后一层用Tanh
四,D网络中使用LeakyRelu激活函数
五,在generator和discriminator上都使用batchnorm,解决初始化差的问题,帮助梯度传播到每一层,防止generator把所有的样本都收敛到同一点。直接将BN应用到所有层会导致样本震荡和模型不稳定,因此在生成器的输出层和判别器的输入层不使用BN层,可以防止这种现象。
六,使用Adam优化器
七,参数设置参考:LeakyRelu的斜率是0.2;Learing rate = 0.0002;batch size是128.
import torch
import torch.nn as nn
import torch.nn.functional as F
import torch.optim as optim #优化
import numpy as np
import matplotlib.pyplot as plt #绘图
import torchvision #加载图片
from torchvision import transforms #图片变换
#对数据做归一化(-1,1)
transform=transforms.Compose([
#将shanpe为(H,W,C)的数组或img转为shape为(C,H,W)的tensor
transforms.ToTensor(), #转为张量并归一化到【0,1】;数据只是范围变了,并没有改变分布
transforms.Normalize(mean=0.5,std=0.5)#数据归一化处理,将数据整理到[-1,1]之间;可让数据呈正态分布
])
#下载数据到指定的文件夹
train_ds = torchvision.datasets.MNIST('data/',
train=True,
transform=transform,
download=True)
#数据的输入部分
train_dl=torch.utils.data.DataLoader(train_ds,batch_size=64,shuffle=True)
使用长度为100的noise作为输入,也可以使用torch.randn(batchsize,100,1,1)
class Generator(nn.Module):
def __init__(self):
super(Generator,self).__init__()
self.linear1 = nn.Linear(100,256*7*7)
self.bn1=nn.BatchNorm1d(256*7*7)
self.deconv1 = nn.ConvTranspose2d(256,128,
kernel_size=(3,3),
stride=1,
padding=1) #生成(128,7,7)的二维图像
self.bn2=nn.BatchNorm2d(128)
self.deconv2 = nn.ConvTranspose2d(128,64,
kernel_size=(4,4),
stride=2,
padding=1) #生成(64,14,14)的二维图像
self.bn3=nn.BatchNorm2d(64)
self.deconv3 = nn.ConvTranspose2d(64,1,
kernel_size=(4,4),
stride=2,
padding=1) #生成(1,28,28)的二维图像
def forward(self,x):
x=F.relu(self.linear1(x))
x=self.bn1(x)
x=x.view(-1,256,7,7)
x=F.relu(self.deconv1(x))
x=self.bn2(x)
x=F.relu(self.deconv2(x))
x=self.bn3(x)
x=torch.tanh(self.deconv3(x))
return x
class Discriminator(nn.Module):
def __init__(self):
super(Discriminator,self).__init__()
self.conv1 = nn.Conv2d(1,64,kernel_size=3,stride=2)
self.conv2 = nn.Conv2d(64,128,kernel_size=3,stride=2)
self.bn = nn.BatchNorm2d(128)
self.fc = nn.Linear(128*6*6,1)
def forward(self,x):
x= F.dropout2d(F.leaky_relu(self.conv1(x)))
x= F.dropout2d(F.leaky_relu(self.conv2(x)) ) #(batch,128,6,6)
x = self.bn(x)
x = x.view(-1,128*6*6) #展平
x = torch.sigmoid(self.fc(x))
return x
#设备的配置
device='cuda' if torch.cuda.is_available() else 'cpu'
#初化生成器和判别器把他们放到相应的设备上
gen = Generator().to(device)
dis = Discriminator().to(device)
#交叉熵损失函数
loss_fn = torch.nn.BCELoss()
#训练器的优化器
d_optimizer = torch.optim.Adam(dis.parameters(),lr=1e-5)
#训练生成器的优化器
g_optimizer = torch.optim.Adam(dis.parameters(),lr=1e-4)
def generate_and_save_images(model,epoch,test_input):
prediction = np.squeeze(model(test_input).detach().cpu().numpy())
fig = plt.figure(figsize=(4,4))
for i in range(prediction.shape[0]):
plt.subplot(4,4,i+1)
plt.imshow((prediction[i]+1)/2,cmap='gray')
plt.axis('off')
plt.savefig('image_at_epoch_{:04d}.png'.format(epoch))
plt.show()
test_input = torch.randn(16,100 ,device=device) #16个长度为100的随机数
D_loss = []
G_loss = []
#训练循环
for epoch in range(30):
#初始化损失值
D_epoch_loss = 0
G_epoch_loss = 0
count = len(train_dl.dataset) #返回批次数
#对数据集进行迭代
for step,(img,_) in enumerate(train_dl):
img =img.to(device) #把数据放到设备上
size = img.shape[0] #img的第一位是size,获取批次的大小
random_seed = torch.randn(size,100,device=device)
#判别器训练(真实图片的损失和生成图片的损失),损失的构建和优化
d_optimizer.zero_grad()#梯度归零
#判别器对于真实图片产生的损失
real_output = dis(img) #判别器输入真实的图片,real_output对真实图片的预测结果
d_real_loss = loss_fn(real_output,
torch.ones_like(real_output,device=device)
)
d_real_loss.backward()#计算梯度
#在生成器上去计算生成器的损失,优化目标是判别器上的参数
generated_img = gen(random_seed) #得到生成的图片
#因为优化目标是判别器,所以对生成器上的优化目标进行截断
fake_output = dis(generated_img.detach()) #判别器输入生成的图片,fake_output对生成图片的预测;detach会截断梯度,梯度就不会再传递到gen模型中了
#判别器在生成图像上产生的损失
d_fake_loss = loss_fn(fake_output,
torch.zeros_like(fake_output,device=device)
)
d_fake_loss.backward()
#判别器损失
disc_loss = d_real_loss + d_fake_loss
#判别器优化
d_optimizer.step()
#生成器上损失的构建和优化
g_optimizer.zero_grad() #先将生成器上的梯度置零
fake_output = dis(generated_img)
gen_loss = loss_fn(fake_output,
torch.ones_like(fake_output,device=device)
) #生成器损失
gen_loss.backward()
g_optimizer.step()
#累计每一个批次的loss
with torch.no_grad():
D_epoch_loss +=disc_loss
G_epoch_loss +=gen_loss
#求平均损失
with torch.no_grad():
D_epoch_loss /=count
G_epoch_loss /=count
D_loss.append(D_epoch_loss)
G_loss.append(G_epoch_loss)
generate_and_save_images(gen,epoch,test_input)
print('Epoch:',epoch)
因篇幅有限,这里展示第一张和最后一张,这里我训练了30个epoch,有条件的可以多训练几次,训练越多效果越明显哦
希望我的文章能对你有所帮助。欢迎点赞 ,评论,关注,⭐️收藏