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

降智警告:本人为编程新手,遵守面向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.array
    image = 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 + 570
    elif num < 1812 :
        x, y, w, h = width - 820, 75, width, 75 + 570
    else :
        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 imageio

from 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

你可能感兴趣的:(Python-从视频到gif(imageio,moviepy,ffmpeg))