九、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;
}
相关推荐
格林威15 小时前
紫外UV相机在机器视觉检测方向的应用
人工智能·数码相机·opencv·计算机视觉·视觉检测·uv
2401_8414956417 小时前
【计算机视觉】图像去雾技术
人工智能·python·opencv·算法·计算机视觉·技术·图像去雾
星期天要睡觉21 小时前
计算机视觉(opencv)——基于 dlib 轮廓绘制
人工智能·opencv·计算机视觉
一百天成为python专家1 天前
机器学习之逻辑回归(梯度下降,Z标准化,0-1归一化)
人工智能·opencv·机器学习·计算机视觉·数据挖掘·数据分析·逻辑回归
Blossom.1181 天前
AI“点亮”萤火虫:边缘机器学习让微光成像走进4K时代
人工智能·pytorch·python·深度学习·数码相机·opencv·机器学习
山烛1 天前
计算机视觉:OpenCV+Dlib 人脸检测
图像处理·人工智能·opencv·计算机视觉·人脸识别·dlib
AI technophile1 天前
OpenCV计算机视觉实战(25)——立体视觉详解
人工智能·opencv·计算机视觉
曾经的三心草1 天前
OpenCV2-图像基本操作-阈值与平滑处理-形态学-梯度运算
python·opencv
星期天要睡觉2 天前
计算机视觉(opencv)——基于 dlib 实现图像人脸检测
人工智能·opencv·计算机视觉