九、OpenCV中视频的录制

文章目录

一、OpenCV 视频录制概述

在 OpenCV 中,视频录制主要靠 cv::VideoWriter 类。它可以把连续的 cv::Mat 帧写入一个视频文件(如 .avi、.mp4 等)。

VideoWriter类

构造函数:

cpp 复制代码
VideoWriter::VideoWriter(const String &filename,
                         int fourcc,
                         double fps,
                         Size frameSize,
                         bool isColor = true);

参数说明:

  • filename:保存的视频文件名,例如 "output.avi"
  • fourcc:视频编码方式,如 CV_FOURCC('M','J','P','G') 或 VideoWriter::fourcc('M','J','P','G')
  • fps:帧率(帧每秒)
  • frameSize:视频每帧的尺寸,Size(width, height)
  • isColor:是否为彩色视频,true 彩色,false 灰度

常用编码(fourcc):

cpp 复制代码
int fourcc = VideoWriter::fourcc('M','J','P','G'); // AVI 常用
int fourcc_mp4 = VideoWriter::fourcc('a','v','c','1'); // MP4 H264

二、示例代码

示例 1:从摄像头录制视频

cpp 复制代码
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;

int main() {
    VideoCapture cap(0); // 打开摄像头
    if (!cap.isOpened()) {
        cout << "Cannot open camera" << endl;
        return -1;
    }

    // 视频参数
    int frame_width = static_cast<int>(cap.get(CAP_PROP_FRAME_WIDTH));
    int frame_height = static_cast<int>(cap.get(CAP_PROP_FRAME_HEIGHT));
    VideoWriter writer("output.avi",
                       VideoWriter::fourcc('M','J','P','G'),
                       30, Size(frame_width, frame_height));

    Mat frame;
    while (true) {
        cap >> frame;
        if (frame.empty()) break;

        writer.write(frame); // 写入视频

        imshow("Camera", frame);
        if (waitKey(1) == 27) break; // 按 ESC 停止
    }

    cap.release();
    writer.release();
    destroyAllWindows();
    return 0;
}

示例 2:将视频文件处理后保存

cpp 复制代码
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;

int main() {
    VideoCapture cap("opencv_demo.mp4");
    if (!cap.isOpened()) return -1;

    int width = cap.get(CAP_PROP_FRAME_WIDTH);
    int height = cap.get(CAP_PROP_FRAME_HEIGHT);
    VideoWriter writer("output_processed.avi",
        VideoWriter::fourcc('M', 'J', 'P', 'G'),
        30, Size(width, height));

    Mat frame;
    while (true) {
        cap >> frame;
        if (frame.empty()) break;

        // 处理帧,例如灰度
        cvtColor(frame, frame, COLOR_BGR2GRAY);
        cvtColor(frame, frame, COLOR_GRAY2BGR); // 保持三通道以写入

        writer.write(frame);
    }
    cap.release();
    writer.release();
    return 0;
}

示例3:自动读取指定文件夹下所有图片生成视频

cpp 复制代码
#include <opencv2/opencv.hpp>
#include <iostream>
#include <filesystem>
#include <vector>
#include <algorithm>
using namespace cv;
using namespace std;
namespace fs = std::filesystem;

int main() {
    string folderPath = "images"; // 图片文件夹路径
    string outputVideo = "output_video.avi";
    double fps = 30.0; // 帧率

    vector<string> imgFiles;

    // 遍历文件夹,收集图片路径
    for (const auto& entry : fs::directory_iterator(folderPath)) {
        if (entry.is_regular_file()) {
            string ext = entry.path().extension().string();
            // 支持 jpg/png/bmp
            if (ext == ".jpg" || ext == ".png" || ext == ".bmp")
                imgFiles.push_back(entry.path().string());
        }
    }

    if (imgFiles.empty()) {
        cout << "No images found in folder!" << endl;
        return -1;
    }

    // 按文件名排序
    sort(imgFiles.begin(), imgFiles.end());

    // 读取第一张图片获取尺寸
    Mat firstImg = imread(imgFiles[0]);
    if (firstImg.empty()) {
        cout << "Failed to read " << imgFiles[0] << endl;
        return -1;
    }
    Size frameSize(firstImg.cols, firstImg.rows);

    // 创建 VideoWriter
    VideoWriter writer(outputVideo,
                       VideoWriter::fourcc('M','J','P','G'),
                       fps,
                       frameSize,
                       true);

    if (!writer.isOpened()) {
        cout << "Cannot open VideoWriter!" << endl;
        return -1;
    }

    // 写入每张图片
    for (const auto& file : imgFiles) {
        Mat img = imread(file);
        if (img.empty()) {
            cout << "Failed to read " << file << endl;
            continue;
        }

        // 尺寸不一致就 resize
        if (img.size() != frameSize) {
            resize(img, img, frameSize);
        }

        writer.write(img);
    }

    writer.release();
    cout << "Video saved as " << outputVideo << endl;
    return 0;
}
相关推荐
一招定胜负1 天前
OpenCV轮廓检测完全指南:从原理到实战
人工智能·opencv·计算机视觉
dazzle1 天前
计算机视觉处理(OpenCV基础教学(二十二):霍夫变换技术详解)
人工智能·opencv·计算机视觉
格林威1 天前
印刷电路板阻焊层缺失识别:防止短路风险的 7 个核心策略,附 OpenCV+Halcon 实战代码!
人工智能·数码相机·opencv·机器学习·计算机视觉·视觉检测·工业相机
saoys1 天前
Opencv 学习笔记:形态学开 / 闭运算(解决噪点与孔洞问题)
笔记·opencv·学习
Pyeako1 天前
Opencv计算机视觉--边界填充&图像形态学
人工智能·python·opencv·计算机视觉·pycharm·图像形态学·边缘填充
曾帅1681 天前
uniapp安卓启动图
android·opencv·uni-app
智驱力人工智能1 天前
矿场轨道异物AI监测系统 构建矿山运输安全的智能感知防线 轨道异物检测 基于YOLO的轨道异物识别算法 地铁隧道轨道异物实时预警技术
人工智能·opencv·算法·安全·yolo·边缘计算
格林威1 天前
基于灰度投影的快速图像配准:适用于产线在线对位的 5 个核心方法,附 OpenCV+Halcon 实战代码!
人工智能·数码相机·opencv·机器学习·计算机视觉·视觉检测·工业相机
sali-tec1 天前
C# 基于OpenCv的视觉工作流-章9-均值滤波
人工智能·opencv·算法·计算机视觉·均值算法
子夜江寒1 天前
基于 OpenCV 的图像边缘检测与轮廓分析
图像处理·opencv·计算机视觉