同步快速判斷視訊是否可以播放

KyXu發表於2017-09-05

背景

拿到一個視訊的 url 地址(無論是遠端還是本地),有時候在播放之前需要檢測該視訊是否可以播放(本地可能是檔案損壞,遠端地址情況更復雜),下面介紹兩種適用不同情況的方法來實現。

常用的非同步方法

import UIKit
import AVFoundation

class ViewController: UIViewController {

    var avplayer: AVPlayer!

    override func viewDidLoad() {
        super.viewDidLoad()

        let url = URL(string: "http://gslb.miaopai.com/stream/24fONfescp-SRz61DjJz62WO1LLIwjIQXHthNg__.mp4")!

        avplayer = AVPlayer(url: url)
        avplayer.addObserver(self, forKeyPath: "status", options: .new, context: nil)
    }

    override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
        if (avplayer == object as? AVPlayer) && (keyPath == "status") {
            print((avplayer.status == .readyToPlay) ? "readyToPlay" : "fail")
        }
    }

}複製程式碼

如程式碼所示,我們可以使用 AVPlayer (MPMoviePlayerController 在 iOS 9 被 deprecated) 嘗試解析 url,進行視訊播放。然後可以通過 KVO 在它的 status 屬性變為 “readyToPlay” 的時候,進行播放等操作。在這裡這個 status 只能非同步獲取,如果你直接在初始化 AVPlayer 之後就同步獲取這個值,這個值會是 “unknown”。

如果你在這裡需要直接使用這個 AVPlayer 進行視訊播放,那麼推薦使用這個 KVO 方法。否則會有這樣幾個缺點:

  • avplayer 用於判斷視訊可播放性,本應該是區域性變數,現在因為 KVO 它的 scope 被擴大了
  • KVO 方法在程式碼可讀性上不如同步的直接判斷
  • 控制檯會預設列印下面這些東西:
2017-09-05 14:43:36.983707+0800 VideoPlayable[40473:10508878] [aqme] 254: AQDefaultDevice (1): skipping input stream 0 0 0x0
2017-09-05 14:43:38.992581+0800 VideoPlayable[40473:10508878] [aqme] 254: AQDefaultDevice (173): skipping input stream 0 0 0x0
2017-09-05 14:43:41.000864+0800 VideoPlayable[40473:10508878] [aqme] 254: AQDefaultDevice (173): skipping input stream 0 0 0x0複製程式碼

同步判斷方法

import UIKit
import AVFoundation

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        let url = URL(string: "http://gslb.miaopai.com/stream/24fONfescp-SRz61DjJz62WO1LLIwjIQXHthNg__.mp4")!
        let avasset = AVAsset(url: url)
        print(avasset.isPlayable)
    }
}複製程式碼

經過一番折騰,發現直接通過 url 新建出 AVAsset,即可通過
isPlayable 屬性判斷視訊是否可以正常播放
。如果僅僅是想要判斷視訊可播放性,而不需要使用 AVPlayer,建議使用這種方法,可以將 AVAsset 的建立邏輯加入到你的自定義播放器的 init 方法中。

我嘗試在 iPhone 6s 上使用本地的一個約 800k 的 mp4 視訊進行粗略測試,發現新建 AVAsset 耗時約 0.02s,新建 AVPlayer 非同步返回耗時約 0.05s,這種同步方法大概快了一倍。

相關文章