实现漫画效果---OpenCV-Python开发指南(52)

漫画实现原理

在网上随便找一个漫画的图像,我们都可以清楚的看到漫画都比较粗糙,笔触夸张,也就是说色彩并不细腻,比如漫画的头发,看上去就是一个颜色,不仅如此漫画还保留了很多墨水绘制的线条,因此我们实现漫画的步骤,可以分为如下5步:

  1. 应用双边滤波器来减少图像的色彩
  2. 将彩色图像转换为灰度图像
  3. 使用边缘检测算法提取灰度图像的边缘信息
  4. 对于检测的边缘进行增强并二值化产生粗线条的特征图像
  5. 将3步骤处理的图像与原图像进行叠加,最终实现漫画效果

实现漫画效果的程序步骤

通过上面的文字,我们了解了漫画实现的原理。现在,我们将这些步骤统一转换为代码中的步骤。具体实现步骤如下:

  1. 双边滤波不仅能保留边缘信息,同时也用于减少图像的色彩。所以我们需要使用cv2.bilateralFilter()函数。同时辅助使用高斯金字塔能让图像色彩更加的减少。
  2. 彩色转灰色需要使用cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)函数
  3. 边缘信息获取需要用到cv2.adaptiveThreashold()函数,这是一个图像阈值化处理函数,可以从灰度图像中分离目标区域与背景区域。
    具体的原理如下:因为在灰度图像中,灰度值变化明显的区域往往是物体的轮廓(因为背景大多一样),所以将图像分成一小块一小块地去计算阈值会得出图像的轮廓。
  4. 通过中值滤波可以增强并二值化产生粗线条的特征图像。在程序中,可以先进行中值滤波操作,在进行2操作
  5. 将图像叠加通过“与”操作实现,在OpenCV中,cv2.bitwise_and()函数实现“与”操作

实战实现漫画效果

既然我们已经了解了其实现的原理。下面,我们来通过程序直接实现漫画效果,具体代码如下所示:

# 漫画效果实现
def cortoon_effect(img):
    img_color = img
    for _ in range(3):
        img_color = cv2.pyrDown(img_color)
    for _ in range(7):
        img_color = cv2.bilateralFilter(img_color, 50, 50, 50)
    for _ in range(3):
        img_color = cv2.pyrUp(img_color)
    img_gray = cv2.cvtColor(img, cv2.COLOR_RGB2GRAY)
    img_blur = cv2.medianBlur(img_gray, 5)
    img_edge = cv2.adaptiveThreshold(img_blur, 255, cv2.ADAPTIVE_THRESH_MEAN_C, cv2.THRESH_BINARY, blockSize=5, C=2)
    img_edge = cv2.cvtColor(img_edge, cv2.COLOR_GRAY2RGB)
    print(img_color.shape)
    print(img_edge.shape)
    new_img = cv2.bitwise_and(img_color, img_edge)
    return new_img


if __name__ == "__main__":
    img = cv2.imread("52.jpg")
    img = cv2.resize(img, (400, 600))
    cv2.imshow("0", img)
    cv2.imshow("1", cortoon_effect(img))
    cv2.waitKey()
    cv2.destroyAllWindows()

运行之后,效果如下:


1.png

这里选择的图片颜色有些单调,读者可以自行换一张色彩稍微丰富一点的图片试试。博主这里就不测试了。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容