图像分割与修复

发布时间 2023-04-09 16:17:54作者: idazhi

图像的分割与修复

图像的分割

图像分割就是将前景物体从背景中分离出来

图像分割方法
	传统的图像分割方法
		分水岭法
		GrabCut法
		MeanShift法
			基于颜色空间分布的方法
		背景扣除
	基于深度学习的图像分割方法
	

分水龄法


处理步揍
	标记背景
	标记前景
	标记未知域
	进行分割

watershaed(img,masker)
	masker:前景,背景设置不同的值用以区分它们

###距离变换
	计算非零值到离他最近的零值的距离
distanceTransform(img,distanceType,maskSize)
	img:输入的图像 ,会扫描这个图像的所有像素找到菲0值查看周边是否有0值然后计算距离
	distanceType:计算从非0值到0值之间的函数
	maskSize:扫描是lernel的大小

###连通域
connectedComponents(img,connectivity,...)
	img :求图像中所有非0元素的连通域
	connectivity:怎么计算连通域(分上下左右)4:四个点 8:8个点

import cv2
import numpy as np

from matplotlib import pyplot as plt


img = cv2.imread('C:\\Users\dazhi\Desktop\-fenshui.png')

###获取背景
#1通过二值法得到黑白图片
#2通过形态学获取背景
img = cv2.imread('C:\\Users\dazhi\Desktop\-fenshui.png')
#灰度化
gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
#二值化
####ret执行成功或失败,thresh返回值  100最小值:超过之上设置白色之下保留原始数据
#####cv2.THRESH_OTSU自适应阈值 可以把100设为0
ret,thresh = cv2.threshold(gray,0,255,cv2.THRESH_BINARY_INV + cv2.THRESH_OTSU)

###开运算 去除噪点
kernel = np.ones((3,3),np.int8)
open1 = cv2.morphologyEx(thresh,cv2.MORPH_OPEN,kernel,iterations=2)

###膨胀
bg = cv2.dilate(open1,kernel,iterations=1)

###获取前景物体
dist = cv2.distanceTransform(open1,cv2.DIST_L2,5)

###通过plt更直观的查看
#####cmap='gray'灰度
# plt.imshow(dist,cmap='gray')
# plt.show()
# exit()

ret,fg = cv2.threshold(dist,0.7*dist.max(),255,cv2.THRESH_BINARY)

###获取位置区域
fg = np.uint8(fg)
unknow = cv2.subtract(bg,fg)

###创建连通域
ret,marker = cv2.connectedComponents(fg)

marker = marker +1
marker[unknow == 255] = 0

###进行图像分割
result = cv2.watershed(img,marker)
img[result == -1] = [0,0,255]

cv2.imshow("img",img)


cv2.imshow('nuknow',unknow)


cv2.imshow('fg',fg)
cv2.imshow("bg",bg)
cv2.imshow("thresh",thresh)
cv2.waitKey(0)

image-20230409122530654

fenge-02

GrabCut

通过交互的方式获得前景物体

原理
	用户指定前景的大体区域,剩下的为背景区域
	用户还可以明确指定某些地方为前景或背景
	GrabCut采用分段迭代的方法分析前景物体形成模型树
	最后根据权重决定某个像素是前景还是背景

	主题结构
	鼠标事件的处理
	调用GrabCut实现前景与背景的分离
	
###主体程序的实现
###定义两个方法 调用窗口 调用run方法 绑定onmouse窗口

import cv2
import numpy as np

class App:
    ###主流程
    def onmouse(self,event,x,y,flags,param):
        print("onmouse")

    ###对鼠标进行控制

    def run(self):
        print("run...")

        cv2.namedWindow('input')
        ####绑定onmouse窗口
        cv2.setMouseCallback('input',self.onmouse)

        img = cv2.imread('C:\\Users\dazhi\Desktop\-fenshui.png')
        cv2.imshow('input',img)
        cv2.waitKey(0)

###调用run方法
App().run() 
###添加鼠标事件
 
import cv2
import numpy as np

