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

11

前几天我被要求检查一下在下载视频的同时播放视频有多难。我知道这是一项简单的任务,因为之前有人告诉过我。所以,我检查了一下,结果发现非常简单。

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

问题在于如何访问缓冲区并将其存储到磁盘中。

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()
}
2个回答

25

解决这个问题的方法是使用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 {

}

我在GitHub上用这段代码创建了一个小演示。


2
为什么这里需要AVAssetResourceLoaderDelegate?我没有看到其中任何一个被使用:https://developer.apple.com/reference/avfoundation/avassetresourceloaderdelegate - Lane Rettig
5
当我尝试运行这段代码时,出现以下错误:Error Domain=AVFoundationErrorDomain Code=-11838 "Operation Stopped" UserInfo={NSUnderlyingError=0x61800024ebe0 {Error Domain=NSOSStatusErrorDomain Code=-12109 "(null)"}, NSLocalizedFailureReason=该媒体不支持此操作, NSLocalizedDescription=操作停止}。我确认了outputURLoutputFileType是正确的。 - Lane Rettig
运行良好。我们在逐个播放url时需要注意哪些预防措施? - KavyaKavita
这看起来很不错,但我尝试了演示项目,它没有保存,给了我错误提示。可选(Error Domain=AVFoundationErrorDomain Code=-11823 "无法保存")。 - user3069232
1
请问一下,这是否将从缓存中导出视频文件?由于我们是通过新资源进行下载,并且无法理解它如何获取当前播放器项缓冲区,所以只是好奇。 - GoodSp33d
显示剩余5条评论

3
Calm团队已经将我们的实现开源了,作为一个CocoaPod可用。它被称为PersistentStreamPlayer。
其功能包括:
- 流式传输音频文件,一旦有数据就开始播放 - 也会将流式传输的数据保存到文件URL中,一旦缓冲完成 - 提供timeBuffered,有助于在UI中显示缓冲进度条 - 处理缓冲流卡顿(例如慢速网络)后重新启动音频文件 - 简单的播放、暂停和销毁方法(销毁清除所有内存资源) - 不会在内存中保留音频文件数据,因此支持不适合RAM的大型文件
你可以在这里找到它:https://github.com/calmcom/PersistentStreamPlayer

1
它看起来与这段代码非常相似:https://gist.github.com/anonymous/83a93746d1ea52e9d23f 我遇到的问题是,当使用AVAssetResourceLoaderDelegate时,readyToPlay事件仅在整个文件下载完成后发生,而以前只要接收到足够的数据以播放视频开头,就会发生。 - Guig
是的,它非常类似于那段代码,因为我从那里作为分支点开始。不过,我还添加了几个很好的功能。 - Tyler Sheaffer
关于 AVAssetResourceLoaderDelegate,@Guig 注意到你的 PersistentStreamPlayer 实例正在实现该协议,所以我不确定你是如何连接以便监听它的。如果你想在仓库上开一个问题,我很乐意与你一起解决这个问题,或者随时欢迎在那里开一个 Pull Request。 - Tyler Sheaffer
@Frade 理论上应该可以,但我只在生产环境中测试过音频。如果它不完全符合您的用例,那么分叉并提交请求以使视频支持工作应该比从头开始容易得多。干杯。 - Tyler Sheaffer
请注意:这段代码尚未完成。如果在文件仍在下载时进行搜索,它无法正确保存文件。 - Greg Ellis
显示剩余5条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接