使用OpenCV和Dlib实现面部换脸
创作时间:
作者:
@小白创作中心
使用OpenCV和Dlib实现面部换脸
引用
CSDN
1.
https://blog.csdn.net/m0_73697499/article/details/143066484
面部换脸技术是计算机视觉领域的一个有趣应用,它通过检测面部特征点并进行图像融合,实现将一张人脸替换到另一张人脸上的效果。本文将详细介绍如何使用OpenCV和Dlib库来实现这一功能。
环境搭建
首先,我们需要安装必要的库。请确保已经安装了opencv-python、numpy以及dlib。如果还未安装,可以通过以下命令进行安装:
pip install opencv-python numpy dlib
注意:由于dlib依赖于CMake和其他一些编译工具,因此在安装过程中可能会遇到一些配置问题。如果安装遇到困难,建议查阅官方文档或寻求社区帮助。
实现步骤
我们将通过以下几个步骤来实现面部特征点检测及面部融合:
- 定义关键点集合:首先定义一组面部特征点集合,包括下巴、眉毛、眼睛、鼻子、嘴巴等部位的关键点。
- 获取面部掩模:根据关键点生成面部的二值掩模。
- 求解仿射变换矩阵:通过关键点计算两幅图像之间的仿射变换矩阵。
- 图像融合:利用仿射变换矩阵将一张图像的脸部特征融合到另一张图像中。
代码实现
- 定义关键点集合
JAW_POINTS = list(range(0, 17))
RIGHT_BROW_POINTS = list(range(17, 22))
LEFT_BROW_POINTS = list(range(22, 27))
NOSE_POINTS = list(range(27, 35))
RIGHT_EYE_POINTS = list(range(36, 42))
LEFT_EYE_POINTS = list(range(42, 48))
MOUTH_POINTS = list(range(48, 61))
FACE_POINTS = list(range(17, 68))
# 关键点集
POINTS = [LEFT_BROW_POINTS + RIGHT_EYE_POINTS +
LEFT_EYE_POINTS + RIGHT_BROW_POINTS + NOSE_POINTS + MOUTH_POINTS]
POINTStuple = tuple(POINTS)
- 获取面部掩模
def getFaceMask(im, keypoints):
im = np.zeros(im.shape[:2], dtype=np.float64)
for p in POINTS:
points = cv2.convexHull(keypoints[p])
cv2.fillConvexPoly(im, points, color=1)
im = np.array([im, im, im]).transpose((1, 2, 0))
im = cv2.GaussianBlur(im, ksize=(25, 25), sigmaX=0)
return im
- 求解仿射变换矩阵
def getM(points1, points2):
points1 = points1.astype(np.float64)
points2 = points2.astype(np.float64)
c1 = np.mean(points1, axis=0)
c2 = np.mean(points2, axis=0)
points1 -= c1
points2 -= c2
s1 = np.std(points1)
s2 = np.std(points2)
points1 /= s1
points2 /= s2
U, S, Vt = np.linalg.svd(points1.T * points2)
R = (U * Vt).T
return np.hstack(((s2 / s1) * R, c2.T - (s2 / s1) * R * c1.T))
- 获取关键点
def getKeypoints(im):
rects = detector(im, 1)
shape = predictor(im, rects[0])
s = np.matrix([[p.x, p.y] for p in shape.parts()])
return s
- 图像融合
def normalColor(a, b):
aGauss = cv2.GaussianBlur(a, (111, 111), 0)
bGauss = cv2.GaussianBlur(b, (111, 111), 0)
weight = aGauss / bGauss
where_are_inf = np.isinf(weight)
weight[where_are_inf] = 0
return b * weight
- 主流程
导入图片
a = cv2.imread("data/hg.png") # 换脸A图片
b = cv2.imread("data/people1.png") # 换脸B图片
导入模型
detector = dlib.get_frontal_face_detector()
predictor = dlib.shape_predictor("data/shape_predictor_68_face_landmarks.dat")
获取关键点,获取图片的人脸掩膜
aKeyPoints = getKeypoints(a) # 获取A图片的68关键点
bKeyPoints = getKeypoints(b) # 获取B图片的68关键点
bOriginal = b.copy() # 不对原来的图片b进行破坏和修改
aMask = getFaceMask(a, aKeyPoints) # 获取图片A的人脸掩膜
cv2.imshow("aMask", aMask)
cv2.waitKey()
bMask = getFaceMask(b, bKeyPoints) # 获取图片B的人脸掩膜
cv2.imshow("bMask", bMask)
cv2.waitKey()
求出b脸仿射变换到a脸的变换矩阵
M = getM(aKeyPoints[POINTStuple], bKeyPoints[POINTStuple])
"""将b的脸部(bmask)根据M仿射变换到a上"""
dsize = a.shape[:2][::-1]
# 目标输出与图像a大小一致
# 需要注意,shape是〈行,列〉,warpAffine参数dsize是〈列,行〉
# 使用a.shape[2:][:,::-1],获取a的〈列,行〉
# 函数warpAffine(src, M, dsize, dst=None, flags=None, borderMode=None, borderValue=None)
# src:输入图像
# M:运算矩阵;2行3列的,
# dsize:运算后矩阵的大小,也就是输出图片的尺寸
# dst:输出图像
# flags:插值方法的组合,与resize函数中的插值一样,可以查看cv2.resize
# borderMode:边界模式, BORDER_TRANSPARENT表示边界透明
# borderValue:在恒定边框的情况下使用的borderValue值,默认情况下,它是 0
bMaskWarp = cv2.warpAffine(bMask, M, dsize,
borderMode=cv2.BORDER_TRANSPARENT,
flags=cv2.WARP_INVERSE_MAP)
cv2.imshow("bMaskWarp", bMaskWarp)
cv2.waitKey()
使用仿射矩阵M,将b映射到a
bWrap = cv2.warpAffine(b, M, dsize,
borderMode=cv2.BORDER_TRANSPARENT,
flags=cv2.WARP_INVERSE_MAP)
cv2.imshow("bWrap", bWrap)
cv2.waitKey()
求b图片的仿射到图片a的颜色值
# 求b图片的仿射到图片a的颜色值,b的颜色值改为a的颜色
bcolor = normalColor(a, bWrap)
cv2.imshow("bcolor", bcolor)
cv2.waitKey()
输出 换脸结果
# ===========step8:换脸(mask区域用bcolor,imask区城用a)=============
out = a * (1.0 - mask) + bcolor * mask
# =========输出原始人脸、换脸结果===============
cv2.imshow("a", a)
cv2.imshow("b", bOriginal)
cv2.imshow("out", out / 255)
cv2.waitKey()
cv2.destroyAllWindows()
代码解析
- 定义关键点集合:根据68个面部特征点的位置,定义了不同的面部区域。
- 获取面部掩模:根据面部特征点生成面部的二值掩模。
- 求解仿射变换矩阵:通过关键点计算两幅图像之间的仿射变换矩阵。
- 图像融合:利用仿射变换矩阵将一张图像的脸部特征融合到另一张图像中,并调整颜色使其更加自然。
总结
通过上述代码,我们实现了基于OpenCV和Dlib的面部特征点检测及面部融合功能。面部特征点检测是很多计算机视觉应用的基础,而面部融合则是其中一种有趣的应用实例。希望本文能够帮助你更好地理解和应用这些技术,同时也鼓励你在实践中不断探索新的应用场景和技术。
热门推荐
孩子上课传纸条该如何教育
汽车保养脱保的影响有哪些?怎样避免汽车保养脱保?
髋关节积液:从病因到治疗的全面解析
臀肌挛缩症保守治疗方法
海南必吃十大特色美食:从文昌鸡到椰奶糕,尽显海岛风味
「狗狗能吃&不能吃的天然蔬果」有哪些?蔬果營養成分、危害一次告訴你
注意力机制:让机器学会“挑重点”
近视手术到底安不安全?有哪些风险?央视给出了答案……
规划未来,享受金色年华,如何制定您的养老金方案
游戏产业与文化融合的典范,《魔域》IP的多元探索与展望
怀孕能喝鸭屎香柠檬茶吗
守护“海底草原”:破局中国海草床生态保护与修复
中国的脊梁人物及事迹:了解这些英雄人物的事迹
雕塑的魅力:探索艺术品的来源、制作周期与公司背后的故事
在官方店买到二手手机能索赔吗
打羽毛球打多久?这份科学运动指南请收好
【以案释法】夫妻共同债务那些事儿——从一起借款纠纷看法律规定
动物世界与人类社会,这些中短篇新作如何呈现?
房屋租赁个体经营如何办税:法律规定与实务操作指南
惊悚喜剧《诡才之道》:反类型创作引人共鸣
破局“幽灵外卖”,威海开始给“看不见的后厨”装上“电子眼”
寻找国内种植牙技术领先的医院推荐:揭秘顶尖口腔医疗机构有哪些
食品中金黄色葡萄球菌检测技术!
左心房增大是什么原因引起的
香菜对高血压有好处吗?研究这样说
鲍鱼不能和什么食物一起吃
cad好学吗?怎么入门?过来人经验谈,让你少走弯路
企业不开票收入的账务处理与税务申报指南
蚊子叮咬不可怕,可怕的是登革热!
R7 7745HX和R7 7840H性能对比:差距有多大?