溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

發(fā)布時間:2020-10-26 17:11:27 來源:億速云 閱讀:687 作者:Leah 欄目:開發(fā)技術

本篇文章給大家分享的是有關利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

環(huán)境變量

此電腦--屬性--高級系統(tǒng)設置--環(huán)境變量
在系統(tǒng)變量(也就是下面那一半)處找到新建,按如下所示的方法填寫

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

再將%FFMPEG_HOME%以及%FFMPEG_HOME%\bin寫入系統(tǒng)變量的Path中
然后一路確定即可

驗證

win+R,cmd
輸入ffmpeg -version

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

ffmpeg的使用

對于我將B站PC端緩存的音頻mp4和視頻mp4文件合并的需求,需要用到的命令為:
ffmpeg.exe -i audio1.mp4 -i video.mp4 -acodec copy -vcodec copy output.mp4
可以把mp4的文件設置成絕對路徑,這樣就可以轉換指定路徑的文件以及保存到指定路徑了,比如這樣:
ffmpeg.exe -i "E:\嗶哩嗶哩視頻\ss27993\77413703\1\audio1.mp4" -i "E:\嗶哩嗶哩視頻\ss27993\77413703\1\video.mp4" -acodec copy -vcodec copy "E:\B站導出視頻\Dr.STONE石紀元\第22話寶物.mp4
通過PC端緩存的未合并的視頻和音頻,全都是命名為video.mp4和audio.mp4
PS:有些兄弟是導出的手機端緩存的視頻和音頻,是m4s格式的,方法也一樣
但光有這條命令還不夠,需要自己手動一個個操作,太麻煩了
因此我還需要使用Python來自動幫我完成工作

Python實現自動處理

雖然Python可以實現自動化,減少時間的浪費,但最快的還是以后記得緩存時勾選自動合并

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

文件結構

PC端緩存的視頻保存的文件結構有很多種,我只是根據我碰到的情況寫的,但大同小異,修改起來也不麻煩,只是再加個if和else罷了

番劇緩存結構

緩存的番劇是這種結構:

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法
利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法
利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

上面舉的例子是Dr.stone石紀元,我緩存的鬼滅之刃也是如此
特點是在一個以視頻ID號名稱的文件夾(ss27993)后,跟著許多子文件夾(57983089等),然后在這些子文件夾中又有一個或多個子文件夾(比如1),然后緩存的視頻保存在這個文件夾里,里面有一個info文件(就是json格式),還有audio1.mp4和video.mp4。
PS:
還有一個xml文件,是彈幕信息,暫時我不知道怎么處理

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

常規(guī)緩存結構

除了番劇,一般的視頻緩存的結構是這樣的

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法
利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

不難看出,這比番劇要少一個層級

文件信息

文件信息主要由info文件和dvi文件來記錄
這兩種文件都可以直接以json文件來處理,也就是,首先open函數打開文件,然后用json.load轉成字典。。。
然后我還發(fā)現了一個特點是,視頻和音頻所在的目錄下是info文件,而它的上一層目錄下是dvi文件
雖然文件格式基本是一致的,但是里面的鍵-值關系卻不一致,單集視頻的名稱在番劇中對應的是鍵Description,在其他視頻中對應的是PartName
視頻總的名稱保存在外層的目錄下的info文件或dvi文件中,番劇中對應的鍵是SeasonTitle,在其他視頻中對應的是Title

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

具體問題具體分析,首先由于我實踐得較少,這樣的總結不一定對,然后以后也許也會有新的格式、新的變化

代碼

具體代碼

以下是我的Python代碼,你可以先試試能不能用,用不了的話,可以在理解的基礎上修改。理解不了的話可以看我的后面的解釋,以及代碼中的注釋,對于運行過程中一些變量的值,我都把它放在注釋中了,方便你理解。

# -*- coding = utf-8 -*-
# @time:2020/10/17/017 23:09
# Author:cyx
# @File:main.py.py
# @Software:PyCharm

# 從.info文件中獲得了Title信息,但是如果其中有某些特殊字符,保存時可能出現問題
def get_correct_title(title):

  error_set = ['/', '\\', ':', '*', '&#63;', '"', '|', '<', '>', '\b', ' ', '.']
  correct_title = title
  # print(title)
  for c in correct_title:
    if c in error_set:
      correct_title = correct_title.replace(c, '')
  return correct_title

def popen(cmd):
  # https://blog.csdn.net/qq_41451161/article/details/82901235
  try:
    popen = subprocess.Popen(cmd, stdout=subprocess.PIPE)
    popen.wait()
    lines = popen.stdout.readlines()
    return [line.decode('gbk') for line in lines]
  except BaseException as e:
    return -1

