OpenCV的简单抠图尝试(白色背景的图片)

使用场景

        一键抠图,但是,仅限于大面积背景的物品图片,本脚本主要是针对近白色背景的图片。

需要用到的库

import cv2
import numpy as np
from PIL import Image
import DwkCvShow

先定义背景的颜色范围(注意是HSV色彩空间的范围)

# 定义白色在HSV中的范围
# H(Hue)表示色调,范围是0-179;
# S(Saturation)表示饱和度,范围是0-255;
# V(Value)表示亮度,范围是0-255。
# lower_white = np.array([0, 0, 168])
lower_white = np.array([0, 0, 150])
upper_white = np.array([172, 111, 255])

读取图片并且转换成HSV颜色空间

# 读取图片
img = cv2.imread('./input/002.jpg')

# 将BGR图像转换为HSV图像
hsv = cv2.cvtColor(img, cv2.COLOR_BGR2HSV)
print("转换成hsv色彩空间")
DwkCvShow.resize_and_show_image(hsv)

创建白色掩码并且降噪处理

# 创建一个白色的掩码
mask = cv2.inRange(hsv, lower_white, upper_white)
# 定义结构元素
kernel = np.ones((5, 5), np.uint8)
print("创建一个白色掩码")
DwkCvShow.resize_and_show_image(mask)

# 先腐蚀
eroded = cv2.erode(mask, kernel, iterations=2)
print("创建一个白色掩码先腐蚀")
DwkCvShow.resize_and_show_image(eroded)

# 再膨胀
dilated = cv2.dilate(eroded, kernel, iterations=2)
print("创建一个白色掩码先腐蚀再膨胀")
DwkCvShow.resize_and_show_image(dilated)

反转掩码并且降噪处理

# 反转掩码:将白色变为黑色,将黑色变为白色
mask_inv = cv2.bitwise_not(dilated)
print("反转掩码")
DwkCvShow.resize_and_show_image(mask_inv)

# 定义结构元素
kernel = np.ones((5, 5), np.uint8)

# 先腐蚀
eroded = cv2.erode(mask_inv, kernel, iterations=2)
print("反转掩码后先腐蚀")
DwkCvShow.resize_and_show_image(eroded)
# 再膨胀
dilated = cv2.dilate(eroded, kernel, iterations=2)
print("反转掩码后先腐蚀再膨胀")
DwkCvShow.resize_and_show_image(dilated)

绘制产品物品的最大轮廓

# 最终二值图
# 找到所有轮廓
contours, hierarchy = cv2.findContours(dilated, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)

# 找到最大的轮廓
max_contour = max(contours, key=cv2.contourArea)
# # 使用Douglas-Peucker算法来平滑轮廓
# epsilon = 0.005*cv2.arcLength(max_contour, True)
# approx = cv2.approxPolyDP(max_contour, epsilon, True)

# 创建一个全黑的背景,大小和原图像一样
background = np.zeros_like(dilated)

# 在黑色背景上绘制白色的最大轮廓
# cv2.drawContours(background, [approx], -1, (255), thickness=cv2.FILLED)
cv2.drawContours(background, [max_contour], -1, (255), thickness=cv2.FILLED)
DwkCvShow.resize_and_show_image(background)

抠图并且保存图片

# 使用反转的掩码移除背景
res = cv2.bitwise_and(img, img, mask=background)
print("白色的掩码反转后移除背景")
print("准备写入新的图像")
DwkCvShow.resize_and_show_image(res)

# 将OpenCV图像转换为PIL图像
res = cv2.cvtColor(res, cv2.COLOR_BGR2RGB)
pil_img = Image.fromarray(res)

# 创建一个新的透明背景图像
transparent_img = Image.new("RGBA", pil_img.size)
for x in range(pil_img.width):
    for y in range(pil_img.height):
        r, g, b = pil_img.getpixel((x, y))
        if (r, g, b) != (0, 0, 0):
            transparent_img.putpixel((x, y), (r, g, b, 255))

# 保存透明背景图像
transparent_img.save('颜色掩码方法+轮廓绘制.png', 'PNG')

总结

        简单的算法是很难实现平滑抠图的,需要用到高等数学。大概的方向是计算曲线前面部分导数(当然不仅仅是导数,总之跟曲线的末端切线方向有关)。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CCSBRIDGE

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值