《自拍教程66》Python ffmpeg批量壓縮視頻

来源:https://www.cnblogs.com/zipython/archive/2020/05/19/12920095.html
-Advertisement-
Play Games

案例故事: 測試過程中發現Bug視頻, 需要提供給開發用於解Bug的參考, 但是視頻拍攝後,太大且無法在微信客戶端傳輸的問題, 於是乎出現過測試人員通過winzip分批壓縮(part1, part2, part3), 再通過微信傳輸視頻壓縮包的"亂象": 作為測試總監,手底下的人這麼"壓縮視頻“我是 ...


案例故事: 測試過程中發現Bug視頻, 需要提供給開發用於解Bug的參考,
但是視頻拍攝後,太大且無法在微信客戶端傳輸的問題,
於是乎出現過測試人員通過winzip分批壓縮(part1, part2, part3),
再通過微信傳輸視頻壓縮包的"亂象":

作為測試總監,手底下的人這麼"壓縮視頻“我是覺得丟人的,
(1).視頻文件已經是二進位文件了,其實winzip已經壓縮不了什麼,
winzip壓縮軟體一般只適合壓縮文本數據文件。
(2).視頻壓縮應該使用Video的編碼技術實現二次編碼壓縮,業界最常用的肯定是ffmpeg.exe工具。


還有比如當碰到一個Bug的視頻太大了,無法作為附件上傳到Bug系統,
會做視頻壓縮是合格的測試人員的必備能力之一,
本篇主要介紹如何通過ffmpeg 來實現批量壓縮視頻。


視頻的基本知識點
  1. 視頻文件是由視頻流,音頻流組成的將一系列圖片快速播放產生的動態圖像,音頻的聚合體, 視頻文件的音頻流一般非常小,但是視頻流非常大,視頻流的大小主要取決於編碼技術,解析度,幀率這3個因素。

  2. 編碼技術Codec,是壓縮多張圖片的編碼技術,比如多張圖片組成的一個視頻,
    如果相連圖片的像素相差不大,則只記錄差異像素點即可,
    從而實現了不影響畫質的情況下,將視頻文件最小化,
    ffmpeg的預設的編碼格式是:H.264, 其實還有很多編碼格式,
    比如Mpeg4, WMV10,H.263等等。

  3. 解析度Resolution, 是視頻每一幀(每張圖片)的圖片大小,是由一個一個像素點(pixel)組成的。

  4. 幀率是fps, 每秒鐘的圖片數,一般每秒4張圖片以上(>4fps)就可以有明顯的視頻動畫效果。

  5. 視頻容器是Container, 是用於封裝視頻流,音頻流的一個容器格式,一般有.mp4, .3gp, .avi, .mov等等。

  6. 比特率bitrate,是每秒鐘的數據量,其數據量大小基本是受視頻編碼格式,解析度,幀率3者因素影響的。

  7. 視頻每做一次壓縮,視頻的數據量就會減少,且不可逆!

準備階段
  1. ffmpeg的下載地址可以去:ffmpeg - 音視頻圖像編解碼工具這篇文章查看。
    視頻壓縮的常用命令模板是:
    ffmpeg -i input.mp4 -s 640x480 -r 12 -y output.mp4
    以上命令模板可以將input.mp4進行重編碼(按幀率12fps,解析度640x480),
    並另存為output.mp4 , -y的意思是如果已經有這個文件,不詢問直接覆蓋。

  2. 如果要批量壓縮視頻,我們還是用輸入輸出模式,文件結構如下:

	+---Input_Video    #批量放入待壓縮的視頻
	|       1.mp4
	|       2.mp4
	|       
	+---Output_Video   #批量輸出已壓縮的視頻, 加一個尾碼_c,代表以及轉換完。
	|       1_c.mp4
	|       2_c.mp4
	|
	\convert_video.py  #Python視頻轉碼腳本,雙擊運行即可
  1. 記得將ffmpeg.exe 丟到系統Path環境變數路徑下去。