if __name__ == '__main__':
  import os
  import json
  import subprocess

  # ffmpeg -i video.m4s -i audio.m4s -c:v copy -c:a aac -strict experimental output.mp4
  # ffmpeg.exe -i audio1.mp4 -i video.mp4 -acodec copy -vcodec copy output.mp4
  AVhao = input("請輸入視頻AV號:")
  superPath = "E:\\嗶哩嗶哩視頻" + "\\" + AVhao
  partDirs = [] # 保存每P視頻所在的文件夾路徑
  paths = os.listdir(superPath) # 獲取當前路徑下所有的文件(包括文件夾)名稱
  # paths = ['8','9']
  # 有時候,會莫名其妙的少了幾個視頻,可以通過重載來重新加載缺失的視頻
  # print(paths)
  # paths
  # ['27993.info', '57983089', '58612211', '59811008', '60862133', '61898240', '62925012', '64005445', '65020725', '66013155', '66808912', '67587875', '68398229', '69175748', '70021307', '70873680', '71617211', '73379440', '74051851', '74974157', '75746600', '76619409', '77413703', '78266594', '79070874', 'cover.jpg', 'desktop.ini']
  # 獲取每P視頻所在的文件夾路徑
  savePos = ''
  seq = []
  # 鑒于有些up主命名時毫無規(guī)律,導出后無法正常排序,只能手動排序了
  # 根據AV號名文件夾下的子文件夾的名稱進行排序,但是番劇的話不是這樣排序,不過番劇單集的名稱很規(guī)范,不需要這樣
  for p in paths:
    if '.' not in p:
      seq.append(p)
    if 'info' in p:
      # print(p)
      # p:
      # 27993.info
      info = superPath + "\\" + p
      with open(info, 'r', encoding='utf-8') as load_f:
        load_dict = json.load(load_f)
        projectTitle = load_dict['SeasonTitle']
        projectTitle = get_correct_title(projectTitle)
        savePos = 'E:\\B站導出視頻\\' + projectTitle
        print('savePos: ', savePos)

    if 'dvi' in p:
      # print(p)
      # P:
      # 328738595.dvi
      dvi = superPath + "\\" + p
      with open(dvi, 'r', encoding='utf-8') as load_f:
        load_dict = json.load(load_f)
        projectTitle = load_dict['Title']
        projectTitle = get_correct_title(projectTitle)
        savePos = 'E:\\B站導出視頻\\' + projectTitle
        print('savePos: ', savePos)
    # 防止文件存在時再次生成該文件夾出現錯誤
    try:
      os.mkdir(savePos)
      break
    except:
      pass

    subDir = superPath + "\\" + p
    if os.path.isdir(subDir):
      # print(subDir)
      # 所有子文件夾的路徑保存在partDirs中
      partDirs.append(subDir)
  # print("partDirs: ",partDirs)
  # partDirs: ['E:\\嗶哩嗶哩視頻\\ss27993\\57983089', 'E:\\嗶哩嗶哩視頻\\ss27993\\58612211', 'E:\\嗶哩嗶哩視頻\\ss27993\\59811008', 'E:\\嗶哩嗶哩視頻\\ss27993\\60862133', 'E:\\嗶哩嗶哩視頻\\ss27993\\61898240', 'E:\\嗶哩嗶哩視頻\\ss27993\\62925012', 'E:\\嗶哩嗶哩視頻\\ss27993\\64005445', 'E:\\嗶哩嗶哩視頻\\ss27993\\65020725', 'E:\\嗶哩嗶哩視頻\\ss27993\\66013155', 'E:\\嗶哩嗶哩視頻\\ss27993\\66808912', 'E:\\嗶哩嗶哩視頻\\ss27993\\67587875', 'E:\\嗶哩嗶哩視頻\\ss27993\\68398229', 'E:\\嗶哩嗶哩視頻\\ss27993\\69175748', 'E:\\嗶哩嗶哩視頻\\ss27993\\70021307', 'E:\\嗶哩嗶哩視頻\\ss27993\\70873680', 'E:\\嗶哩嗶哩視頻\\ss27993\\71617211', 'E:\\嗶哩嗶哩視頻\\ss27993\\73379440', 'E:\\嗶哩嗶哩視頻\\ss27993\\74051851', 'E:\\嗶哩嗶哩視頻\\ss27993\\74974157', 'E:\\嗶哩嗶哩視頻\\ss27993\\75746600', 'E:\\嗶哩嗶哩視頻\\ss27993\\76619409', 'E:\\嗶哩嗶哩視頻\\ss27993\\77413703', 'E:\\嗶哩嗶哩視頻\\ss27993\\78266594', 'E:\\嗶哩嗶哩視頻\\ss27993\\79070874']
  videoPos = ''
  i = 0
  for p in partDirs:
    # print(p)
    # 列出子文件夾中的所有文件
    sublist = os.listdir(p)
    # 檢查info文件是否在當前子文件夾中
    for file in sublist:
      # print(file)
      # file:
      # 1
      # 57983089.
      # dvi
      # cover.jpg
      # desktop.ini
      if 'info' in file:
        infoPos = p + "\\" + file
        videoPos = p
      else:
        subsubDir = p + "\\" + file
        if os.path.isdir(subsubDir):
          # print(subsubDir)
          # subsubDir: E:\嗶哩嗶哩視頻\ss27993\57983089\1
          subsubList = os.listdir(subsubDir)
          for subsubFile in subsubList:
            if 'info' in subsubFile:
              infoPos = subsubDir + "\\" + subsubFile
              videoPos = subsubDir
              break
      with open(infoPos, 'r', encoding='utf-8') as load_f:
        load_dict = json.load(load_f)
        if 'ss' in AVhao:
          videoTitle = load_dict['Description']
        else:
          videoTitle = load_dict['PartName']

        videoTitle = get_correct_title(videoTitle)
        print('videoTitle: ', videoTitle)
        videoDir = videoPos + "\\" + 'video.mp4'
        audioDir = videoPos + "\\" + 'audio1.mp4'
        # print('videoDir: ', videoDir)
        # print('audioDir: ', audioDir)
        # videoDir: E:\嗶哩嗶哩視頻\ss27993\74051851\1\video.mp4
        # audioDir: E:\嗶哩嗶哩視頻\ss27993\74051851\1\audio1.mp4
        if 'ss' in AVhao:
          outDir = savePos + "\\" + videoTitle + '.mp4'
        else:
          outDir = savePos + "\\" + seq[i] + '_' + videoTitle + '.mp4'
          i += 1
        # 對于那些命名很規(guī)范的視頻,可以不用自己再排序,進行一下重載,不規(guī)范的視頻再把這句注釋掉就好
        outDir = savePos + "\\" + videoTitle + '.mp4'
        # command = 'cd ' + superPath + '\\64 && ' # && 多名命令
        # command = 'cd ' + 'E:\\ProgramFiles\\ffmpeg' + ' && '
        command = 'E:\\FFmpeg\\bin\\ffmpeg.exe -i ' + '"' + audioDir + '"' ' -i ' + '"' + videoDir + '"'+ ' -acodec copy -vcodec copy ' + '"' + outDir + '"'
        # print("保存地址",outDir)
        # 保存地址 E:\B站導出視頻\[Lynda視頻]音頻錄制錄音技巧教程(中英雙語字幕)全集130課時AudioRecordingTechniques混音錄音棚音樂工作室歌曲調音\98錄制獨奏薩克斯演奏技巧二.mp4
        # print(command)
        # command = 'E:\\FFmpeg\\bin\\ffmpeg.exe -i "E:\嗶哩嗶哩視頻\ss27993\77413703\1\audio1.mp4" -i "E:\嗶哩嗶哩視頻\ss27993\77413703\1\video.mp4" -acodec copy -vcodec copy "E:\B站導出視頻\Dr.STONE石紀元\第22話寶物.mp4'
        # os.system(command)
        popen(command)
        # ffmpeg.exe -i audio1.mp4 -i video.mp4 -acodec copy -vcodec copy output.mp4
        break

