IT评测·应用市场-qidao123.com

标题: OpenCV中直线、曲线和圆的拟合方法 [打印本页]

作者: 愛在花開的季節    时间: 2025-3-22 03:30
标题: OpenCV中直线、曲线和圆的拟合方法
一、直线拟合

二、曲线拟合

三、圆拟合

四、代码

1.‌最小二乘法直线拟合

  1. #include <opencv2/opencv.hpp>
  2. #include <vector>
  3. using namespace cv;
  4. using namespace std;
  5. int main() {
  6.     // 模拟输入点集(可替换为实际图像边缘点)
  7.     vector<Point2f> points = {{50, 100}, {100, 150}, {150, 200}, {200, 250}};
  8.     // 直线拟合
  9.     Vec4f lineParams;
  10.     fitLine(points, lineParams, DIST_L2, 0, 0.01, 0.01);
  11.     // 解析直线参数:vx, vy 是方向向量,x0, y0 是直线上一点
  12.     float vx = lineParams;
  13.     float vy = lineParams;
  14.     float x0 = lineParams;
  15.     float y0 = lineParams;
  16.     // 计算直线端点(延伸100像素)
  17.     Point pt1(x0 - 100*vx, y0 - 100*vy);
  18.     Point pt2(x0 + 100*vx, y0 + 100*vy);
  19.     // 可视化
  20.     Mat img = Mat::zeros(300, 300, CV_8UC3);
  21.     for (const auto& p : points) {
  22.         circle(img, p, 3, Scalar(0, 255, 0), FILLED); // 绘制输入点
  23.     }
  24.     line(img, pt1, pt2, Scalar(0, 0, 255), 2); // 绘制拟合直线
  25.     imshow("Line Fitting", img);
  26.     waitKey(0);
  27.     return 0;
  28. }
复制代码
2. 曲线拟合

  1. #include <opencv2/opencv.hpp>
  2. #include <vector>
  3. #include <cmath>
  4. using namespace cv;
  5. using namespace std;
  6. int main() {
  7.     // 输入点集(二次曲线y = ax^2 + bx + c 比如:y = 0.5x² + 3x + 10 的采样)
  8.     vector<Point2f> points;
  9.     for (int x = 0; x <= 20; x += 2) {
  10.         float y = 0.5*pow(x,2) + 3*x + 10;
  11.         points.emplace_back(x*10, y*10); // 放大坐标便于显示
  12.     }
  13.     // 构建矩阵方程 Ax = b
  14.     Mat A(points.size(), 3, CV_32F);
  15.     Mat b(points.size(), 1, CV_32F);
  16.     for (size_t i = 0; i < points.size(); i++) {
  17.         float x = points[i].x;
  18.         A.at<float>(i, 0) = x*x;
  19.         A.at<float>(i, 1) = x;
  20.         A.at<float>(i, 2) = 1;
  21.         b.at<float>(i) = points[i].y;
  22.     }
  23.     // 解方程 (A^T * A) * coeff = A^T * b
  24.     Mat coeff;
  25.     solve(A, b, coeff, DECOMP_NORMAL | DECOMP_SVD);
  26.     // 生成拟合曲线点
  27.     vector<Point> curvePoints;
  28.     for (int x = 0; x <= 200; x += 2) {
  29.         float y = coeff.at<float>(0)*x*x + coeff.at<float>(1)*x + coeff.at<float>(2);
  30.         curvePoints.emplace_back(x, cvRound(y));
  31.     }
  32.     // 可视化
  33.     Mat img = Mat::zeros(800, 800, CV_8UC3);
  34.     for (const auto& p : points) {
  35.         circle(img, p, 5, Scalar(0, 255, 0), FILLED);
  36.     }
  37.     polylines(img, curvePoints, false, Scalar(0, 0, 255), 2);
  38.     imshow("Curve Fitting", img);
  39.     waitKey(0);
  40.     return 0;
  41. }
复制代码
关键点‌:

在C++中,我们可以利用Eigen库或手动实现最小二乘法:
  1. #include <iostream>
  2. #include <vector>
  3. #include <Eigen/Dense>
  4. struct Point {
  5.     double x, y;
  6. };
  7. int main() {
  8.     std::vector<Point> points = {{1, 2}, {2, 5}, {3, 10}, {4, 17}};
  9.     int n = points.size();
  10.     Eigen::MatrixXd A(n, 3); // 二次方程有3个参数a, b, c
  11.     Eigen::VectorXd b(n); // y值
  12.     Eigen::Vector3d coeffs; // 系数向量
  13.     for (int i = 0; i < n; ++i) {
  14.         A(i, 0) = points[i].x * points[i].x; // x^2
  15.         A(i, 1) = points[i].x;               // x
  16.         A(i, 2) = 1;                        // 1 (constant term)
  17.         b(i) = points[i].y;                // y值
  18.     }
  19.     coeffs = A.bdcSvd(Eigen::ComputeThinU | Eigen::ComputeThinV).solve(b);
  20.     std::cout << "Coefficients: " << coeffs.transpose() << std::endl; // a, b, c
  21.     return 0;
  22. }
复制代码
3. 圆拟合

  1. #include <opencv2/opencv.hpp>
  2. #include <vector>
  3. using namespace cv;
  4. using namespace std;
  5. int main() {
  6.     // 生成模拟点集(实际应用中可通过findContours获取轮廓)
  7.     vector<Point2f> points;
  8.     for (int angle = 0; angle < 360; angle += 30) {
  9.         float radian = angle * CV_PI / 180;
  10.         points.emplace_back(100 + 50*cos(radian), 100 + 50*sin(radian));
  11.     }
  12.     // 最小包围圆拟合
  13.     Point2f center;
  14.     float radius;
  15.     minEnclosingCircle(points, center, radius);
  16.     // 可视化
  17.     Mat img = Mat::zeros(200, 200, CV_8UC3);
  18.     for (const auto& p : points) {
  19.         circle(img, p, 3, Scalar(0, 255, 0), FILLED);
  20.     }
  21.     circle(img, center, cvRound(radius), Scalar(0, 0, 255), 1);
  22.     imshow("Circle Fitting", img);
  23.     waitKey(0);
  24.     return 0;
  25. }
复制代码



免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。




欢迎光临 IT评测·应用市场-qidao123.com (https://dis.qidao123.com/) Powered by Discuz! X3.4