基于GAN的face2face实现

作者: 斯坦因和他的狗 | 来源:发表于2017-09-12 13:09 被阅读1538次
face2face.gif

本文实现依赖python2.7、tensorflow、opencv、dlib,训练生成对抗模型(GAN),实现图像合成。

face2face是image2image或被称为pix2pix众多有趣应用中的一个。
更多应用案例与原理论文,请参考
Image-to-Image Translation with Conditional Adversarial Nets
Image-to-Image Translation in Tensorflow by Christopher Hesse
Dat Tran博客 Face2face

更多应用案例
  • Step 1 利用opencv和dlib准备训练集
  • Step 2 利用tensorflow训练模型
  • Step 3 Export Model & Freeze Model
  • Step 4 调用模型

step 1 准备训练集

  1. 在当前目录创建original与landmark文件夹。每个文件夹包含400张含有人脸的图片。
# -*- coding: utf-8 -*-
from __future__ import division
import cv2
import dlib
import numpy as np
import os

os.makedirs('original') # 创建文件夹,用于保存原始视频中截取的帧
os.makedirs('landmarks') # 创建文件夹,用于保存描绘有人脸特征的图片
DOWNSAMPLE_RATIO = 4 # 图片缩小比例,小图片加快人脸检测与特征提取速度
photo_number = 400 # 从视频中提取400张含有人脸特征的帧
video_path = 'angela_merkel_speech.avi' # 用于训练的含有人脸的视频路径
detector = dlib.get_frontal_face_detector()
predictor = dlib.shape_predictor('shape_predictor_68_face_landmarks.dat')

def reshape_for_polyline(array):
    return np.array(array, np.int32).reshape((-1, 1, 2))

def prepare_training_data():
    cap = cv2.VideoCapture(video_path)
    count = 0
    while cap.isOpened():
        ret, frame = cap.read() # 读取视频帧
        frame_resize = cv2.resize(frame, (0,0), fx=1 / DOWNSAMPLE_RATIO, fy=1 / DOWNSAMPLE_RATIO)
        gray = cv2.cvtColor(frame_resize, cv2.COLOR_BGR2GRAY)
        faces = detector(gray, 1) # 识别人脸位置
        black_image = np.zeros(frame.shape, np.uint8) # 创建一张黑色图片用于描绘人脸特征

        if len(faces) == 1:
            for face in faces:
                detected_landmarks = predictor(gray, face).parts() # 提取人脸特征
                landmarks = [[p.x * DOWNSAMPLE_RATIO, p.y * DOWNSAMPLE_RATIO] for p in detected_landmarks]

                jaw = reshape_for_polyline(landmarks[0:17])
                left_eyebrow = reshape_for_polyline(landmarks[22:27])
                right_eyebrow = reshape_for_polyline(landmarks[17:22])
                nose_bridge = reshape_for_polyline(landmarks[27:31])
                lower_nose = reshape_for_polyline(landmarks[30:35])
                left_eye = reshape_for_polyline(landmarks[42:48])
                right_eye = reshape_for_polyline(landmarks[36:42])
                outer_lip = reshape_for_polyline(landmarks[48:60])
                inner_lip = reshape_for_polyline(landmarks[60:68])

                color = (255, 255, 255) # 人脸特征用白色描绘
                thickness = 3 # 线条粗细

                cv2.polylines(img=black_image, 
                              pts=[jaw,left_eyebrow, right_eyebrow, nose_bridge],
                              isClosed=False,
                              color=color,
                              thickness=thickness)
                cv2.polylines(img=black_image, 
                              pts=[lower_nose, left_eye, right_eye, outer_lip,inner_lip],
                              isClosed=True,
                              color=color,
                              thickness=thickness)

            # 保存图片
            cv2.imwrite("original2/{}.png".format(count), frame)
            cv2.imwrite("landmarks2/{}.png".format(count), black_image)
            count += 1

# 执行准备数据函数
prepare_training_data()
  1. 改变图片尺寸(调整为正方形)、拼接图片(用于训练)
    这步涉及的函数有点多,主要是利用tensorflow对jpeg与png图片的读取、保存、裁剪、缩放、拼接,直接根据下面步骤执行就可以。不过建议对tensorflow图片处理细节感兴趣的小伙伴看源代码,会有很多收获。
    github repo affinelayer/pix2pix-tensorflow
# Clone the repo from Christopher Hesse's pix2pix TensorFlow implementation
git clone https://github.com/affinelayer/pix2pix-tensorflow.git

# Move the original and landmarks folder into the pix2pix-tensorflow folder
mv face2face-demo/landmarks face2face-demo/original pix2pix-tensorflow/photos

# Go into the pix2pix-tensorflow folder
cd pix2pix-tensorflow/