代碼說明

如你所見我的編程水平不高,模塊化做的很差,不便于理解,所以有必要進行說明。
直接從main開始看起吧。

 AVhao = input("請輸入視頻AV號:")
  superPath = "E:\\嗶哩嗶哩視頻" + "\\" + AVhao
  partDirs = [] # 保存每P視頻所在的文件夾路徑
  paths = os.listdir(superPath) # 獲取當前路徑下所有的文件(包括文件夾)名稱

首先是用input接收AV號或BV號的輸入,放入AVhao變量中。
然后用superPath變量存放你需要合并的視頻的根目錄。比如我在B站緩存的所有視頻存放在E:\嗶哩嗶哩視頻下,注意程序中要有兩條\,然后superPath就是E:\嗶哩嗶哩視頻\AVhao
os.listdir(),括號中的參數必須是一個真實的路徑,這個函數可以得到這個路徑下所有的文件和文件夾的名稱。
我用paths來存放E:\嗶哩嗶哩視頻\AVhao路徑下所有的文件名稱和文件夾名稱。
為了防止我的表達能力有限帶來的理解上的不便,你可以看圖,paths對應的是下圖中的內容:

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

這個變量的類型是列表,所以可以用for循環(huán)來遍歷。

savePos = ''
  seq = []
  for p in paths:
    if '.' not in p:
      seq.append(p)
    if 'info' in p:
      # print(p)
      # p:
      # 27993.info
      info = superPath + "\\" + p
      with open(info, 'r', encoding='utf-8') as load_f:
        load_dict = json.load(load_f)
        projectTitle = load_dict['SeasonTitle']
        projectTitle = get_correct_title(projectTitle)
        savePos = 'E:\\B站導出視頻\\' + projectTitle
        print('savePos: ', savePos)

    if 'dvi' in p:
      # print(p)
      # P:
      # 328738595.dvi
      dvi = superPath + "\\" + p
      with open(dvi, 'r', encoding='utf-8') as load_f:
        load_dict = json.load(load_f)
        projectTitle = load_dict['Title']
        projectTitle = get_correct_title(projectTitle)
        savePos = 'E:\\B站導出視頻\\' + projectTitle
        print('savePos: ', savePos)
    # 防止文件存在時再次生成該文件夾出現錯誤
    try:
      os.mkdir(savePos)
      break
    except:
      pass

    subDir = superPath + "\\" + p
    if os.path.isdir(subDir):
      # print(subDir)
      # 所有子文件夾的路徑保存在partDirs中
      partDirs.append(subDir)  

