Python实用记录(一):如何将不同类型视频按关键帧提取并保存图片,实现图片裁剪功能

下面我定义读取的视频类型包括两个-------------avi和mp4.
但是我电脑里只有mp4类型的视频,所以就只演示一个了,两种格式运行都是一样的道理。

1定义read_pic()来读取路径下全部图片

def read_pic(path):
    if os.path.exists(path):
        print(1)
    else:
        print(2)
    dirnames = sorted(os.listdir(path))
    # print(dirname)
    n = len(dirnames)
    print(n)
    Srcimg= [] # 创建一个集合用于保存所有图片
    for dirname in dirnames:
        # print("正在读取第%d张图片" % i)
        # fromfile()函数读回数据时需要用户指定元素类型,并对数组的形状进行适当的修改,indecode下中文路径也可以运行
        img = cv2.imdecode(np.fromfile(path + dirname, dtype=np.uint8), -1)
        #####保存图片#########
        # cv2.imwrite(output_path + "/" + dirname, img)
        #img_path = os.path.join(input_path, dirname)
        # f.write(img_path+'\n')
        Srcimg.append(img)
        print(img_path)
    Srcimg = np.array(Srcimg)

    return Srcimg

也可以用glob来读取该文件夹下全部图片,非常简洁好记,代码如下:

import glob
import os
import cv2

IMAGE_PATH='D:/pycharm/4kinds_detectface_module/test'
filenames = glob.glob(os.path.join(IMAGE_PATH,'*.jpg'))
nums = len(filenames)
print(filenames)
print(nums)
for i in range(nums):
    img=cv2.imread(filenames[i])
    cv2.imshow('img',img)
    cv2.waitKey(0)

2定义读取mp4类型的视频,25帧读取一张图片

def splitFrames_mp4(sourceFileName):

    # 在这里把后缀接上
    video_path = os.path.join('E:/qq/MobileFile/', sourceFileName + '.mp4')
    times = 0
    # 提取视频的频率,每25帧提取一个,也就是一秒一张图
    frameFrequency = 25
    # 输出图片到当前目录vedio文件夹下
    outPutDirName = 'D:/pycharm/facenet/data/lfw_5/' + sourceFileName + '/'
    # 如果文件目录不存在则创建目录
    Datas = []
    if not os.path.exists(outPutDirName):
        os.makedirs(outPutDirName)

    camera = cv2.VideoCapture(video_path)
    while True:
        times+=1
        res, image = camera.read()
        if not res:
            # print('not res , not image')
            break
        if times%frameFrequency==0:
            # cv2.imwrite(outPutDirName + str(times)+'.jpg', image)
            print(outPutDirName + str(times)+'.jpg')
            Datas.append(image)
        # cv2.imwrite(outPutDirName + str(times) + '.jpg', image)
        # print(times)

    print('图片提取结束')
    # print(Datas)
    camera.release()
    return Datas

3定义读取avi类型的视频,25帧读取一张图片

def splitFrames(sourceFileName):

    # 在这里把后缀接上
    video_path = os.path.join(im_file, sourceFileName + '.avi')
    outPutDirName = 'D:/pycharm/facenet/data/lfw_5/' + sourceFileName + '/'

    if not os.path.exists(outPutDirName):
        #如果文件目录不存在则创建目录
        os.makedirs(outPutDirName)
    datas = []
    cap = cv2.VideoCapture(video_path) # 打开视频文件
    num = 1
    while True:
        num = num + 1
        # success 表示是否成功,data是当前帧的图像数据;.read读取一帧图像,移动到下一帧
        success, data = cap.read()
        if not success:
            break

        if num % 25 == 0:
            print(outPutDirName + str(num)+'.jpg')
            # cv2.imwrite( outPutDirName +str(num)+".jpg", data)
            datas.append(data)
    print('图片提取结束')
    cap.release()
    return datas

4完整代码:

import os
import cv2
import numpy as np


