| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 人工智能 -> 【学习笔记】李宏毅2021春机器学习课程第6.1节:生成式对抗网络 GAN(一) -> 正文阅读 |
|
[人工智能]【学习笔记】李宏毅2021春机器学习课程第6.1节:生成式对抗网络 GAN(一) |
文章目录1 能够作为生成器的神经网络 GAN
generative adversarial network,它的缩写是GAN,中文名称生成式对抗网络,这也是不少人在还没有接触机器学习之前都有听说过的一个很出名的模型。它其实有很多各式各样的变形,你可以在网路上找到一个GAN的动物园: 2 动漫人物头像生成直接通过一个例子来介绍什么是GAN,以及GAN要做什么,怎么实现的。假设我们现在的任务是让机器生成二次元人物的头像,假设现在是 unconditional generation,就是没有输入x,只有输入的一个随机变量z 那之后我们在讲到 conditional generation 的时候,我们会再把x加回来,那输入的这个z是什么呢? 实际上我们可以假设z是从一个normal distribution里采样出来的向量,这个向量通常会是一个 low-dimensional 的向量,一般定位50,100维,它的大小是由你自己决定的。 那现在我们从一个normal distribution里面采样出一个向量z,然后输入给GAN,GAN就给你一个对应的输出,那我们希望对应的输出就是一个二次元人物的脸。而一张图片就是一个非常高维的向量,所以generator实际上做的就是产生一个非常高维的向量。 当你输入的向量不同的时候,你的输出就会跟着改变,所以你从这个normal distribution里面采样到不同的z,那么每次输出的y也就不一样,但我们希望不管采样到什么z,输出来的都是动漫人物的脸。 3 判别器(Discriminator)在GAN里面一个特别的地方就是,除了generator以外,我们要多训练一个discriminator。discriminator的作用是,输入一张图片,输出是一个数值,这个数值越大就代表现在输入的这张图片越像是真实的二次元人物的图像。 而discriminator的架构完全是你自己设计的,你可以用CNN,也可以用 transformer 等等,只要能够产生出你想要的输入输出,就可以了。 在这个例子里面,因为discriminator的输入是一张图片,很显然选择CNN很比较有优势,毕竟CNN在处理影像上有很多优点。 4 从自然选择看GAN的基本思想为什么除了生成器之外,我们还需要多训练一个判别器呢,这里其实GAN的基本思想可以从生物进化的角度来看,我这里复述一下李宏毅老师举的一个比较有趣的例子: 上面这张图似乎没什么特别的哈?不就是一个树枝和一篇枯叶嘛,但其实这不是一片枯叶,这是枯叶蝶的拟态,枯叶蝶长得跟枯叶非常像,因此它可以躲避天敌。但枯叶蝶的祖先其实并不是长得像枯叶一样,也许他们原来也是五彩斑斓的,但为什么他们变成长得像枯叶一样,是因为有天择的压力。 这个不是普通的麻雀,这个是波波(一种宝可梦),波波会吃枯叶蝶的祖先,在天择的压力之下,枯叶蝶就变成棕色的。因为波波它只会吃彩色的东西,它看到彩色的东西知道是蝴蝶,就把它吃掉,那看到棕色的东西,波波就不会去吃它。 但是逐渐这样下去,只会吃彩色蝴蝶的波波慢慢的找不到足够的食物,也会被大自然淘汰了,在自然选择中获胜的波波都进化了,它们进化成了比比鸟,比比鸟在判断一个蝴蝶能不能吃的时候不会只看颜色,它会看它的纹路,它知道说没有叶脉的是蝴蝶,有叶脉的才是真正的枯叶。 在天择的压力之下,枯叶蝶就产生了拟态,产生了叶脉,想要骗过比比鸟,但是比比鸟它也有可能会再进化成大比鸟,那大比鸟可能可以分辨枯叶蝶跟枯叶的不同。 那这个是一个物种演化的故事,对应到GAN中的相关概念,枯叶蝶就是generator,那波波就是discriminator。 回到我们之前的例子中来,现在我们generator要做的事情,是画出二次元的人物头像,那generator学习画出二次元人物头像的过程是这样的: 第一代的generator它的参数几乎是完全随机的,所以它根本就不知道到底要怎么画二次元的人物,所以它画出来的东西就是一些莫名其妙的东西。 那discriminator学习的目标是分辨generator的输出与真正的动漫头像的不同,在现在的状况下可能非常的容易,对discriminator来说它只要看图片里面有没有两个黑黑的圆球,有眼睛就是真正的二次元人物,没有眼睛就是generator产生出来的东西。 接下来generator就调整它的里面的参数,Generator就进化了,它调整它里面的参数,调整的目标是为了骗过discriminator,假设discriminator判断一张图片是不是真实的依据是有没有眼睛,那generator就产生眼睛出来,以期能够骗过discriminator: 所以第二代的generator可以产生眼睛,这样就可以骗过第一代的discriminator,但是discriminator也是会进化的,第二代的discriminator会试图分辨generator产生的图片,跟真实图片之间的差异,它可能会发现说,generator产生的图片都没有头发也没有嘴巴,真实图片是有头发的也有嘴巴的。 接下来第三代的generator就会想办法去骗过第二代的discriminator,既然第二代的discriminator是看有没有头发和嘴巴来判断是不是真正的二次元人物,那第三代的generator就会把嘴巴加上去。 那discriminator也会逐渐的进步,它会越来越严苛,在这样的左右互搏之后,我们期望Generator产生出来的图片可以越来越像真实二次元的人物。 可以看到我们的generator跟discriminator中间有一个对抗的关系,所以就用了adversarial这个词语,总而言之,generator跟discriminator既是对抗的,也是互相成就的,其中任何一方太弱的话,都会导致双方都训练不起来。 5 GAN 的具体实现过程接下来介绍一下GAN的具体实现过程是怎样的,generator和discriminator,他们就是两个network,我们假设generator跟discriminator的参数都已经初始化过了。 步骤一: 固定 generator G 的参数,只更新discriminator D初始化完以后,接下来训练的第一步是,固定住你的generator的参数,只训练你的discriminator。 因为一开始generator的参数都是随机初始化的,并且我们又固定住了generator的参数,那它的输出完全都是乱七八糟的图片。 那假设我们有一个database,这个database里面有很多二次元人物的头像,从这个图库里面去sample一些二次元人物的头像出来,用这些真正的二次元人物头像,跟generator产生出来的结果,去训练你的discriminator。 discriminator训练的目标是要分辨真正的二次元人物跟generator产生出来的二次元人物之间的差异,具体一点来说,实际上我们可能会把这些真正的人物都标1,Generator产生出来的图片都标0。 接下来对于discriminator来说,这就是一个分类的问题,你就把真正的人脸当作类别1,Generator产生出来的图片当作类别2,然后训练一个classifier就结束了。或者看做一个回归的问题,输出的值越接近1,就代表越接近真实图片;而越接近1,就代表越接近假的图片,这两种办法都可以。 步骤二: 固定 discriminator D 的参数,只更新generator G我们训练完discriminator以后,接下来固定住discriminator,改为训练generator。 我们要让generator想办法去骗过discriminator,因为刚才discriminator已经学会分辨真图跟假图的差异,generator产生的图片如果可以骗过discriminator,那在discriminator足够强大的情况下,生成的图片就可以假乱真了。 实际的操作方法是这样的,你有一个generator,generator从gaussian distribution sample出来一个向量作为输入,然后输出一个图片的向量。 接下来我们把这个图片输入到Discriminator里面,Discriminator会给这个图片一个分数,分数越高表示越接近真实图片。 那Generator训练的目标就是要Discriminator输出的值越大越好,如果Generator调整参数之后输出的图片可以蒙骗Discriminator,也就是Discriminator会给予高分,那意味着Generator产生出来的图片是比较真实的。 所以现在讲了两个步骤
接下来就是重复这两个步骤反复的训练discriminator和generator,期待discriminator跟generator都可以做得越来越好,直到generator产生图片的效果能让我们比较满意。 6 动漫头像生成的具体实验结果以下的结果是李宏毅老师在17年的时候做的, dataset的地址: https://zhuanlan.zhihu.com/p/24767059,训练了100个来回之后的结果如下: 这时generator还不知道在做些什么,但训练了1000个回合后的结果如下: discriminator 和 generator 各自训练这样反复一千次以后,机器就产生了眼睛,机器知道说人脸就是要有两个眼睛,所以它就把眼睛标上去,训练到两千次的时候,你发现嘴巴就出来了: 训练到五千次的时候,已经开始有一点人脸的样子了,而且你发现说机器学到了动画人物啊,就是要有那个水汪汪的大眼睛,所以它给每个人的眼睛呢都涂得非常的大: 接下来是训练两万个回合的结果: 然后是训练五万个回合的结果: 那你会发现这些生成的动漫人物大体上还不错,只是有一些比较崩坏,如果你有真的非常好的资料的话,也许你可以做出真的很好的结果。 这里有一个链接: https://www.gwern.net/images/gan/stylegan/2019-02-11-stylegan-danbooru2017faces-interpolation.mp4 这个是用StyleGAN做的,那用StyleGAN可以做到这个地步: 可以看到效果已经相当好了,完全辨别不出来是机器自己产生的,这个结果还是很惊人的。 除了产生动画人物以外,当然也可以产生真实的人脸,有一个技术叫做progressive GAN,它可以产非常高清的人脸,你可能会问产生人脸有什么用呢,我去路边拍一个人产生出来的照片不是更像真的吗? 但是用GAN你可以产生你没有看过的人脸,甚至根本不存在的人脸。这就是GAN的神奇之处。 |
|
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 | -2025/1/2 22:51:31- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |