opencv使用GrabCut算法提取前景物体

void grabCut(InputArray image, InputOutputArray mask, Rect rect, InputOutputArray bgdModel, InputOutputArray fgdModel, int iterCount, int mode)
 
 

**********参数说明****************

  • image –输入8位3通道图像

  • mask

               输入\输出的8位单通道mask。当mode设为GC_INIT_WITH_RECT时,由函数初始化,  当mode设为GC_INIT_WITH_MASK,允许用户输入自己标记的掩码。

mask的取值为以下四种:

    • GC_BGD (=0)defines an obvious background pixels.
    • GC_FGD(=1) defines an obvious foreground (object) pixel.
    • GC_PR_BGD(=2) defines a possible background pixel.
    • GC_PR_BGD (=3)defines a possible foreground pixel.
  • rect – 包含前景的矩形
  • bgdModel – 背景模型. Do not modify it while you are processing the same image.
  • fgdModel – 前景模型. Do not modify it while you are processing the same image.
  • iterCount – 迭代次数
  • mode –操作代码,可能为以下值:

    • GC_INIT_WITH_RECT -----矩形框初始化函数;
    • GC_INIT_WITH_MASK ------mask初始化函数;
    • GC_EVAL ---执行分割。


简单的例子:


 
 
  1. #include"highgui.h"
  2. #include"cv.h"
  3. using namespace cv;
  4. Rect selection;
  5. Mat img,img0;
  6. Point prePt(-1,-1);
  7. void onMouse(int event,int x,int y,int flags,void* param){
  8. if (event==CV_EVENT_LBUTTONDOWN){
  9. prePt = Point(x, y);
  10. }
  11. else if (event == CV_EVENT_MOUSEMOVE && (flags && CV_EVENT_FLAG_LBUTTON)){
  12. img = img0.clone();
  13. rectangle(img, Rect(prePt.x, prePt.y, abs(x - prePt.x), abs(y - prePt.y)), Scalar( 0, 0, 255), 3);
  14. selection = Rect(prePt.x, prePt.y, abs(x - prePt.x), abs(y - prePt.y));
  15. }
  16. else if (event == CV_EVENT_LBUTTONUP || !(flags & CV_EVENT_FLAG_LBUTTON)){
  17. prePt=Point( -1, -1);
  18. }
  19. imshow( "原图", img);
  20. }
  21. int main(){
  22. img = imread( "D:\\nvshen.jpg", 1);
  23. img0 = img.clone();
  24. namedWindow( "原图", 1);
  25. imshow( "原图", img);
  26. setMouseCallback( "原图", onMouse);
  27. Mat result; // 4种可能结果
  28. Mat bgModel, fgModel; // 背景/前景
  29. for (;;){
  30. int c = waitKey( 10);
  31. if (( char)c == 'p'){
  32. grabCut(img0, result, selection, bgModel, fgModel, 5, GC_INIT_WITH_RECT);
  33. compare(result,GC_PR_FGD,result,CMP_EQ); //得到前景mask
  34. Mat foreground(img.size(),CV_8UC3,Scalar::all(255));
  35. img0.copyTo(foreground,result);
  36. imshow( "grabcut",foreground);
  37. }
  38. if ( char(c) == 'q') return 0;
  39. }
  40. return 0;
  41. }

笔记:

compare(result,GC_PR_FGD,result,CMP_EQ);
 
 
本例的结果标记像素包含可能属于前景的像素(GC_PR_FGD)和可能属于背景的像素(GC_PR_BGD),矩形之外的为确定属于背景的像素(GC_BGD),不包含GC_FGD。

通过result与可能属于前景的像素作比较,得到前景mask即可提取前景。

另一种方法是 result&=1;替代上句。

四种可能的结果值分别为0,1,2,3.即00000000,00000001,00000010,00000011.因为结果不包含1,所以核对第一位即可。

本例只是大致提取了前景,通过输入带标记的mask可提高准确度。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值