# Resize original images
python tools/process.py \
  --input_dir photos/original \
  --operation resize \
  --output_dir photos/original_resized
  
# Resize landmark images
python tools/process.py \
  --input_dir photos/landmarks \
  --operation resize \
  --output_dir photos/landmarks_resized
  
# Combine both resized original and landmark images
python tools/process.py \
  --input_dir photos/landmarks_resized \
  --b_dir photos/original_resized \
  --operation combine \
  --output_dir photos/combined
  
# Split into train/val set
python tools/split.py \
  --dir photos/combined

执行完上面的代码,模型的训练数据就已经准备就绪了。整个 process.py文件,基本是以下结构。我觉得这是值得一书的东西,以备不时之需。

import tensorflow as tf

# 创建一个万金油般的create_op函数
def create_op(func, **placeholders):
    op = func(**placeholders)

    def f(**kwargs):
        feed_dict = {}
        for argname, argvalue in kwargs.items():
            placeholder = placeholders[argname]
            feed_dict[placeholder] = argvalue
        return tf.get_default_session().run(op, feed_dict=feed_dict)

    return f

# 创建你的operation函数
encode_jpeg = create_op(
    func=tf.image.encode_jpeg,
    image=tf.placeholder(tf.uint8),
)

# 调用你的operation函数
decode_jpeg(contents=contents)

step 2 训练模型

  1. 网络结构简介
    我之前做相关分享的ppt, 人脸识别原理与pix2pix分享 网盘地址第23页开始有pix2pix相关内容。 理论上的网络结构 上图是理论结构,但是为了加快训练速度,代码实现的是下图网络结构。 实际使用结构 每个encode与decode模块细节 encode与decode模块细节
  2. 如果你比较着急可以直接执行以下代码,开始训练。我使用的GPU是英伟达的titanx,花了90分钟。
python pix2pix.py \
  --mode train \
  --output_dir face2face-model \
  --max_epochs 200 \
  --input_dir photos/combined/train \
  --which_direction AtoB
  1. 如果希望深入了解细节,请看下面代码。但是以下代码不用直接执行用于训练模型:) 如果预先没有CNN卷积神经网络相关的知识,那么下面的代码会让气氛很尴尬的呢。
  • 定义卷积
def conv(batch_input, out_channels, stride):
    ```输入结构:[batch, in_height, in_width, in_channels],
       卷积核结构: [filter_width, filter_height, in_channels, out_channels]
       输出结构: [batch, out_height, out_width, out_channels] 
       选用4x4的卷积核 + padding 1 + 步长stride,输出结构 VALID```
    with tf.variable_scope("conv"):
        in_channels = batch_input.get_shape()[3] # 输入图片的通道数
        # 初始化 4X4卷积核,使用random_normal_initializer初始化
        filter = tf.get_variable("filter",
                                [4, 4, in_channels, out_channels],
                                dtype=tf.float32,
                                initializer=tf.random_normal_initializer(0, 0.02)) 
        # padding 1
        padded_input = tf.pad(batch_input,
                              [[0, 0], [1, 1], [1, 1], [0, 0]], 
                              mode="CONSTANT")
        # 2D 卷积 步长为传参的stride
        conv = tf.nn.conv2d(padded_input, filter, [1, stride, stride, 1], padding="VALID")
        return conv
  • 定义激活函数
    使用leaky ReLu激活函数,下图是leakReLu与ReLu的对比
    • ReLu 激活函数优点:
      a) 在刺激大于0的区域,不会出现梯度为0的问题。
      b) 计算效率高。
      c) 模型loss下降收敛快。大约是tanh与sigmoid激活函数的6倍。
    • Leaky ReLu 激活函数优点:
      a) ReLu的优点都有。
      b) 不会出现梯度为0的问题。
      c) 无论什么时候神经元都会被激活。
leaky ReLu与ReLu的对比
你可能对 tf.identity(x) 的作用带有疑问,what is tf.identity used for?
def lrelu(x, a):
    with tf.name_scope("lrelu"):
        # leak: a*x/2 - a*abs(x)/2;   linear: x/2 + abs(x)/2
        x = tf.identity(x)
        return (0.5 * (1 + a)) * x + (0.5 * (1 - a)) * tf.abs(x)
  • 定义batchnorm
def batchnorm(input):
    with tf.variable_scope("batchnorm"):
        input = tf.identity(input)

        # 定义batch norm 中需要训练的两个参数offset与scale
        channels = input.get_shape()[3]
        offset = tf.get_variable("offset", 
                                 [channels], 
                                 dtype=tf.float32,
                                 initializer=tf.zeros_initializer())
        scale = tf.get_variable("scale", 
                                [channels], dtype=tf.float32,
                                initializer=tf.random_normal_initializer(1.0, 0.02))
      
        mean, variance = tf.nn.moments(input, axes=[0, 1, 2], keep_dims=False)
        variance_epsilon = 1e-5
        normalized = tf.nn.batch_normalization(input, 
                                               mean, variance, 
                                               offset, scale, 
                                               variance_epsilon=variance_epsilon)
        return normalized

