当前位置: 代码迷 >> 综合 >> Python-从视频到gif(imageio,moviepy,ffmpeg)
  详细解决方案

Python-从视频到gif(imageio,moviepy,ffmpeg)

热度:97   发布时间:2024-01-09 16:30:28.0

降智警告:本人为编程新手,遵守面向CSDN编程原则,代码架构具有个人特色,仅供参考

 

前言:因为之前看到一个视频里的ed还挺喜欢的,所以想截取下来,做成gif,结果做出来几十M,所以本文还会涉及截取并裁剪视频帧的内容

 

内容分为五个部分:①截取视频,②视频分帧(可选),③图片裁剪(可选),④音频提取(可选),⑤gif/mp4生成

需要用到的库有moviepy,imageio,skimage(scikit-image),numpy,PIL,os

 

一.截取视频

直接使用moviepy.editor可以截取视频,还可以改变帧尺寸

import moviepy.editor as mpy#获取视频内容
clip = mpy.VideoFileClip('ori.mp4')#按时间划分截取(如下,截取22分25秒到23分55秒的内容)
content = clip.subclip((22,25),(23,55))#截取视频的同时改变帧大小,改为480x360
#content = clip.subclip((22,25),(23,55)).resize((480, 360))#将截取的视频写入到新文件保存
content.write_videofile('get.mp4')

如果提示需要ffmpeg.exe

import imageio
imageio.plugins.ffmpeg.download()

 

二.视频分帧

将截取的视频按帧生成图片,如果不需要修改帧(去字幕等),可以跳过

import imageio 
import skimage 
import numpy as np #视频的绝对路径 
filename = 'get.mp4'#使用ffmpeg读取视频内容
reader = imageio.get_reader(filename, 'ffmpeg') for i, im in enumerate(reader) :#将每一帧转为np.arrayimage = skimage.img_as_float(im).astype(np.float64)#将每一帧写入为新图片imageio.imsave(str(i) + '.jpg', image)

 

三.图片裁剪

这里需要你使用其他方式定位你要裁剪下来的图片的区域,找到该区域左上角和右下角的坐标

import os
from PIL import Image#获取图片列表
pic = os.listdir()for p in pic :#若不是要进行操作的图片或是之前已经操作过的图片,则跳过if p.split('.')[1] != 'jpg' or p.split('.')[0][-3:] == 'new':continue#打开图片img = Image.open(p)#获取图片尺寸width, height = img.size#图片编号,后面生成新图片要用num = int(p.split('.')[0])#设置自定义的裁剪位置if num < 1037 :x, y, w, h = 0, 75, 820, 75 + 570elif num < 1812 :x, y, w, h = width - 820, 75, width, 75 + 570else :x, y, w, h = 0, 75, 820, 75 + 570 # 开始截取region = img.crop((x, y, w, h))# 另保存为新图片region.save(p.split('.')[0] + "_new.jpg")
    '''new = img.crop((x,y,w,h))x               w---------------------------------|       |               |       |y|-------A----------------       ||       |               |       ||       |               |       ||       |               |       ||       |               |       |h|-----------------------B       ||                               |---------------------------------'''

 

四.音频提取

从截取的视频里提取出音频

下载ffmpeg.exe,这个东西可以用来处理音频视频等,功能强大,这里仅是用来提取音频

官网下载,在get the packages 里按系统选,可以直接下载exe格式的(我是windows)

跳转后按系统要求选择,点download build

下载后在bin文件夹里找到ffmpeg.exe,跟要提取的视频放在同一目录下,使用以下命令:

ffmpeg -i video.mp4(原视频) -vn audio.mp3(提取音频) -vn

然后就会生成一个audio.mp3文件,就是提取出来的音频

 

五.gif/mp4生成

gif的话很简单

1.使用图片生成gif

import os
import imageio#获取图片列表
pic = [i for i in os.listdir() if i.split('.')[0][-3:] == 'new']#将图片按编号排序
pic = sorted(pic, key = lambda x : int(x.split('.')[0][:-4]))#目标gif的文件名
filename = 'fin.gif'#获取帧列表
frames = [imageio.imread(i) for i in pic]#将帧合成gif
imageio.mimsave(filename, frames, 'GIF', duration = 1 / 24)

 

2.根据视频生成gif(可以直接截取视频生成gif)

import moviepy.editor as mpy#获取视频内容
clip = mpy.VideoFileClip('ori.mp4')#按时间划分截取(如下,截取22分25秒到23分55秒的内容)
content = clip.subclip((22,25),(23,55))#截取视频的同时改变帧大小,改为480x360
#content = clip.subclip((22,25),(23,55)).resize((480, 360))#生成gif
content.write_gif('get.gif')

使用图片跟音频生成视频(图片命名格式:编号_new.jpg)

import os
import imageiofrom moviepy.editor import *#获取图片列表
pic = [i for i in os.listdir() if i.split('.')[0][-3:] == 'new']#将图片按编号排序
pic = sorted(pic, key = lambda x : int(x.split('.')[0][:-4]))#将图片按帧率(fps)划分,fps可调
clip = ImageSequenceClip(pic, fps = 24)#使用音频与帧表合成视频
clip.write_videofile('new.mp4', audio = "audio.mp3")

 

总结:使用imageio和moviepy基本可以满足视频操作的需要,特殊需求可以使用ffmpeg作为辅助。PIL可以对图片进行操作,两者相结合就可以自定义的生成动态文件了。

 

参考:

gif制作 : https://blog.csdn.net/Spade_/article/details/79516322

图片切割:https://blog.csdn.net/t8116189520/article/details/80271661

音频提取:https://blog.csdn.net/cjs68/article/details/49993473

  相关解决方案