我設置了一個savePos變量,用來表示合并后的視頻保存的位置,因為我希望將視頻保存在一個我自己指定的文件夾下,同時這個文件夾的名稱是這個視頻的名稱,比如Dr.stone石紀元。
因為想自動化操作,所以我通過緩存文件夾中的info文件和dvi文件來找到視頻的名稱。
使用open函數打開這兩個文件中的一個,因為不確定文件結構是什么樣的,所以我用了兩個if語句。
然后再用json.load函數將其加載為字典,并根據對應的鍵讀取對應的值,從而可以拼接處對應的保存地址savePos。
由于創(chuàng)建已經存在的同名文件夾會發(fā)生錯誤,為避免這種可能,我將創(chuàng)建目錄的操作放在了try語句下。
創(chuàng)建目錄用的是os.mkdir()函數,括號中是一個絕對路徑。
然后我用subDir來表示子文件夾的名稱,注意!是子文件夾,而不是文件。
我用os.path.isdir(subDir)來進行判斷,如果是文件夾而不是文件的話,就加到partDirs列表中,partDirs.append(subDir)
這個列表中的每個元素都是一個子文件夾的絕對路徑,比如:E:\嗶哩嗶哩視頻\ss27993\57983089
而這個名為seq的顯得很突兀,這個其實我也是后來加的,這個列表的作用在于記錄當前子文件夾的名稱,也就是在AVhao文件夾的下一層,如果不是番劇的話,應當有許多個文件夾分別是1、2、3等等,這些其實對應的是播放列表中的順序。

利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法
利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法

而之所以使用if '.' not in p,因為這一層的文件夾全都是用來表示播放順序的,因此不存在后綴名,從而也就沒有“.”。

