【opencv450-samples】calcOpticalFlowFarneback计算光流-移动像素的移动方向

46 篇文章 4 订阅

#include "opencv2/video/tracking.hpp"
#include "opencv2/imgproc.hpp"
#include "opencv2/videoio.hpp"
#include "opencv2/highgui.hpp"

#include <iostream>

using namespace cv;
using namespace std;

static void help(char** argv)
{
    cout <<
        "\nThis program demonstrates dense optical flow algorithm by Gunnar Farneback\n"
        "Mainly the function: calcOpticalFlowFarneback()\n"
        "Call:\n"
        << argv[0]
        << "This reads from video camera 0\n" << endl;
}
//没step个像素 绘制一次。  cflowmap:上一彩色图像    flow:当前图像相对上一图像的光流
static void drawOptFlowMap(const Mat& flow, Mat& cflowmap, int step,
    double, const Scalar& color)
{
    for (int y = 0; y < cflowmap.rows; y += step)
        for (int x = 0; x < cflowmap.cols; x += step)
        {
            const Point2f& fxy = flow.at<Point2f>(y, x);
            line(cflowmap, Point(x, y), Point(cvRound(x + fxy.x), cvRound(y + fxy.y)),
                color);//光流直线
            circle(cflowmap, Point(x, y), 2, color, -1);//上一图像上的离散点
        }
}

int main(int argc, char** argv)
{
    cv::CommandLineParser parser(argc, argv, "{help h||}{video|balltest.mp4|}");
    if (parser.has("help"))
    {
        help(argv);
        return 0;
    }
    //VideoCapture cap(0);
    VideoCapture cap(samples::findFile(parser.get<String>("video")));
    help(argv);
    if (!cap.isOpened())
        return -1;

    Mat flow, cflow, frame;
    UMat gray, prevgray, uflow;
    namedWindow("flow", 1);

    for (;;)
    {
        cap >> frame;
        if (frame.empty())
        {
            break;
        }
        cvtColor(frame, gray, COLOR_BGR2GRAY);

        if (!prevgray.empty())//上一灰度图非空
        {   //利用用Gunnar Farneback的算法计算全局性的稠密光流算法(即图像上所有像素点的光流都计算出来)
            /// <summary>
            /// /
            /// </summary>
            /// prevgray : 输入前一帧图像
            /// gray : 输入后一帧图像
            /// uflow: flow 输出的光流
            /// 0.5:pyr_scale 金字塔上下两层之间的尺度关系
            /// 3:levels :金字塔层数
            /// 15: winsize 均值窗口大小,越大越能denoise并且能够检测快速移动目标,但会引起模糊运动区域
            /// 3:iterations 迭代次数
            /// 5: poly_n 像素领域大小,一般为5,7等
            /// 1.2 : poly_sigma 高斯标注差,一般为1-1.5
            /// 0: flags 计算方法。主要包括OPTFLOW_USE_INITIAL_FLOW和OPTFLOW_FARNEBACK_GAUSSIAN
            /// <returns></returns>
            /*CV_EXPORTS_W void calcOpticalFlowFarneback(InputArray prev, InputArray next, InputOutputArray flow,
                double pyr_scale, int levels, int winsize,
                int iterations, int poly_n, double poly_sigma,
                int flags);*/
            calcOpticalFlowFarneback(prevgray, gray, uflow, 0.5, 3, 15, 3, 5, 1.2, 0);//计算光流
            cvtColor(prevgray, cflow, COLOR_GRAY2BGR);//上一灰度图的彩色图
            uflow.copyTo(flow);//光流
            drawOptFlowMap(flow, cflow, 12, 1.5, Scalar(0, 255, 0));//在上一彩色图上绘制绿色光流图
            imshow("flow", cflow);
        }
        if (waitKey(30) >= 0)
            break;
        std::swap(prevgray, gray);//更新上一灰度图
    }
    return 0;
}

运行效果

参考:

OpenCV中CalcOpticalFlowFarneback()函数分析_Dream_yz的博客-CSDN博客https://blog.csdn.net/yzhang6_10/article/details/51225545

OpenCV 光流 | D栈 - Delft Stackicon-default.png?t=M4ADhttps://www.delftstack.com/zh/howto/python/opencv-optical-flow/

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Lucas-Kanade光流算法是一种基于局部区域的光流算法,它假设图像中任意两帧之间的像素值变化是平滑的,然后使用局部区域内的像素值变化来估计每个像素的运动向量。在本文中,我们将介绍如何在OpenCV中自实现Lucas-Kanade光流算法。 步骤一:读取图像 首先,我们需要读取两张待计算光流的图像。在本例中,我们将使用名为“frame1”和“frame2”的两张图像。 Mat frame1 = imread("frame1.jpg"); Mat frame2 = imread("frame2.jpg"); 步骤二:提取关键点 接下来,我们需要从两个图像中提取关键点。我们可以使用OpenCV中的FAST或SIFT等算法来提取关键点。在本例中,我们将使用FAST算法。 vector<KeyPoint> keypoints1, keypoints2; int threshold = 20; // 设置FAST算法的阈值 bool nonmaxSuppression = true; // 设置是否进行非极大值抑制 FAST(frame1, keypoints1, threshold, nonmaxSuppression); FAST(frame2, keypoints2, threshold, nonmaxSuppression); 步骤三:计算光流 现在我们已经提取了关键点,接下来我们需要计算这些关键点的光流向量。我们可以使用OpenCV中的calcOpticalFlowPyrLK函数来计算光流向量。该函数使用金字塔表示法和Lucas-Kanade算法计算光流向量。 vector<uchar> status; vector<float> err; Size winSize = Size(21, 21); // 设置光流计算窗口的大小 int maxLevel = 3; // 设置金字塔的最大层数 TermCriteria criteria = TermCriteria(TermCriteria::COUNT+TermCriteria::EPS, 30, 0.01); // 设置终止条件 calcOpticalFlowPyrLK(frame1, frame2, keypoints1, keypoints2, status, err, winSize, maxLevel, criteria); 步骤四:绘制光流 最后,我们可以将光流向量绘制在第一张图像上,以便我们可以观察到光流的效果。 for (int i = 0; i < keypoints1.size(); i++) { if (status[i]) { Point2f p1 = keypoints1[i].pt; Point2f p2 = keypoints2[i].pt; line(frame1, p1, p2, Scalar(0, 0, 255), 2); } } imshow("Optical Flow", frame1); 完整代码:
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值