class App:
    ###主流程
    startX = 0
    startY = 0
    flag_rect = False
    def onmouse(self,event,x,y,flags,param):
        ###判断事件能否触发
        if event == cv2.EVENT_LBUTTONDOWN:
            ###当左键按下为True
            self.flag_rect = True
            self.startX = x
            self.startY = y
            print('LBUTTONDOWN')
        elif event == cv2.EVENT_LBUTTONUP:
            ####self.startX,self.startY起始点
            ####(x,y)终止点
            self.flag_rect = False
            cv2.rectangle(self.img,
                          (self.startX,self.startY),
                          (x,y),
                          (0,0,255),3)

            print('LBUTTONUP')
        elif event == cv2.EVENT_MOUSEMOVE:
            ###实现滑动的时候进行框选
            if self.flag_rect == True:

                ###每次都从原始的数据中拷贝一份,每次执行对信图进行绘制
                self.img = self.img2.copy()
                cv2.rectangle(self.img,
                            (self.startX,self.startY),
                            (x,y),
                            (255,0,0),3)
             
            print('MOUSEMOVE')
        print("onmouse")

    ###对鼠标进行控制

    def run(self):
        print("run...")

        cv2.namedWindow('input')
        ####绑定onmouse窗口
        cv2.setMouseCallback('input',self.onmouse)

        self.img = cv2.imread('C:\\Users\dazhi\Desktop\-fenshui.png')
        ###对img进行保存 保存原始的数据
        self.img2 = self.img.copy()
        ###此处while作用是不断刷新 目的是显示所画的框
        while(1):
            cv2.imshow('input',self.img)
            k = cv2.waitKey(100)

            if k == 27:
                break

###调用run方法
App().run() 
###进行图像分割
grabCut(img,mask,rect,bgdModel,fgdModel,5)

	mask分割之后产生的掩码是多少,拿到掩码后就可以将分割后的图像摘取出来
		0: 背景 1: 前景 2:可能是背景  3:可能是前景
	rect 区域 选取一张图的某个区域的时候会有起点和大小 通过鼠标选取的区域
		
	bgdModel,fgdModel背景和前景的model
	5 次数
	
import cv2
import numpy as np

class App:
    ###主流程
    startX = 0
    startY = 0
    flag_rect = False
    rect = (0,0,0,0)
    def onmouse(self,event,x,y,flags,param):
        ###判断事件能否触发
        if event == cv2.EVENT_LBUTTONDOWN:
            ###当左键按下为True
            self.flag_rect = True
            self.startX = x
            self.startY = y
            print('LBUTTONDOWN')
        elif event == cv2.EVENT_LBUTTONUP:
            ####self.startX,self.startY起始点
            ####(x,y)终止点
            self.flag_rect = False
            cv2.rectangle(self.img,
                          (self.startX,self.startY),
                          (x,y),
                          (0,0,255),3)
            ###起点 宽和高
            self.rect = (min(self.startX,x),min(self.startY,y),
                      abs(self.startX - x),
                      abs(self.startY -y))

            print('LBUTTONUP')
        elif event == cv2.EVENT_MOUSEMOVE: 
            ###实现滑动的时候进行框选
            if self.flag_rect == True:

                ###每次都从原始的数据中拷贝一份,每次执行对信图进行绘制
                self.img = self.img2.copy()
                cv2.rectangle(self.img,
                            (self.startX,self.startY),
                            (x,y),
                            (255,0,0),3)
             
            print('MOUSEMOVE')
        print("onmouse")

    ###对鼠标进行控制

    def run(self):
        print("run...")

        cv2.namedWindow('input')
        ####绑定onmouse窗口
        cv2.setMouseCallback('input',self.onmouse)

        self.img = cv2.imread('C:\\Users\dazhi\Desktop\-fenshui.png')
        ###对img进行保存 保存原始的数据
        self.img2 = self.img.copy()

        ###[:2]取前两个值
        self.mask = np.zeros(self.img.shape[:2],dtype=np.uint8)

        self.output = np.zeros(self.img.shape,np.uint8)
        ###此处while作用是不断刷新 目的是显示所画的框
        while(1):
            cv2.imshow('input',self.img)
            cv2.imshow('output',self.output)
            k = cv2.waitKey(100)

            if k == 27:
                break

            if k == ord('g'):

                bgdmodel = np.zeros((1,65),np.float64)
                fgdmodel = np.zeros((1,65),np.float64)
                cv2.grabCut(self.img2,self.mask,self.rect,
                            bgdmodel,fgdmodel,1,
                            cv2.GC_INIT_WITH_RECT)
            mask2 = np.where((self.mask == 1)|(self.mask == 3),255,0).astype('uint8')
            self.output = cv2.bitwise_and(self.img2,self.img2,mask=mask2)


