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

Gab*_*ana 6 avfoundation ios avplayer avurlasset swift

几天前,我被要求检查从互联网下载视频时播放视频有多困难.我知道这是一件容易的事,因为不久前有人告诉过我.所以,我检查过,这非常简单.

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

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

Stackoverflow中的许多答案都说它也不可能.特别是视频.

我播放视频的原始代码:

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()
}
Run Code Online (Sandbox Code Playgroud)

Gab*_*ana 19

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

第一步是添加通知以了解视频何时完成.然后我们可以开始将其保存到磁盘.

override func viewDidLoad() {

    super.viewDidLoad()

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

    ...
}

deinit {

    NSNotificationCenter.defaultCenter().removeObserver(self)
}
Run Code Online (Sandbox Code Playgroud)

我们的功能实现:

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)
        })
    }
}
Run Code Online (Sandbox Code Playgroud)

最后,我们需要创建AVURLAssetAVAssetResourceLoaderDelegate 的委托:

lazy var asset: AVURLAsset = {

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

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

    return asset
}()
Run Code Online (Sandbox Code Playgroud)

和:

extension ViewController : AVAssetResourceLoaderDelegate {

}
Run Code Online (Sandbox Code Playgroud)

我在GitHub中使用此代码创建了一个小型演示.

  • 当我尝试运行此代码时出现以下错误:`Error Domain = AVFoundationErrorDomain Code = -11838"Operation Stopped"UserInfo = {NSUnderlyingError = 0x61800024ebe0 {Error Domain = NSOSStatusErrorDomain Code = -12109"(null)"},NSLocalizedFailureReason =此媒体不支持该操作.,NSLocalizedDescription = Operation Stopped}`.我确保`outputURL`和`outputFileType`是正确的. (4认同)
  • 为什么这里需要AVAssetResourceLoaderDelegate?我看不到使用这些任何东西:https://developer.apple.com/reference/avfoundation/avassetresourceloaderdelegate (2认同)