阅读(2305) (0)

OpenCV添加边框到您的图像

2017-09-07 10:50:47 更新

目标

在本教程中,您将学习如何:

理论

注意
下面的解释属于Bradski和Kaehler 的“ 学习OpenCV ”一书。
  1. 在前面的教程中,我们学会了使用卷积来对图像进行操作。自然产生的一个问题是如何处理边界。如果评估点位于图像的边缘,我们如何卷积它们?
  2. 大多数OpenCV功能是将给定的图像复制到另一个稍大的图像上,然后自动填充边界(通过下面的示例代码中解释的任何方法)。这样,可以在所需要的像素上执行卷积而没有问题(在操作完成之后,额外的填充被切割)。
  3. 在本教程中,我们将简要介绍如何为图像定义额外的填充(边框):BORDER_CONSTANT:使用常量值(即黑色或0)来填充图像0BORDER_REPLICATE:原始边缘的行或列被复制到额外的边框。“守则”部分将会更清楚地看到这一点。

Code

  1. 这个程序是做什么的?
    • 加载图像
    • 让用户选择在输入图像中使用什么样的填充。有两个选择:
      1. 常数值边框:对整个边框应用一个常量值的填充。该值将随机每0.5秒更新一次。
      2. 复制边框:边框将从原始图像边缘的像素值复制。
    • 当用户按“ESC”时程序完成
  2. 教程代码如下所示。您也可以从这里下载
#include "opencv2/imgproc.hpp"
#include "opencv2/imgcodecs.hpp"
#include "opencv2/highgui.hpp"
using namespace cv;
Mat src, dst;
int top, bottom, left, right;
int borderType;
const char* window_name = "copyMakeBorder Demo";
RNG rng(12345);
int main( int argc, char** argv )
{
  String imageName("../data/lena.jpg"); // by default
  if (argc > 1)
  {
      imageName = argv[1];
  }
  src = imread( imageName, IMREAD_COLOR ); // Load an image
  if( src.empty() )
    {
      printf(" No data entered, please enter the path to an image file \n");
      return -1;
    }
  printf( "\n \t copyMakeBorder Demo: \n" );
  printf( "\t -------------------- \n" );
  printf( " ** Press 'c' to set the border to a random constant value \n");
  printf( " ** Press 'r' to set the border to be replicated \n");
  printf( " ** Press 'ESC' to exit the program \n");
  namedWindow( window_name, WINDOW_AUTOSIZE );
  top = (int) (0.05*src.rows); bottom = (int) (0.05*src.rows);
  left = (int) (0.05*src.cols); right = (int) (0.05*src.cols);
  dst = src;
  imshow( window_name, dst );
  for(;;)
       {
         char c = (char)waitKey(500);
         if( c == 27 )
           { break; }
         else if( c == 'c' )
           { borderType = BORDER_CONSTANT; }
         else if( c == 'r' )
           { borderType = BORDER_REPLICATE; }
         Scalar value( rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255) );
         copyMakeBorder( src, dst, top, bottom, left, right, borderType, value );
         imshow( window_name, dst );
       }
  return 0;
}

说明

  • 首先我们声明我们要使用的变量:
Mat src, dst;
int top, bottom, left, right;
int borderType;
const char* window_name = "copyMakeBorder Demo";
RNG rng(12345);

特殊的注意力值应该是随机数发生器的变量rng。我们用它来生成随机边框颜色,我们将尽快看到。

  • 像往常一样,我们加载我们的源图像src
  String imageName("../data/lena.jpg"); // by default
  if (argc > 1)
  {
      imageName = argv[1];
  }
  src = imread( imageName, IMREAD_COLOR ); // Load an image
  if( src.empty() )
    {
      printf(" No data entered, please enter the path to an image file \n");
      return -1;
    }
  • 在给出如何使用程序的简短介绍后,我们创建一个窗口:
  namedWindow(window_name,WINDOW_AUTOSIZE);
  • 现在我们初始化定义边框大小的参数(topbottomleft and right)。我们给它们一个src大小的5%的值。
  top =(int)(0.05 * src.rows); bottom =(int)(0.05 * src.rows);
  left =(int)(0.05 * src.cols); right =(int)(0.05 * src.cols);
  • 程序运行在for循环中。如果用户按“c”或“r”,则borderType变量分别取BORDER_CONSTANTBORDER_REPLICATE的值:
         char c = (char)waitKey(500);
         if( c == 27 )
           { break; }
         else if( c == 'c' )
           { borderType = BORDER_CONSTANT; }
         else if( c == 'r' )
           { borderType = BORDER_REPLICATE; }
  • 在每次迭代(0.5秒后),变量被更新...
         Scalar value( rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255) );

RNG变量rng生成的随机值。该值是随机抽取的数字范围[0,255]

         copyMakeBorder(src,dst,top,bottom,left,right,borderType,value);

参数解释:

  1. src:源图像
  2. dst:目的地图像
  3. topbottomleftright:图像每边边框的长度(以像素为单位)。我们将它们定义为图像的原始大小的5%。
  4. borderType:定义应用什么类型的边框。对于该示例,它可以是常量或复制的。
  5. value:如果borderType为BORDER_CONSTANT,则这是用于填充边框像素的值。
  • 我们在以前创建的图像中显示输出图像
         imshow(window_name,dst);

结果

  1. 在编译上面的代码之后,可以执行它作为参数作为图像的路径。结果应该是:

  • 默认情况下,它的边框设置为BORDER_CONSTANT。因此,将显示连续的随机彩色边界。
  • 如果按“r”,边框将成为边缘像素的副本。
  • 如果按“c”,随机彩色边框将再次出现
  • 如果按“ESC”,程序将退出。

下面的屏幕截图显示边框如何更改颜色以及BORDER_REPLICATE选项的外观:

OpenCV添加边框到您的图像