OpenCV读取视频、存储视频、跳帧、分辨率、总帧数

import cv2


def read_write():
    '''

    文件扩展名.avi:
        cv2.VideoWriter_fourcc('I','4','2','0')---未压缩的YUV颜色编码,4:2:0色度子采样。兼容性好,但文件较大。
        cv2.VideoWriter_fourcc('P','I','M','1')---MPEG-1编码类型。随机访问,灵活的帧率、可变的图像尺寸、定义了I-帧、P-帧和B-帧 、运动补偿可跨越多个帧 、半像素精度的运动向量 、量化矩阵、GOF结构 、slice结构 、技术细节、输入视频格式。
        cv2.VideoWriter_fourcc('X','V','I','D')---MPEG-4编码类型,视频大小为平均值,MPEG4所需要的空间是MPEG1或M-JPEG的1/10,它对运动物体可以保证有良好的清晰度,间/时间/画质具有可调性。
    文件扩展名.ogv:
        cv2.VideoWriter_fourcc('T','H','E','O')---OGGVorbis,音频压缩格式,有损压缩,类似于MP3等的音乐格式,兼容性差。
    文件扩展名.flv:
        cv2.VideoWriter_fourcc('F','L','V','1')---FLV是FLASH VIDEO的简称,FLV流媒体格式是一种新的视频格式。由于它形成的文件极小、加载速度极快,使得网络观看视频文件成为可能,它的出现有效地解决了视频文件导入Flash后,使导出的SWF文件体积庞大,不能在网络上很好的使用等缺点。
    '''
    cap = cv2.VideoCapture(r'1.mp4')
    target = r'2.mp4'
    i = 0
    # 帧率
    fps = int(round(cap.get(cv2.CAP_PROP_FPS)))
    # 分辨率-宽度
    width = int(cap.get(cv2.CAP_PROP_FRAME_WIDTH))
    # 分辨率-高度
    height = int(cap.get(cv2.CAP_PROP_FRAME_HEIGHT))
    # 总帧数
    frame_counter = int(cap.get(cv2.CAP_PROP_FRAME_COUNT))
    target_video = cv2.VideoWriter(target, cv2.VideoWriter_fourcc('X', 'V', 'I', 'D'), fps, (width, height))
    while True:
        # 可以设置跳过前面多少帧
        # cap.set(cv2.CAP_PROP_POS_FRAMES, 100)
        _, frame = cap.read()
        if not _:
            break
        target_video.write(frame)
        i += 1
    cap.release()
    target_video.release()
    cv2.destroyAllWindows()

  • 7
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
要将OpenCV处理视频流编码为H.264格式,可以使用FFmpeg库进行编码。以下是一些基本步骤: 1. 安装FFmpeg库,并将其包含到项目中。 2. 从OpenCV读取视频流,并将每一存储为Mat对象。 3. 将Mat对象转换为AVFrame对象,使用av_image_fill_arrays()函填充AVFrame的属性。 4. 创建AVCodecContext对象,并设置编码参。这些参包括视频分辨率率、比特率等。 5. 使用avcodec_open2()函打开编码器。 6. 使用avcodec_encode_video2()函将每一AVFrame编码为H.264格式。 7. 将编码后的据写入文件或网络流中。 以下是一个简单的例子: ```cpp #include <opencv2/opencv.hpp> #include <libavcodec/avcodec.h> #include <libavformat/avformat.h> #include <libavutil/opt.h> #include <libavutil/imgutils.h> int main(int argc, char* argv[]) { cv::VideoCapture cap("myvideo.mp4"); // 创建AVCodecContext对象 AVCodecContext* codec_ctx = avcodec_alloc_context3(nullptr); codec_ctx->codec_id = AV_CODEC_ID_H264; codec_ctx->codec_type = AVMEDIA_TYPE_VIDEO; codec_ctx->pix_fmt = AV_PIX_FMT_YUV420P; codec_ctx->width = cap.get(CV_CAP_PROP_FRAME_WIDTH); codec_ctx->height = cap.get(CV_CAP_PROP_FRAME_HEIGHT); codec_ctx->time_base = { 1, cap.get(CV_CAP_PROP_FPS) }; codec_ctx->bit_rate = 1000000; // 打开编码器 AVCodec* codec = avcodec_find_encoder(codec_ctx->codec_id); avcodec_open2(codec_ctx, codec, nullptr); // 创建AVFrame对象 AVFrame* frame = av_frame_alloc(); frame->format = codec_ctx->pix_fmt; frame->width = codec_ctx->width; frame->height = codec_ctx->height; av_image_alloc(frame->data, frame->linesize, codec_ctx->width, codec_ctx->height, codec_ctx->pix_fmt, 32); // 创建AVPacket对象 AVPacket pkt; av_init_packet(&pkt); while (cap.isOpened()) { cv::Mat mat_frame; cap >> mat_frame; // 将Mat对象转换为AVFrame对象 cv::Mat yuv_frame; cv::cvtColor(mat_frame, yuv_frame, CV_BGR2YUV_I420); memcpy(frame->data[0], yuv_frame.data, codec_ctx->width * codec_ctx->height); memcpy(frame->data[1], yuv_frame.data + codec_ctx->width * codec_ctx->height, codec_ctx->width * codec_ctx->height / 4); memcpy(frame->data[2], yuv_frame.data + codec_ctx->width * codec_ctx->height * 5 / 4, codec_ctx->width * codec_ctx->height / 4); // 编码 int ret = avcodec_send_frame(codec_ctx, frame); while (ret >= 0) { ret = avcodec_receive_packet(codec_ctx, &pkt); if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) break; else if (ret < 0) return -1; // 将编码后的据写入文件或网络流中 fwrite(pkt.data, 1, pkt.size, outfile); } } // 清理资源 av_packet_unref(&pkt); av_frame_free(&frame); avcodec_free_context(&codec_ctx); return 0; } ``` 注意:此例子仅用于演示基本的编码过程,实际应用中还需要处理异常情况、添加错误处理等。另外,编码后的H.264据需要使用相应的解码器进行解码,例如FFmpeg库中的avcodec_decode_video2()函

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小殊小殊

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值