def read_pic(path):
    if os.path.exists(path):
        print(1)
    else:
        print(2)
    dirnames = sorted(os.listdir(path))
    # print(dirname)
    n = len(dirnames)
    print(n)
    # f = open('neg.txt', 'w')
    Srcimg= []
    for dirname in dirnames:
        # print("正在读取第%d张图片" % i)
        # fromfile()函数读回数据时需要用户指定元素类型,并对数组的形状进行适当的修改,indecode下中文路径也可以运行
        img = cv2.imdecode(np.fromfile(path + dirname, dtype=np.uint8), -1)
        #####保存图片#########

        # cv2.imwrite(output_path + "/" + dirname, img)
        img_path = os.path.join(input_path, dirname)
        # f.write(img_path+'\n')
        Srcimg.append(img)
        print(img_path)
    Srcimg = np.array(Srcimg)

    return Srcimg

def splitFrames_mp4(sourceFileName):

    # 在这里把后缀接上
    video_path = os.path.join('E:/qq/MobileFile/', sourceFileName + '.mp4')
    times = 0
    num = 0
    # 提取视频的频率,每25帧提取一个,也就是一秒一张图
    frameFrequency = 25
    # 输出图片到当前目录vedio文件夹下
    outPutDirName = 'D:/pycharm/arithmetic/' + sourceFileName + '/'
    # 如果文件目录不存在则创建目录
    Datas = []
    if not os.path.exists(outPutDirName):
        os.makedirs(outPutDirName)

    camera = cv2.VideoCapture(video_path)
    while True:
        times+=1
        res, image = camera.read()
        if not res:
            # print('not res , not image')
            break
        if times%frameFrequency==0:
            num+=1
            # cv2.imwrite(outPutDirName + str(times)+'.jpg', image)
            print(outPutDirName + str(times)+'.jpg')
            cv2.imwrite(outPutDirName+str(times)+'.jpg',image)
            Datas.append(image)
        # cv2.imwrite(outPutDirName + str(times) + '.jpg', image)
        # print(times)
    print('--------一共有{}-------'.format(num))
    print('图片提取结束')
    # print(Datas)
    camera.release()
    return Datas

# 从.avi 类型的视频中提取图像
def splitFrames(sourceFileName):

    # 在这里把后缀接上
    video_path = os.path.join(im_file, sourceFileName + '.avi')
    outPutDirName = 'D:/pycharm/facenet/data/lfw_5/' + sourceFileName + '/'

    if not os.path.exists(outPutDirName):
        #如果文件目录不存在则创建目录
        os.makedirs(outPutDirName)
    datas = []
    cap = cv2.VideoCapture(video_path) # 打开视频文件
    num = 1
    while True:
        num = num + 1
        # success 表示是否成功,data是当前帧的图像数据;.read读取一帧图像,移动到下一帧
        success, data = cap.read()
        if not success:
            break
        # im = Image.fromarray(data, mode='RGB') # 重建图像
        # im.save('C:/Users/Taozi/Desktop/2019.04.30/' +str(num)+".jpg") # 保存当前帧的静态图像
        # cv2.imwrite( outPutDirName +str(num)+".jpg", data)
        # data.append(data)

        if num % 25 == 0:
            print(outPutDirName + str(num)+'.jpg')
            datas.append(data)
    print('图片提取结束')
    cap.release()
    return datas

if __name__ == '__main__':
    im_file = 'E:/qq/MobileFile/'
    # input_path = 'D:/pycharm/10kinds-light-face-detector-align-recognition-master/lfw_5/test/'
    for im_name in os.listdir(im_file):
        C_file = os.path.splitext(im_name)[-1]
        if C_file == '.mp4':
            print('~~~~~~~~~~ 从.mp4 视频提取图像 ~~~~~~~~~~~~~~~')
            #
            sourceFileName = os.path.splitext(im_name)[0]
            splitFrames_mp4(sourceFileName)
        if C_file == '.avi':
            print('~~~~~~~~~~ 从.avi 视频提取图像 ~~~~~~~~~~~~~~~')
            #
            sourceFileName = os.path.splitext(im_name)[0]
            splitFrames(sourceFileName)