Python批處理腳本形式

記住批處理腳本的精髓:批量順序執行語句

# coding=utf-8

import os

NEW_RESOLUTION = "640x480"  # 目標解析度,常量
NEW_FPS = 12  # 目標幀率,常量

curpath = os.getcwd()  # 獲取當前路徑
input_dir = os.path.join(curpath, "Input_Video")
output_dir = os.path.join(curpath, "Output_Video")
input_video_list = os.listdir(input_dir)  # 獲取視頻列表

# 如果沒有Output_Video這個文件夾,則創建這個文件夾
if not os.path.exists(output_dir):
    os.mkdir(output_dir)

# 開始批量二次編碼壓縮視頻轉碼
for each_video in input_video_list:
    video_name, _ = os.path.splitext(each_video)  # _是沒意義,就只是一個無用代號,占個坑而已
    ffmpeg_command = ("ffmpeg -i %s%s%s -s %s -r %s -y %s%s%s_c.mp4" % (
        input_dir, os.sep, each_video, NEW_RESOLUTION, NEW_FPS, output_dir, os.sep, video_name))
    print(ffmpeg_command)
    os.system(ffmpeg_command)

os.system("pause")

Python面向過程函數形式

面向過程函數的編程思維應該是這樣的:
你需要多少個功能(函數),才能做成這個事。
最好把功能(函數)都儘量封裝好,只暴露一些的參數介面即可。

# coding=utf-8

import os


def convert_video(input_video_path, new_resolution, new_fps, output_video_path):
    ffmpeg_command = ("ffmpeg -i %s -s %s -r %s -y %s" % (
        input_video_path, new_resolution, new_fps, output_video_path))
    print(ffmpeg_command)
    os.system(ffmpeg_command)


curpath = os.getcwd()  # 獲取當前路徑
input_dir = os.path.join(curpath, "Input_Video")
output_dir = os.path.join(curpath, "Output_Video")
input_video_list = os.listdir(input_dir)  # 獲取視頻列表

# 如果沒有Output_Video這個文件夾,則創建這個文件夾
if not os.path.exists(output_dir):
    os.mkdir(output_dir)

# 開始批量二次編碼壓縮視頻轉碼
for each_video in input_video_list:
    video_name, _ = os.path.splitext(each_video)  # _是沒意義,就只是一個無用代號,占個坑而已
    input_video_path = input_dir + os.sep + each_video
    output_video_path = output_dir + os.sep + video_name + "_c.mp4"
    convert_video(input_video_path, "640x480", "12", output_video_path)
os.system("pause")

Python面向對象類形式

面向對象類的編程思維應該是這樣的:
如果給你一個空白的世界,在這個世界里你需要哪些種類的事物,
這些種類的事物都具備哪些共有的屬性與方法,
這些種類(類)的事物(對象),和其他種類(其他類)的事物(其他對象)有什麼關係。
儘量把這些類封裝好,只暴露對外的屬性(變數)和方法(函數)即可。

# coding=utf-8

import os


class VideoConverter(object):
    def __init__(self, input_video_path, new_resolution, new_fps, output_video_path):
        self.input_video_path = input_video_path
        self.new_resolution = new_resolution
        self.new_fps = new_fps
        self.output_video_path = output_video_path

    def convert_video(self):
        ffmpeg_command = ("ffmpeg -i %s -s %s -r %s -y %s" % (
            self.input_video_path, self.new_resolution, self.new_fps, self.output_video_path))
        print(ffmpeg_command)
        os.system(ffmpeg_command)


if __name__ == '__main__':
    curpath = os.getcwd()  # 獲取當前路徑
    input_dir = os.path.join(curpath, "Input_Video")
    output_dir = os.path.join(curpath, "Output_Video")
    input_video_list = os.listdir(input_dir)  # 獲取視頻列表

    # 如果沒有Output_Video這個文件夾,則創建這個文件夾
    if not os.path.exists(output_dir):
        os.mkdir(output_dir)

    # 開始批量二次編碼壓縮視頻轉碼
    for each_video in input_video_list:
        video_name, _ = os.path.splitext(each_video)  # _是沒意義,就只是一個無用代號,占個坑而已
        input_video_path = input_dir + os.sep + each_video
        output_video_path = output_dir + os.sep + video_name + "_c.mp4"
        v_obj = VideoConverter(input_video_path, "640x480", "12", output_video_path)
        v_obj.convert_video()
    os.system("pause")

