OpenCV-基本边缘检测算子Sobel实现
2018-02-12 17:27
816 查看
简要描述
sobel算子主要用于获得数字图像的一阶梯度,常见的应用是边缘检测。
原理
算子使用两个3*3的矩阵(图1)算子使用两个3*3的矩阵(图1)去和原始图片作卷积,分别得到横向G(x)和纵向G(y)的梯度值,如果梯度值大于某一个阈值,则认为该点为边缘点
图1:卷积矩阵
图2:卷积运算
图像直接卷积实现Sobel://图像直接卷积实现Sobel
bool sobelEdge(const cv::Mat image, cv::Mat &result, uchar threshold)
{
CV_Assert(image.channels() == 1);
//初始化水平核因子
cv::Mat sobelx = (cv::Mat_<float>(3, 3) << 1, 0, -1, 2, 0, -2, 1, 0, -1);
//初始化垂直核因子
cv::Mat sobely = (cv::Mat_<float>(3, 3) << 1, 2, 1, 0, 0, 0, -1, -2, -1);
result = cv::Mat::zeros(image.rows - 2, image.cols - 2, image.type());
double graMag = 0;
for (int i = 1; i < image.rows - 1; i++)
{
for (int j = 1; j < image.cols - 1; j++)
{
float edgex = 0, edgey = 0;
//遍历计算水平与垂直梯度
for (int k = -1; k < 2; k++)
{
for (int p = -1; p < 2; p++)
{
edgex += (float)image.at<uchar>(k + i, p + j) * sobelx.at<float>(1 + k, 1 + p);
edgey += (float)image.at<uchar>(k + i, p + j) * sobely.at<float>(1 + k, 1 + p);
}
}
//计算梯度模长
graMag = sqrt(pow(edgex, 2) + pow(edgey, 2));
//二值化
result.at<uchar>(i - 1, j - 1) = ((graMag > threshold) ? 255 : 0);
}
}
return true;
}
运行结果:
非极大值抑制Sobel边缘实现://非极大值抑制实现Sobel竖直细化边缘
bool SobelVerEdge(cv::Mat image, cv::Mat &result)
{
CV_Assert(image.channels() == 1);
image.convertTo(image, CV_32FC1);
//水平方向的Sobel算子
cv::Mat sobelx = (cv::Mat_<float>(3, 3) << -0.125, 0, 0.125, -0.25, 0, 0.25, -0.125, 0, 0.125);
//卷积操作
cv::Mat ConResMat;
cv::filter2D(image, ConResMat, image.type(), sobelx);
//计算梯度的幅度
cv::Mat graMagMat;
cv::multiply(ConResMat, ConResMat, graMagMat);
//根据梯度幅度及参数设置阈值(mean获取各个通道的均值)
int scaleVal = 4;
double thresh = scaleVal * cv::mean(graMagMat).val[0];
cv::Mat resulttempMat = cv::Mat::zeros(graMagMat.size(), graMagMat.type());
float *pDataMag = (float*)graMagMat.data;
float *pDataRes = (float*)resulttempMat.data;
const int rows = ConResMat.rows, cols = ConResMat.cols;
for (int i = 1; i != rows - 1; i++)
{
for (int j = 1; j != cols - 1; j++)
{
//计算该点梯度与水平或垂直梯度值的大小并比较结果
bool b1 = (pDataMag[i * cols + j] > pDataMag[i * cols + j - 1]);
bool b2 = (pDataMag[i * cols + j] > pDataMag[i * cols + j + 1]);
bool b3 = (pDataMag[i * cols + j] > pDataMag[(i - 1) * cols + j]);
bool b4 = (pDataMag[i * cols + j] > pDataMag[(i - 1) * cols + j]);
//判断邻域梯度是否满足大于水平或垂直的条件
//并根据自适应阈值参数进行二值化
pDataRes[i * cols + j] = 255 * ((pDataMag[i * cols + j] > thresh) && ((b1 && b2) || (b3 && b4)));
}
}
resulttempMat.convertTo(resulttempMat, CV_8UC1);
result = resulttempMat.clone();
return true;
}
运行结果:
在OpenCV中,sobel算子在C++中的函数原型如下:
void Sobel(InputArray src, OutputArray dst, int ddepth, int dx, int dy, int ksize=3, double scale=1, double delta=0, int borderType=BORDER_DEFAULT ) InputArray src:输入的原图像,Mat类型
OutputArray dst:输出的边缘检测结果图像,Mat型,大小与原图像相同。
int ddepth:输出图像的深度,针对不同的输入图像,输出目标图像有不同的深度,具体组合如下:
- 若src.depth() = CV_8U, 取ddepth =-1/CV_16S/CV_32F/CV_64F
- 若src.depth() = CV_16U/CV_16S, 取ddepth =-1/CV_32F/CV_64F
- 若src.depth() = CV_32F, 取ddepth =-1/CV_32F/CV_64F
- 若src.depth() = CV_64F, 取ddepth = -1/CV_64F
注:ddepth =-1时,代表输出图像与输入图像相同的深度。
int dx:int类型dx,x 方向上的差分阶数,1或0
int dy:int类型dy,y 方向上的差分阶数,1或0
其中,dx=1,dy=0,表示计算X方向的导数,检测出的是垂直方向上的边缘;dx=0,dy=1,表示计算Y方向的导数,检测出的是水平方向上的边缘。
int ksize:为进行边缘检测时的模板大小为ksize*ksize,取值为1、3、5和7,其中默认值为3。
代码实现:int main()
{
cv::Mat image = cv::imread("1.jpg", 0);
if (image.empty()) return -1;
cv::imshow("image", image);
cv::Mat edgeMat, edgexMat, edgeyMat;
//求x方向Sobel边缘
cv::Sobel(image, edgexMat, CV_16S, 1, 0, 3, 1, 0, cv::BORDER_DEFAULT);
//求y方向Sobel边缘
cv::Sobel(image, edgeyMat, CV_16S, 0, 1, 3, 1, 0, cv::BORDER_DEFAULT);
//线性变换,转换输入数组元素为8位无符号整形
cv::convertScaleAbs(edgexMat, edgexMat);
cv::convertScaleAbs(edgeyMat, edgeyMat);
//x与y方向边缘叠加
cv::addWeighted(edgexMat, 0.5, edgeyMat, 0.5, 0, edgeMat);
cv::imshow("edgeMat", edgeMat);
cv::waitKey(0);
return 0;
}
运行结果:
相关文章推荐
- OpenCV之imgproc 模块. 图像处理(2)实现自己的线性滤波器 给图像添加边界 Sobel 导数 Laplace 算子 Canny 边缘检测
- OpenCV环境下Laplace(拉普拉斯)和Roberts基本边缘检测算子的实现代码
- 利用Canny边缘检测算子进行边缘检测的原理及OpenCV代码实现
- OpenCV-边缘检测算子Canny实现
- OpenCV使用Sobel滤波器实现图像边缘检测
- 基于OPENCV的CANNY边缘检测算子详细代码实现
- OpenCV,三大边缘检测Canny,Sobel,Laplacian,及MFC实现
- Sobel(索贝尔)边缘检测算子分析及opencv程序测试
- OpenCV-边缘检测算子Marr-Hildreth实现
- Sobel边缘检测的OpenCV实现
- 【opencv学习之二十六】边缘检测算子:Canny,Sobel,Laplacian
- opencv实现sobel边缘检测
- opencv实现Sobel 算子
- Opencv实现Canny算子边缘检测
- 【OpenCV图像处理入门学习教程四】基于LoG算子的图像边缘检测
- 【OpenCV入门教程之十二】OpenCV边缘检测:Canny算子,Sobel算子,Laplace算子,Scharr滤波器合辑
- 使用OpenCV对图像作边缘检测(Canny、Sobel、Laplace)
- Opencv图像识别从零到精通(19)----Robert,prewitt,Sobel边缘检测
- opencv实践程序4——canny实现摄像头的边缘检测,高斯背景建模
- 几种边缘检测算子的比较Roberts,Sobel,Prewitt,LOG,Canny