如何使用 AVURLAsset 流式传输视频并将缓存数据保存到磁盘

How to stream a video with AVURLAsset and save to disk the cached data

几天前,有人要求我检查从 Internet 下载视频时播放视频的难度。我知道这是一项容易的任务,因为刚才有人告诉我。所以,我检查了一下,超级简单。

问题是我想将视频保存到磁盘,以免强迫用户一次又一次地下载它。

问题是访问缓冲区并将其存储到磁盘。

Whosebug 中的许多答案都说这是不可能的。特别是视频。

我播放视频的原始代码:

import AVFoundation

....

//MARK: - Accessors

lazy var player: AVPlayer = {

    var player: AVPlayer = AVPlayer(playerItem: self.playerItem)

    player.actionAtItemEnd = AVPlayerActionAtItemEnd.None

    return player
}()

lazy var playerItem: AVPlayerItem = {

    var playerItem: AVPlayerItem = AVPlayerItem(asset: self.asset)

    return playerItem
}()

lazy var asset: AVURLAsset = {

    var asset: AVURLAsset = AVURLAsset(URL: self.url)

    return asset
}()

lazy var playerLayer: AVPlayerLayer = {

    var playerLayer: AVPlayerLayer = AVPlayerLayer(player: self.player)

    playerLayer.frame = UIScreen.mainScreen().bounds
    playerLayer.backgroundColor = UIColor.clearColor().CGColor

    return playerLayer
}()

var url: NSURL = {

    var url = NSURL(string: "https://clips.vorwaerts-gmbh.de/big_buck_bunny.mp4")

    return url!
}()

//MARK: - ViewLifeCycle

override func viewDidLoad() {

    super.viewDidLoad()

    view.layer.addSublayer(playerLayer)

    player.play()
}

这个问题的解决方法是使用AVAssetExportSessionAVAssetResourceLoaderDelegate:

第一步是添加通知以了解视频何时结束。然后我们就可以开始存盘了。

override func viewDidLoad() {

    super.viewDidLoad()

    NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(playerItemDidReachEnd(_:)), name: AVPlayerItemDidPlayToEndTimeNotification, object: nil)

    ...
}

deinit {

    NSNotificationCenter.defaultCenter().removeObserver(self)
}

我们函数的实现:

func playerItemDidReachEnd(notification: NSNotification) {

    if notification.object as? AVPlayerItem  == player.currentItem {

        let exporter = AVAssetExportSession(asset: asset, presetName: AVAssetExportPresetHighestQuality)

        let filename = "filename.mp4"

        let documentsDirectory = NSFileManager.defaultManager().URLsForDirectory(NSSearchPathDirectory.DocumentDirectory, inDomains: NSSearchPathDomainMask.UserDomainMask).last!

        let outputURL = documentsDirectory.URLByAppendingPathComponent(filename)

        exporter?.outputURL = outputURL
        exporter?.outputFileType = AVFileTypeMPEG4

        exporter?.exportAsynchronouslyWithCompletionHandler({

            print(exporter?.status.rawValue)
            print(exporter?.error)
        })
    }
}

最后我们需要让我们的 AVURLAsset AVAssetResourceLoaderDelegate 委托:

lazy var asset: AVURLAsset = {

    var asset: AVURLAsset = AVURLAsset(URL: self.url)

    asset.resourceLoader.setDelegate(self, queue: dispatch_get_main_queue())

    return asset
}()

并且:

extension ViewController : AVAssetResourceLoaderDelegate {

}

I created a small demo with this code in GitHub.

Calm 团队已将我们的实现开源。它可以作为 CocoaPod 使用。它被称为PersistentStreamPlayer

功能包括:

  • 流式传输音频文件,第一个数据可用后立即开始播放
  • 还会在缓冲区完成后立即将流数据保存到文件 URL 公开 timeBuffered,有助于在 UI
  • 中显示缓冲区进度条
  • 处理在缓冲流停顿(例如网络速度慢)后重新启动音频文件
  • 简单的播放、暂停和销毁方法(销毁清除所有内存资源)
  • 不会将音频文件数据保存在内存中,因此它支持无法放入 RAM 的大文件

您可以在这里找到它:https://github.com/calmcom/PersistentStreamPlayer