本案例素材下載

包括:Input_Video(含一個H.264_1280x720_24fps.mp4視頻),Python腳本
跳轉至官網下載
武散人出品,請放心下載!

小提示以上3種形式,只是為了訓練培養編程思維,其實主要的核心代碼就是ffmpeg命令那麼一條,
如果不涉及批量處理,直接敲ffmpeg原始命令即可實現轉碼,
以上基本可以實現將100M的視頻壓縮到10M左右。


更多更好的原創文章,請訪問官方網站:www.zipython.com
自拍教程(自動化測試Python教程,武散人編著)
原文鏈接:https://www.zipython.com/#/detail?id=52f5f5ed29a14614bc522754af3b7b4e
也可關註“武散人”微信訂閱號,隨時接受文章推送。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1 前言 良好的代碼習慣是一個優秀程式員應該具備的品質,但靠人的習慣與記憶來保證代碼質量,始終不是一件靠譜的事。在電腦行業應該深知,只要是人為的,都會有操作風險。本文講解如何通過 搭建代碼檢測平臺 ,並使用它來檢測 項目的代碼。 2 Docker安裝SonarQube 2.1 安裝 通過 安裝,方 ...
  • Spring Boot 教程 Mybatis 1. 什麼是Mybatis? MyBatis 是一款優秀的持久層框架,它支持自定義 SQL、存儲過程以及高級映射。MyBatis 免除了幾乎所有的 JDBC 代碼以及設置參數和獲取結果集的工作。MyBatis 可以通過簡單的 XML 或註解來配置和映射原 ...
  • Spring Boot 教程 HelloWorld 1. Spring Boot 的由來 大家都知道,Spring框架是Java生態中舉足輕重的輕量型框架,幫助我們廣大的大佬們進行Java開發。Spring框架的發展非常的迅速,從最開始的Spring核心容器幫助我們管理Java對象,到後來各種業務上 ...
  • C++語言規定,提供預設值時必須按從右至左的順序提供,即有預設值的形參必須在形參列表的最後。如果有某個形參沒有預設值,則它左側的所有形參都不能有預設值。調用函數時,主調函數的實參與被調函數的形參按從左至右的順序進行匹配對應。 ...
  • 序言 What is Webscoket ? websocket 應用場景 簡版群聊實現 代碼例子 小結 Webscoket Websokcet 是一種單個 "TCP" 連接上進行 "全雙工" 通信的協議,通過 "HTTP" /1.1 協議的101狀態碼進行握手。 http://websocket. ...
  • Java 1.1Java簡介 Java是一種跨平臺的,面向對象的程式設計語言。無論是電腦還是手機,到處都運行著JAVA開發的應用程式;JAVA程式可以在任何電腦、操作系統以及支持JAVA的硬體設備上運行。 什麼是JAVA語言? Java是前Sun公司(現甲骨文股份有限公司Oracle)於1995年 ...
  • C++語言中加入了面向對象的概念,雖然C語言的語法絕大部分都被保留在C++語言中,但C++的程式結構與C語言的程式結構存在很大差別。C++語言對C語言做了很多改進,C++語言相對於C語言的最根本的變化是引進了類和對象的概念。 ...
  • Go中也提供了反射機制,與Java一樣Go的反射也是在運行時獲取對象的相關信息,更新對象內部狀態;Golang通過反射可以獲取對象類型、欄位類型與值、調用struct實例方法、更新實例值等; Go關於反射相關的對象、函數都在reflect包中最主要的兩個為:Type與Value; Go提供了下麵兩個 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...