java - 在当前帧之前存储视频帧?

标签 java image opencv image-processing video-processing

我目前有一段显示视频的代码,我正在尝试扩展它以显示当前帧和上一帧之间的差异,从而检测运动。我知道我将不得不使用 absdiff() 函数来显示两个图像之间的对比度,但我不确定如何将前一帧存储为 Mat。有人可以看看我下面的代码并告诉我应该在哪里添加这个“前一帧”代码,以及我应该写什么。我假设它不会花很长时间,我只是找不到任何在线教程...

import org.opencv.core.Core;
import org.opencv.core.Mat;
import org.opencv.core.Size;
import org.opencv.highgui.VideoCapture;
import org.opencv.highgui.Highgui;
import org.opencv.imgproc.Imgproc;

// ********************************************************

public class CaptureVideo {

public static void main(String[] args) throws InterruptedException {

    // load the Core OpenCV library by name

    System.loadLibrary(Core.NATIVE_LIBRARY_NAME);

    // create video capture device object

    VideoCapture cap = new VideoCapture();

    // try to use the hardware device if present

    int CAM_TO_USE = 0;

    // create a new image object

    Mat matFrame = new Mat();

    // try to open first capture device (0)

    try {
        cap.open(CAM_TO_USE);
    } catch (Exception e1) {
        System.out.println("No webcam attached");

        // otherwise try opening a video file 

        try{
            cap.open("files/video.mp4");
        } catch (Exception e2) {
            System.out.println("No video file found");
        }
    }

    // if the a video capture source is now open

    if (cap.isOpened())
    {

        // create a new window object

        Imshow ims = new Imshow("From video source ... ");

        boolean keepProcessing = true;

        while (keepProcessing)
        {   
            // grab the next frame from video source

            cap.grab();

            // decode and return the grabbed video frame

            cap.retrieve(matFrame);

            // if the frame is valid (not end of video for example)

            if (!(matFrame.empty()))
            {
                // *** to any processing here*** 

                // display image with a delay of 40ms (i.e. 1000 ms / 25 = 25 fps)

                ims.showImage(matFrame);
                Thread.sleep(40);
            } else { 
                keepProcessing = false;
            }
        }

    } else {
        System.out.println("error cannot open any capture source - exiting");

    }

    // close down the camera correctly

    cap.release();

  }
}

我已经尝试在任一 else 语句之后添加以下代码,但生成的只是黑屏(我认为这是因为“previousFrame”只是“matFrame”的副本):

            previousFrame = matFrame;

            Core.absdiff(matFrame, previousFrame, diffFrame);

            ims.showImage(diffFrame);

谁能指出我哪里出错了?

最佳答案

执行previousFrame = matFrame; 使previousFrame 指向 matFrame。您没有复制矩阵 - previousFramematFrame 都指的是同一个对象。因此,您实际上是在减去矩阵本身,这就是您得到黑框的原因。

您需要进行正确 复制,因此您需要使用Mat.clone()克隆 矩阵。 .但是,如果您想找出两帧之间的差异,请记住我们至少需要读入两帧才能做到这一点。当我们只读取视频的第一帧时,我们无法真正找到差异帧,因此我们应该检查一下这是否是第一帧。如果它是第一帧,则相应地显示正常帧。在那之后,我们可以通过确保在读取下一帧之前复制前一帧来显示差异帧。

因此,为您的循环执行此操作:

   // ****** Change - add a flag to check to see if the first frame has been read
   boolean firstFrame = true;

   // ****** Change - declare previous frame here
   Mat previousFrame;
   while (keepProcessing)
    {   
        // ****** Change - Save previous frame before getting next one
        // Only do this if the first frame has passed
        if (!firstFrame)
            previousFrame = matFrame.clone();

        // grab the next frame from video source
        cap.grab();

        // decode and return the grabbed video frame
        cap.retrieve(matFrame);

        // if the frame is valid (not end of video for example)
        if (!(matFrame.empty()))
        {
            // **** Change - If we are on the first frame, only show that and
            // set the flag to false
            if (firstFrame) {
                ims.showImage(matFrame);
                firstFrame = false;
            }
            // ***** Change - now show absolute difference after first frame
            else {                
                Core.absdiff(matFrame, previousFrame, diffFrame);
                ims.showImage(diffFrame);                 
            }

            // display image with a delay of 40ms (i.e. 1000 ms / 25 = 25 fps)                
            Thread.sleep(40);
        } else { 
            keepProcessing = false;
        }
    }

关于java - 在当前帧之前存储视频帧?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29099522/

相关文章:

java - Spring-Integration-Kafka outbound-channel-adapter 发送消息

html - 悬停效果不会在图像上触发

python - 如何创建带有图像和文本的复选按钮?

java - 如何使用java在旋转时保留图像的原始颜色

java - 如何使用 EasyMock 测试 Java 中 protected final方法?

java - Mybatis-错误: Could not set parameters for mapping

Python HSI 到 RGB 的转换 - 不是我所期望的

wordpress - 在自定义帖子中上传多张精选图片 (Wordpress)

python - 距离变换 - 该功能无法正常工作

python - Pytesseract 没有给出预期的输出