###调用run方法
App().run() 



####后续未完成 指定前景(使用鼠标在一块区域画 就被指定为前景 分割出来)

image-20230409142520920

meanshift

严格来说该方法并不是用来对图像进行分割的,而是在色彩层面的平滑滤波
他会中和色彩分布相近的颜色,平滑色彩细节,侵蚀掉面积较小的颜色区域
以图像上任一点p为圆心,半径为sp 色彩幅值为sr进行不断的迭代

pyrMeanShiftFiltering(img,double sp,double sr)
	double sp 双精度半径 半径越大模糊度越高 
	double sr 色彩幅值变换范围 变化越大连成一片的区域可能性就越大
	
import cv2
import numpy as np

img = cv2.imread('C:\\Users\dazhi\Desktop\meashift.png')

mean_img = cv2.pyrMeanShiftFiltering(img,20,30)

imgcanny = cv2.Canny(mean_img,150,300)

###提取分割的前景
contours,_=cv2.findContours(imgcanny,cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)

cv2.drawContours(img,contours,-1,(0,0,255),2)

cv2.imshow('canny',imgcanny)

cv2.imshow('img',img)
cv2.imshow('mean_img',mean_img)
cv2.waitKey(0)

fenge-04

背景抠图

原理
	视频是一组连续的帧组成(一幅幅图组成) 
	帧与帧之间关系密切(GOP)
	在GOP中背景几乎是不变的
MOG取背景
	混合高斯模型为基础的前景/背景分割算法

	history :默认200 建模时需要多长时间的参考帧
	nmixtures 高斯范围值 默认5
	backgroundRatio 背景比率 默认0.7
	nosizeSigma 默认0 自动降噪
import cv2
import numpy as numpy

cap = cv2.VideoCapture("C://Users/dazhi/Desktop/opencv/video.mp4")

mog = cv2.bgsegm.createBackgroundSubtractorMOG()

while(True):
    ret,frame = cap.read()
    fgmask = mog.apply(frame)

    cv2.imshow('img',fgmask)

    k = cv2.waitKey(10)
    if k ==27:
        break
cap.release()
cv2.destroyAllWindows()

###mog2
###gmg
import cv2
import numpy as numpy

cap = cv2.VideoCapture("C://Users/dazhi/Desktop/opencv/video.mp4")

# mog = cv2.bgsegm.createBackgroundSubtractorMOG()

###mog2
####好处 可以计算出阴影部分
####缺点:会产生很多噪点
# mog = cv2.createBackgroundSubtractorMOG2()

####
####好处 可以计算出阴影部分,同时减少噪点
####缺点 若采用默认值,则在开始好长时间内没有任何信息显示
###解决办法 调整初始参考帧数量
mog = cv2.bgsegm.createBackgroundSubtractorGMG(10)
while(True):
    ret,frame = cap.read()
    fgmask = mog.apply(frame)

    cv2.imshow('img',fgmask)

    k = cv2.waitKey(10)
    if k ==27:
        break
cap.release()
cv2.destroyAllWindows()

图像修复

inpaint(img,mask,inpaintRadius,flags)
	inpaintRadius:修复的半径 每个点的圆形邻域半径
	
import cv2
import numpy as np

img = cv2.imread('xiufu.png')


###掩码图像 可通过鼠标选取获取范围
mask = cv2.imread('xiufu02.png',0)

# print(img.shape)
# exit()
img = cv2.resize(img,(594,628))
mask = cv2.resize(mask,(594,628),0)

dst = cv2.inpaint(img, mask, 5, cv2.INPAINT_TELEA)

# print(dst.shape)
# exit()
cv2.imshow('img',img)
cv2.imshow('dst',dst)
cv2.imshow('mask',mask)
cv2.waitKey()