一、内容信息与风格信息

图像的特征可以简单的分为内容信息以及风格信息。内容信息表示了图像的细节,风格信息表示了图像的整体“画风”。风格迁移就是将我们想要的风格迁移至另一张图像。

二、网络的特征图

图像输入卷积层后,根据设置的特征图数目不同会产生一定数量的特征图。它们起到了提取图像特征的作用。不同层之间的特征图所能够表示的细节是不一样的。一般来说,层数较低的特征图越能够表示图像的细节;而较高层数图像产生的特征图则更加能够反映图像的整体结构。因此可以用较高层的特征图来表示图像的内容,而特征则需要较多层数的特征图共同决定。
我们想要的目的就是使特征损失函数以及内容损失函数同时尽可能的低。

三、定义损失函数

1、内容损失函数

内容损失函数指的是两张图像整体内容的差异,可通过原图像输入网络获得的特征图与目标图像输入获得的特征图之间的均方误差得到,在pytorch中,可以使用

torch.nn.MSELoss(feature1, feature2)

来计算

2、风格损失函数

风格损失由多层特征图共同决定,我们将所有特征图展平为一维向量,计算同一层中所有特征图的Gram矩阵:即其第i行j列为第i个元素和第j个乘积乘积,程序中可以这么获得

# 输入为特征图
features = input.view(a * b, c * d)    # a为batch_size, b特征图数量,c,d为高宽
torch.mm(features, torch.t(features)) #与转置相乘

之后再计算原图像输入网络获得的特征图与目标图像输入获得的特征图之间的均方误差即可

四、完整代码

选择VGG19网络,并选择第四层卷积层获得的特征图进行内容损失计算,前五层进行特征损失计算。
相关说明见注释

# torch import
import torch
import torch.optim as optim
import torch.nn as nn
import torch.nn.functional as F
from torchvision import datasets, models, transforms
from torch.utils.data import DataLoader

# other import
import numpy as np
import matplotlib.pyplot as plt
from PIL import Image
import copy

device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')

# 图像路径
path_style = 'style.jpg'
path_neirong = 'neirong.jpg'

# 权重
style_weight = 1000
content_weight = 1

# 图像大小
image_size = 256


# pipeline
pipeline = transforms.Compose([
    transforms.Resize((256, 256)),
    transforms.ToTensor()
])


# 图片加载函数
# 输入图片路径,输出与网络匹配的张量
def image_loader(img_path):
    img = Image.open(img_path)
    img_tensor = pipeline(img)
    # 在第一维添加一个维度,为输入网络需要
    img_tensor = img_tensor.unsqueeze(0)
    return img_tensor


# 在GPU上运算
style_img = image_loader(path_style).to(device)
content_img = image_loader(path_neirong).to(device)

assert style_img.size() == content_img.size(), "两张图象大小需要相等"

unload = transforms.ToPILImage()


# 显示图像函数
def img_show(img):
    image = img.clone().cpu()
    image = image.view(3, image_size, image_size)
    image = unload(image)
    plt.imshow(image)
    plt.pause(0.001)


# 切换为动态图模式,遇到show()也会继续执行程序
plt.ion()
plt.figure()
img_show(style_img)
plt.figure()
img_show(content_img)

# 加载vgg19网络
cnn = models.vgg19(pretrained=True).features
cnn.to(device)
content_layers = ['conv_4']
style_layers = ['conv_1', 'conv_2', 'conv_3', 'conv_4', 'conv_5']


# 内容损失,它是一层网络,为nn.module的子类
class ContentLoss(nn.Module):
    # target是内容输入网络的结果
    def __init__(self, target, weight):
        super(ContentLoss, self).__init__()
        # detach()可以将target这几层特征图与之前的动态图解耦,这样就不会操作到原来的特征图
        self.target = target.detach() * weight
        self.weight = weight
        self.criterion = nn.MSELoss()

    # 用以计算目标与输入的误差
    def forward(self, input):
        self.loss = self.criterion(input * self.weight, self.target)
        self.output = input
        return self.output

    # retain_graph 如果设置为False,计算图中的中间变量在计算完后就会被释放
    # 进行一次backward之后,各个节点的值会清除,这样进行第二次backward会报错,如果加上retain_graph==True后,可以再来一次backward。
    def backward(self, retain_graph=True):
        self.loss.backward(retain_graph=retain_graph)
        return self.loss


# 获得gram矩阵函数
def Gram(input):
    a, b, c, d = input.size()
    # 将特征图展平为单一向量
    features = input.view(a * b, c * d)
    # feature与其转置相乘,相当于任意两数相乘
    G = torch.mm(features, features.t())
    # 归一化
    return G.div(a * b * c * d)


# 风格损失
class StyleLoss(nn.Module):
    def __init__(self, target, weight):
        super(StyleLoss, self).__init__()
        self.target = target.detach() * weight
        self.weight = weight
        self.criterion = nn.MSELoss()

    def forward(self, input):
        self.output = input.clone()
        input = input.cuda()
        self.G = Gram(input)
        self.G.mul_(self.weight)
        self.loss = self.criterion(self.G, self.target)
        return self.output

    def backward(self, retain_graph=True):
        self.loss.backward(retain_graph=retain_graph)
        return self.loss


model = nn.Sequential()
model.to(device)
content_losses = []
style_losses = []

# 构建 model
i = 1
for layer in list(cnn):
    # 获得卷积层
    if isinstance(layer, nn.Conv2d):
        name = 'conv_' + str(i)
        model.add_module(name, layer)

        if name in content_layers:
            # 把内容图像传入模型,获取需要达到的特征图
            target = model(content_img).clone()
            # 实例化content_loss层,和其他如conv2d层相似
            content_loss = ContentLoss(target, content_weight)
            content_loss = content_loss.to(device)
            model.add_module('content_loss_' + str(i), content_loss)
            content_losses.append(content_loss)

        if name in style_layers:
            target_feature = model(style_img).clone()
            target_feature = target_feature.to(device)
            target_feature_gram = Gram(target_feature)
            style_loss = StyleLoss(target_feature_gram, style_weight)
            style_loss = style_loss.to(device)
            model.add_module('style_loss_' + str(i), style_loss)
            style_losses.append(style_loss)

    if isinstance(layer, nn.ReLU):
        name = 'relu_' + str(i)
        model.add_module(name, layer)
        i += 1
    if isinstance(layer, nn.MaxPool2d):
        name = 'pool_' + str(i)
        model.add_module(name, layer)


input_img = torch.randn(content_img.size()).to(device)
plt.figure(1)
img_show(input_img)


# 迭代开始
# nn.Parameter将张量转换为可以反向传播的对象
input_parm = nn.Parameter(input_img.data)
# 仅将输入图像传入优化器,仅对输入图像进行反向传播
optimizer = optim.LBFGS([input_parm])
num_step = 700

print('正在构造风格迁移模型')

print('开始优化')
for i in range(num_step):
    input_parm.data.clamp_(0, 1)
    optimizer.zero_grad()
    # 这一步会运行forward
    model(input_parm)
    style_score = 0
    content_score = 0

    for sl in style_losses:
        style_score += sl.backward()
    for cl in content_losses:
        content_score += cl.backward()
    if i % 50 == 0:
        print('正在运行{}轮'.format(i))
        print('风格损失{},\t内容损失{}'.format(style_score, content_score))


    def closure():
        return style_score + content_score

    optimizer.step(closure)

out_put = input_parm.data.clamp_(0, 1)

plt.figure()
img_show(out_put)
plt.ioff()
plt.show()