step 3 Export Model & Freeze Model

  • reduce model,我们需要生成模型用于图像生成,而判别模型可以去掉,以减少模型参数。这里我就不把生成模型重新复制一遍贴出来了。详细请看 github repo datitran/face2face-demo/reduce_model.py。思路是:
    • 首先把pix2pix.py中与生成模型相关部分复制了一份
    • 然后加载训练好的模型
    • 最后保存一个新模型。

reduce_model.py 中值得一书的事情。新建的generate_output函数,用于输入图片,生成图片。reduce_model.py 中所有 tf.variable_scope('名字')都与加载的训练好的模型一模一样,这样加载的模型会把它的参数与新模型的tf.variable_scope('名字')一一对应起来。由于新模型只保留了生成模型相关的tf.variable_scope('名字'),所以新模型的参数大大减少,实现model reduce.

x = tf.placeholder(tf.uint8, shape=(256, 512, 3), name='image_tensor')  # input tensor
y = generate_output(x)  # 输入图片,输出生成的图片 

with tf.Session() as sess:
    # 加载训练好的模型
    saver = tf.train.Saver()
    checkpoint = tf.train.latest_checkpoint(args.input_folder)
    saver.restore(sess, checkpoint)

    # 输出新模型
    saver = tf.train.Saver()
    saver.save(sess, './reduced_model')
  • freeze model,我们把模型保存成一个.pb文件以方便调用
import tensorflow as tf
from tensorflow.python.framework import graph_util

