Opencv C++ 图像的矩

          在图像处理中,矩是用来描述图像形状的一种数学工具,它可以用于计算图像的质心、面积、中心矩等特征。

          对于二值图像,一个像素可能是0或1,将图像看作一个灰度函数f(x,y),可定义第阶矩Mu(p,q):

Mu(p,q)=\sum_{x}\sum_{y}^{}x^{p}y^{q}f(x,y)

  • x,y分别是图像中的像素坐标;
  • p,q是非负整数;

常见的矩:

  • m_{00}:图像面积;
  • m_{10}:所有像素的 x 坐标之和;
  • m_{01}:所有像素的 y 坐标之和;
  • m_{20}:所有像素的 x 坐标的平方之和;
  • m_{02}:所有像素的 y 坐标的平方之和;
  • m_{11}:所有像素的 x 坐标和 y 坐标的积之和;

从这些矩可以进一步计算出图像的中心位置和方向:

  • 图像的质心位置(x_{c},y_{c}):     

x_{c}=m_{10}/m_{00}

y_{c}=m_{01}/m_{00}

  •       图像的主轴方向与x轴之间的夹角θ:

tan2\Theta =(m_{02}-m_{20})/m_{11}

计算图像矩API:

Moments moments	(InputArray array,
                 bool binaryImage = false 
)	

参数解析:

//InputArray array:输入图像(单通道8-bit类型的图像)或点集(1xN或Nx1维,类型为Point或Pointf);
//bool binaryImage = false :输入是否为二值图像(该参数仅用于图像),如果为ture,所有非0像素都被视为1;

代码演示:

#include<iostream>
#include<opencv2/opencv.hpp>

using namespace cv;
using namespace std;

int main(int argc,char**argv)
{
	Mat src = imread("C:/picture/source.jpg");
	Mat grayImg;
	cvtColor(src, grayImg, CV_BGR2GRAY);

	Moments grayImgM = moments(grayImg, false);

	Point center;
	center.x = grayImgM.m10 / grayImgM.m00;
	center.y = grayImgM.m01 / grayImgM.m00;

	circle(src, center, 5, Scalar(0, 0, 255), -1);
	imshow("src", src);

	Canny(grayImg, grayImg,35,70,3);
	imshow("Laplacian", grayImg);
	vector<vector<Point>>contours;
	findContours(grayImg,contours,RETR_EXTERNAL,CHAIN_APPROX_NONE);

	vector<Point>centers(contours.size());
	vector<Moments>contoursM(contours.size());
	cout << contours.size() << endl;
	for (size_t i = 0; i < contours.size(); i++)
	{
		contoursM[i] = moments(contours[i], false);
		centers[i].x = contoursM[i].m10 / contoursM[i].m00;
		centers[i].y = contoursM[i].m01 / contoursM[i].m00;
		//cout << i<<" : " << centers[i] << endl;
		circle(src, centers[i], 1, Scalar(0,0,0), -1);
	}
	imshow("grayImg", src);

	waitKey();
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
要计算轮廓的几何,可以使用OpenCV中的moments函数。该函数可以计算轮廓的各种,包括中心、标准化和 Hu 等。 以下是一个简单的示例代码: ```c++ #include <opencv2/opencv.hpp> using namespace cv; int main() { Mat image = imread("contour.jpg"); Mat gray; cvtColor(image, gray, COLOR_BGR2GRAY); // 阈值分割 Mat binary; threshold(gray, binary, 127, 255, THRESH_BINARY); // 查找轮廓 std::vector<std::vector<Point>> contours; findContours(binary, contours, RETR_LIST, CHAIN_APPROX_SIMPLE); // 计算轮廓的几何 for (auto contour : contours) { Moments moments = moments(contour); double area = moments.m00; double cx = moments.m10 / area; double cy = moments.m01 / area; double mu11 = moments.mu11; double mu20 = moments.mu20; double mu02 = moments.mu02; double nu20 = moments.nu20; double nu02 = moments.nu02; double hu1 = nu20 + nu02; double hu2 = pow(nu20 - nu02, 2) + 4 * pow(nu11, 2); double hu3 = pow(nu30 - 3 * nu12, 2) + pow(3 * nu21 - nu03, 2); double hu4 = pow(nu30 + nu12, 2) + pow(nu21 + nu03, 2); double hu5 = (nu30 - 3 * nu12) * (nu30 + nu12) * (pow(nu30 + nu12, 2) - 3 * pow(nu21 + nu03, 2)) + (3 * nu21 - nu03) * (nu21 + nu03) * (3 * pow(nu30 + nu12, 2) - pow(nu21 + nu03, 2)); double hu6 = (nu20 - nu02) * (pow(nu30 + nu12, 2) - pow(nu21 + nu03, 2)) + 4 * nu11 * (nu30 + nu12) * (nu21 + nu03); double hu7 = (3 * nu21 - nu03) * (nu30 + nu12) * (pow(nu30 + nu12, 2) - 3 * pow(nu21 + nu03, 2)) - (nu30 - 3 * nu12) * (nu21 + nu03) * (3 * pow(nu30 + nu12, 2) - pow(nu21 + nu03, 2)); std::cout << "Area: " << area << std::endl; std::cout << "Center: (" << cx << ", " << cy << ")" << std::endl; std::cout << "mu11: " << mu11 << std::endl; std::cout << "mu20: " << mu20 << std::endl; std::cout << "mu02: " << mu02 << std::endl; std::cout << "nu20: " << nu20 << std::endl; std::cout << "nu02: " << nu02 << std::endl; std::cout << "Hu1: " << hu1 << std::endl; std::cout << "Hu2: " << hu2 << std::endl; std::cout << "Hu3: " << hu3 << std::endl; std::cout << "Hu4: " << hu4 << std::endl; std::cout << "Hu5: " << hu5 << std::endl; std::cout << "Hu6: " << hu6 << std::endl; std::cout << "Hu7: " << hu7 << std::endl; } return 0; } ``` 在这个示例中,我们先对图像进行阈值分割,然后使用findContours函数查找图像中的轮廓。对于每个轮廓,我们使用moments函数计算几何。在这个示例中,我们计算了轮廓的面积、中心、mu11、mu20、mu02、nu20、nu02和7个Hu。最后,我们在控制台上输出这些值。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我就是不按套路出牌

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值