e)

5运行结果:

在这里插入图片描述
在这里插入图片描述

6图片裁剪

裁剪部分主要是根据下面这一行代码进行的,这里要记住(我被这里坑了一下午),
参数tr[1]:左上角或右上角的纵坐标值
参数bl[1]:左下角或右下角的纵坐标值
参数tl[0]:左上角或左下角的横坐标值
参数br[0]:右上角或右下角的横坐标值

crop = img[int(tr[1]):int(bl[1]), int(tl[0]):int(br[0]) ]

在这里插入图片描述

  • 3
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: 您可以使用OpenCV实现视频提取帧并保存为图片python程序。下面是一段示例代码:import cv2 video = cv2.VideoCapture('video.mp4') # Find the number of frames num_frames = int(video.get(cv2.CAP_PROP_FRAME_COUNT)) # Create a directory to store frames dirname = 'frames' if not os.path.exists(dirname): os.mkdir(dirname) # Extract frames from the video and save them for i in range(num_frames): success, image = video.read() cv2.imwrite(os.path.join(dirname, 'frame{:d}.jpg'.format(i)), image) ### 回答2: 下面是一个将视频逐帧转换为图片,并保存的Python程序: ```python import cv2 def extract_frames(video_path, output_folder): # 加载视频 video = cv2.VideoCapture(video_path) count = 0 # 检查视频文件是否成功打开 if not video.isOpened(): print("无法打开视频文件!") return # 读取并保存每一帧图像 while True: ret, frame = video.read() # 读取失败,可能已经到达视频末尾 if not ret: break # 保存图像 image_path = f"{output_folder}/frame_{count}.png" cv2.imwrite(image_path, frame) count += 1 # 释放内存和关闭视频文件 video.release() print(f"视频文件成功转换成了{count}帧图像!") # 测试示例 video_path = "path_to_video/video.mp4" # 视频文件的路径 output_folder = "path_to_output_folder" # 保存图像的文件夹路径 extract_frames(video_path, output_folder) ``` 请将 `path_to_video/video.mp4` 替换为你要提取帧的视频文件的实际路径,将 `path_to_output_folder` 替换为你要保存图像的文件夹的实际路径。 该程序使用OpenCV库来读取视频文件并提取每一帧图像,然后使用OpenCV的 `imwrite` 函数将每一帧保存为独立的图像文件(如PNG格式)。程序运行后,会在指定的输出文件夹中生成连续编号的图像文件,文件名格式为 `frame_0.png`, `frame_1.png` 等。 请确保你的系统中已经安装并正确配置了OpenCV库,并根据需要修改程序的相关路径和设置。 ### 回答3: import cv2 import os def video_to_frames(video_path, output_path): # 使用OpenCV打开视频文件 cap = cv2.VideoCapture(video_path) # 获取每一帧的总数 frame_count = int(cap.get(cv2.CAP_PROP_FRAME_COUNT)) # 输出每一帧的宽度和高度 frame_width = int(cap.get(cv2.CAP_PROP_FRAME_WIDTH)) frame_height = int(cap.get(cv2.CAP_PROP_FRAME_HEIGHT)) # 逐帧保存图像 for i in range(frame_count): # 读取当前帧 success, frame = cap.read() if not success: continue # 设定保存图片的路径和文件名 filename = os.path.join(output_path, "frame_" + str(i) + ".jpg") # 保存当前帧为图像 cv2.imwrite(filename, frame) # 释放视频文件 cap.release() # 测试示例 video_to_frames("video.mp4", "output_folder")

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

ZZY_dl

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

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

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

打赏作者

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

抵扣说明:

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

余额充值