def freeze_graph(model_folder):
    # 获取模型路径
    checkpoint = tf.train.get_checkpoint_state(model_folder)
    input_checkpoint = checkpoint.model_checkpoint_path
    output_graph = './frozen_model.pb'
    output_node_names = 'generate_output/output'

    # 加载 graph 
    saver = tf.train.import_meta_graph(input_checkpoint + '.meta',
                                       clear_devices=True)
    # 取出 graph
    graph = tf.get_default_graph()
    input_graph_def = graph.as_graph_def()

    # 开一个新会话,加载参数,选择需要的节点,保存模型文件
    with tf.Session() as sess:
        saver.restore(sess, input_checkpoint) # 加载graph的参数

        # tensorflow内置函数,将变量转为常量
        output_graph_def = graph_util.convert_variables_to_constants(
            sess,  # 用于取回参数
            input_graph_def,  # 用于取回节点node
            [output_node_names]  # 选择需要的节点名)

        # 将模型写入 .pb文件
        with tf.gfile.GFile(output_graph, 'wb') as f:
            f.write(output_graph_def.SerializeToString())
        print('%d ops in the final graph.' % len(output_graph_def.node))

freeze_graph('./reduced_model')

step 4 调用模型

freeze model大约200MB,这是我训练好的模型网盘地址,模型训练用的是400张图片,200epoch。

import tensorflow as tf
def load_graph(frozen_graph_filename):
    """ 加载 freezed model """
    graph = tf.Graph()
    with graph.as_default():
        od_graph_def = tf.GraphDef()
        with tf.gfile.GFile(frozen_graph_filename, 'rb') as fid:
            serialized_graph = fid.read()
            od_graph_def.ParseFromString(serialized_graph)
            tf.import_graph_def(od_graph_def, name='')
    return graph

graph = load_graph(frozen_model_file)
image_tensor = graph.get_tensor_by_name('image_tensor:0')
output_tensor = graph.get_tensor_by_name('generate_output/output:0')
sess = tf.Session(graph=graph)
# 图片必须是256X256,人脸在靠近中间的位置
generated_image = sess.run(output_tensor, 
                           feed_dict = {image_tensor: image})

相关文章

  • 基于GAN的face2face实现

    本文实现依赖python2.7、tensorflow、opencv、dlib,训练生成对抗模型(GAN),实现图像...

  • no.2-10/23/2018

    一:预定目标 继续对GAN的学习。掌握原始GAN的思想和实现,接触其他GAN变形的原理和实现方法。这周学习了 Co...

  • 2020机器学习GAN(1)

    在 2018 年,深度学习掀起 GAN 的潮流,在许多领域都已经有了应用,而且基于 GAN 项目五花八门。早在**...

  • sketch to photo 基于Gan

    A----- 思路与CYZ的很像,获取sketch,添加颜色指导,LOSS加正则项。CVPR 2017 B 201...

  • GAN-简单的demo

    摘要:本篇博客主要介绍GAN的基本原理与代码实现 GAN的基本原理介绍 GAN是一种生成式对抗网络,它属于一种生成...

  • 使用GAN实现可变形医学图像配准DEFORMABLE MEDIC

    题目 :使用GAN实现可变形医学图像配准 传统的深度学习配准方法都采用迭代方法,作者采用GAN实现了一种端到端的多...

  • GAN 的 keras 实现

    本文结构: 什么是 GAN? 优点? keras 例子? 什么是 GAN? GAN,全称为 Generative ...

  • 扩散模型

    常见的生成模型(Generative Models)如 GAN、VAE 和基于流( Flow-based )的模型...

  • [基于GAN的语义修复] Prior Guided GAN Ba

    原作者:Lahiri等人     笔记整理:ganyd 2020.9.15如有错误或者不清楚之处,欢迎讨论,如需...

  • keras 实现 GAN

    通过 Keras 实现 GAN ,其主要过程如下: 正如上图所示,通过调节 Generator 和 Discrim...

网友评论

  • Ingrid060:你好,最近也在跑已经跑好的模型frozen_model.pb,发现run demo的时候输入的只有自己的视频即source视频,跑起来的时候发现target 视频是默克尔演讲的视频,想问:1、默克尔的视频帧是封装在frozen_model.pb里是么。 2、如果想把source视频和target视频都换成自己的,是要拿自己的target视频数据重新训练一个模型么(也就是说一个target视频就需要训练一个模型吗)。靴靴靴靴啦!
  • 我叫小随:哥哥,我的一直报这个错误,已经改了好几个TensorFlow-gpu的版本了,但都没有解决,能帮忙指导一下吗

    NotFoundError (see above for traceback): Key generator/decoder_2/batchnorm/offset not found in checkpoint
    [[Node: save/RestoreV2_1 = RestoreV2[dtypes=[DT_FLOAT], _device="/job:localhost/replica:0/task:0/device:CPU:0"](_arg_save/Const_0_0, save/RestoreV2_1/tensor_names, save/RestoreV2_1/shape_and_slices)]]
    [[Node: save/RestoreV2/_1 = _Recv[client_terminated=false, recv_device="/job:localhost/replica:0/task:0/device:GPU:0", send_device="/job:localhost/replica:0/task:0/device:CPU:0", send_device_incarnation=1, tensor_name="edge_92_save/RestoreV2", tensor_type=DT_FLOAT, _device="/job:localhost/replica:0/task:0/device:GPU:0"]()]]
    sail_73e8:给几个相似问题的网址:
    https://github.com/datitran/face2face-demo/issues/13
    https://github.com/tensorflow/tensorflow/issues/16069

    我也碰到了这个问题,解决方法是去https://github.com/affinelayer/pix2pix-tensorflow 在commits找到April或以前的版本的pix2pix(我用的那个版本对应编码d6f8e4c)下载后替换之前使用的最新版pix2pix 然后再用tensorflow训练(我使用的是1.2.0版)
  • ae55e5117709:您好!请问您这个GIF是怎么做的。?
  • 我是大河:x = tf.placeholder(tf.uint8, shape=(256, 512, 3), name='image_tensor') # input tensor y = generate_output(x) # 输入图片,输出生成的图片 with tf.Session() as sess: # 加载训练好的模型 saver = tf.train.Saver() checkpoint = tf.train.latest_checkpoint(args.input_folder) saver.restore(sess, checkpoint) # 输出新模型 saver = tf.train.Saver() saver.save(sess, './reduced_model')

    这里你怎么restore的?checkpoint里面根本不一样啊?
    我是大河:我解决了
    __Ryan__fz:我也遇到了相同的问题。会不会是tensorflow版本问题?
    操作系统:Windows 10 v1703
    tensorflow-gpu版本:1.8.0
    操作:
    1. 把reduce_model下载到pix2pix-tensorflow目录下。
    2. 在pix2pix-tensorflow目录下打开命令行。
    3. 按照楼主的上述操作,训练完成后模型存于 pix2pix-tensorflow/face2face-model/ 中
    4. 运行以下命令: python reduce_model.py --model-input face2face-model --model-output output_model

    然后出现了同样的错误,如下
    NotFoundError (see above for traceback): Key generator/decoder_1/deconv/filter not found in checkpoint
    斯坦因和他的狗:@我是大河 你的args.input_folder,文件夹可能不正确吧。我跑的过程没有出错呢
  • 904edc8cb590:这个太赞了,请教大神这个跟CVPR2016的那个Face2Face是不是同一个人做的呀~

本文标题:基于GAN的face2face实现

本文链接:https://www.haomeiwen.com/subject/olxsjxtx.html