videoPos = ''
  i = 0
  for p in partDirs:
    # print(p)
    # 列出子文件夾中的所有文件
    sublist = os.listdir(p)
    # 檢查info文件是否在當前子文件夾中
    for file in sublist:
      # print(file)
      # file:
      # 1
      # 57983089.
      # dvi
      # cover.jpg
      # desktop.ini
      if 'info' in file:
        infoPos = p + "\\" + file
        videoPos = p
      else:
        subsubDir = p + "\\" + file
        if os.path.isdir(subsubDir):
          # print(subsubDir)
          # subsubDir: E:\嗶哩嗶哩視頻\ss27993\57983089\1
          subsubList = os.listdir(subsubDir)
          for subsubFile in subsubList:
            if 'info' in subsubFile:
              infoPos = subsubDir + "\\" + subsubFile
              videoPos = subsubDir
              break
      with open(infoPos, 'r', encoding='utf-8') as load_f:
        load_dict = json.load(load_f)
        if 'ss' in AVhao:
          videoTitle = load_dict['Description']
        else:
          videoTitle = load_dict['PartName']

        videoTitle = get_correct_title(videoTitle)
        print('videoTitle: ', videoTitle)
        videoDir = videoPos + "\\" + 'video.mp4'
        audioDir = videoPos + "\\" + 'audio1.mp4'
        # print('videoDir: ', videoDir)
        # print('audioDir: ', audioDir)
        # videoDir: E:\嗶哩嗶哩視頻\ss27993\74051851\1\video.mp4
        # audioDir: E:\嗶哩嗶哩視頻\ss27993\74051851\1\audio1.mp4
        if 'ss' in AVhao:
          outDir = savePos + "\\" + videoTitle + '.mp4'
        else:
          outDir = savePos + "\\" + seq[i] + '_' + videoTitle + '.mp4'
          i += 1
        # 對于那些命名很規(guī)范的視頻,可以不用自己再排序,進行一下重載,不規(guī)范的視頻再把這句注釋掉就好
        outDir = savePos + "\\" + videoTitle + '.mp4'
        # command = 'cd ' + superPath + '\\64 && ' # && 多名命令
        # command = 'cd ' + 'E:\\ProgramFiles\\ffmpeg' + ' && '
        command = 'E:\\FFmpeg\\bin\\ffmpeg.exe -i ' + '"' + audioDir + '"' ' -i ' + '"' + videoDir + '"'+ ' -acodec copy -vcodec copy ' + '"' + outDir + '"'
        # print("保存地址",outDir)
        # 保存地址 E:\B站導出視頻\[Lynda視頻]音頻錄制錄音技巧教程(中英雙語字幕)全集130課時AudioRecordingTechniques混音錄音棚音樂工作室歌曲調音\98錄制獨奏薩克斯演奏技巧二.mp4
        # print(command)
        # command = 'E:\\FFmpeg\\bin\\ffmpeg.exe -i "E:\嗶哩嗶哩視頻\ss27993\77413703\1\audio1.mp4" -i "E:\嗶哩嗶哩視頻\ss27993\77413703\1\video.mp4" -acodec copy -vcodec copy "E:\B站導出視頻\Dr.STONE石紀元\第22話寶物.mp4'
        # os.system(command)
        popen(command)
        # ffmpeg.exe -i audio1.mp4 -i video.mp4 -acodec copy -vcodec copy output.mp4
        break

這一部分是我用來實現合并特定路徑的視頻和音頻,并最終導出到指定目錄的。
videoPos是待合并的視頻的路徑,音頻文件也在這一目錄下。
for p in partDirs:,在這個for循環(huán)中遍歷的是partDirs列表,這個列表由前面的步驟得到,其中的每個元素都是一個路徑。
接下來的這個if-else是用來區(qū)別番劇和普通視頻合集的,因為它們有不同的目錄結構。
infoPos用來記錄包含單集視頻名稱的info文件的路徑,然后用open函數打開這個info文件,根據AVhao中是否有ss,判斷是否是番劇,如果有ss,則表明是番劇,對應的視頻名稱為Description鍵對應的值。否則的話,對應的視頻名稱為PartName鍵對應的值,但是如果是要保存為文件的話,當然不能直接以這個名稱命名,否則極有可能發(fā)生錯誤,因此我用了一個get_correct_title函數來對標題進行重載,以確保格式正確。
videoDir和audioDir分別是待合并的視頻和音頻的絕對路徑。
然后我根據AVhao中是否有ss,來判斷是否是番劇。因為番劇的單集視頻名稱通常會有序號,所以我可以將輸出視頻的保存路徑設置為保存目錄加視頻名.mp4。
許多up主的視頻名稱沒有體現視頻的先后順序,這樣帶來的問題是導出后順序播放時產生跳集現象。因此我用seq加下劃線的方式來為視頻排序。比如“1_簡介.mp4”。
而對于視頻名稱本身有排序的情況1_1簡介.mp4,這樣有些奇怪,所以我們碰到這種情況時,可以直接在下面添加一個outDir = savePos + "\\" + videoTitle + '.mp4',其他情況不用時注釋掉就好。
最后是用Python程序執(zhí)行Dos命令,我將命令設為command變量,通過for循環(huán),會自動生成不同的命令,然后執(zhí)行命令行有兩種方法,一種是導入os庫,使用os.system(command),另一種是我在https://blog.csdn.net/qq_41451161/article/details/82901235借用的popen函數,這個也可以用,但需要導入subprocess庫。
使用命令行上,有兩個比較坑的地方,一是前面必須給出ffmpeg.exe的絕對路徑,也就是E:\FFmpeg\bin\ffmpeg.exe,在Python中用是這樣的,但直接在命令行中,只要輸入ffmpeg.exe即可(前提是你設置好了環(huán)境變量)。第二個坑是,command賦值時,一定給路徑加上引號,否則的話識別命令時會發(fā)生錯誤。

以上就是利用ffmpeg+Python實現MP4格式音頻與視頻的合并的方法,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI