实现抠图&替换背景图

简述

上一篇博客进行了证件照更换背景颜色,纯蓝色,红色,白色之间的替换,有人私信我,可以不可以把背景换成其他图片,而不是单纯的颜色填充。这在photoshop里面就是选中一个图层然后复制到另外一张图片上去,用代码实现的话和上篇博文换纯色背景思路完全一样,只是在替换颜色时候有了新变化。

获取目标区域(抠图)

将目标区域和背景分离开。

此方法抠图只适合颜色对比比较明显的图片,允许存在少量颜色干扰

加载&缩放

通过imread函数加载图片,resize函数对图像进行缩放。 
(因为找的图片有些大,显示器太小,所以适当缩放)

import cv2
import  numpy as np

img=cv2.imread('zjz.jpg')
img_back=cv2.imread('back.jpg')
#日常缩放
rows,cols,channels = img_back.shape
img_back=cv2.resize(img_back,None,fx=0.7,fy=0.7)
cv2.imshow('img_back',img_back)

rows,cols,channels = img.shape
img=cv2.resize(img,None,fx=0.4,fy=0.4)
cv2.imshow('img',img)
rows,cols,channels = img.shape#rows,cols最后一定要是前景图片的,后面遍历图片需要用到
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

两张图片如下 
这里写图片描述 
(图片源于网络,已经马赛克处理,如有侵权,私信立即删除)

要实现的效果就是,把人物图像抠出来,放在背景图片上面。

获取背景区域

由于背景纯蓝色,所以找到了这些区域,相反的就是我们想要的。 
这里要用到inRange这个函数获取蓝色区域。 
首先需要将图片转换为HSV类型。

#转换hsv
hsv=cv2.cvtColor(img,cv2.COLOR_BGR2HSV)
  • 1
  • 2

获取mask得到蓝色区域

#获取mask
lower_blue=np.array([78,43,46])
upper_blue=np.array([110,255,255])
mask = cv2.inRange(hsv, lower_blue, upper_blue)
cv2.imshow('Mask', mask)
  • 1
  • 2
  • 3
  • 4
  • 5

蓝色区域mask如下图 
这里写图片描述 
黑色区域有明显白点,有少量的颜色干扰,需要进一步优化。

mask优化

通过腐蚀和膨胀操作进行消除个别白点。 
我对于腐蚀和膨胀操作的理解是:

腐蚀操作将会腐蚀图像中白色像素,以此来消除小斑点, 
而膨胀操作将使剩余的白色像素扩张并重新增长回去。


#腐蚀膨胀
erode=cv2.erode(mask,None,iterations=1)
cv2.imshow('erode',erode)
dilate=cv2.dilate(erode,None,iterations=1)
cv2.imshow('dilate',dilate)
  • 1
  • 2
  • 3
  • 4
  • 5

优化后如下图 
这里写图片描述 
黑色区域内白点已经消除,完美分离人物与背景[傲娇]。

替换背景图片

此时已经将图片目标区域抠出来了,只需要再新的背景图上把抠出来的对应点颜色填充上去就好。 
我们首先要确定一个坐标点,这个点决定了要把抠出来的图像放到新背景图片的什么位置,即就是抠出图片左上角(0,0)点在新的背景图片中应该在的位置。 
注意:

扣出的图片应该小于背景图片,确定位置时候应注意,坐标越界后 
会发生异常。注意协调。

#遍历替换
center=[50,50]#在新背景图片中的位置
for i in range(rows):
    for j in range(cols):
        if dilate[i,j]==0:#0代表黑色的点
            img_back[center[0]+i,center[1]+j]=img[i,j]#此处替换颜色,为BGR通道
cv2.imshow('res',img_back)
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

最终效果 
这里写图片描述 
(图片源于网络,已经马赛克处理,如有侵权,私信立即删除)

总结

基本原理和上篇博客所讲相同,重点在于后面的center点的确定(谨防越界),和不同图片间相对应的坐标和颜色的相互复制替换。

完整代码如下:

import cv2
import  numpy as np

img=cv2.imread('zjz.jpg')
img_back=cv2.imread('back.jpg')
#日常缩放
rows,cols,channels = img_back.shape
img_back=cv2.resize(img_back,None,fx=0.7,fy=0.7)
cv2.imshow('img_back',img_back)

rows,cols,channels = img.shape
img=cv2.resize(img,None,fx=0.4,fy=0.4)
cv2.imshow('img',img)
rows,cols,channels = img.shape#rows,cols最后一定要是前景图片的,后面遍历图片需要用到

#转换hsv
hsv=cv2.cvtColor(img,cv2.COLOR_BGR2HSV)
#获取mask
lower_blue=np.array([78,43,46])
upper_blue=np.array([110,255,255])
mask = cv2.inRange(hsv, lower_blue, upper_blue)
cv2.imshow('Mask', mask)

#腐蚀膨胀
erode=cv2.erode(mask,None,iterations=1)
cv2.imshow('erode',erode)
dilate=cv2.dilate(erode,None,iterations=1)
cv2.imshow('dilate',dilate)

#遍历替换
center=[50,50]#在新背景图片中的位置
for i in range(rows):
    for j in range(cols):
        if dilate[i,j]==0:#0代表黑色的点
            img_back[center[0]+i,center[1]+j]=img[i,j]#此处替换颜色,为BGR通道
cv2.imshow('res',img_back)

cv2.waitKey(0)
cv2.destroyAllWindows()
  • 2
    点赞
  • 53
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
iOS上使用OpenCV实现抠图背景图替换功能是可行的。OpenCV是一个开源的计算机视觉库,它提供了各种图像处理和分析技术,包括图像分割和图像合成等功能,非常适合进行图像背景替换。 具体实现步骤如下: 1. 导入OpenCV库:在iOS项目中,首先需要将OpenCV库导入到项目中,并进行相应的配置。 2. 图像分割:使用OpenCV提供的图像分割算法,如GrabCut算法,将目标对象与背景进行分离。该算法需要输入一张包含目标对象的图像,并对其进行初始化,引导GrabCut算法进行分割。 3. 背景替换:根据分割得到的目标对象,将它与另一张背景图像进行合成。可以使用OpenCV提供的透明度混合函数,将目标对象与背景图像进行混合。具体操作是通过像素级的合成运算,计算目标对象像素与背景图像像素之间的混合比例,从而实现替换背景的效果。 4. 后续处理:根据需求,可以对合成后的图像进行调整和优化。例如,可以对合成后的图像进行色彩调整、亮度调整或者模糊处理,以使合成的结果更加自然。 需要注意的是,在实现抠图背景图替换功能时,选择合适的图像分割算法和优化方法是非常重要的。这可能需要根据具体需求和实际情况进行调试和优化。 总结起来,使用iOS上的OpenCV库可以很好地实现抠图背景图替换功能。通过图像分割和背景替换等技术手段,可以将目标对象与不同的背景进行合成,实现抠图背景图替换的效果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值