如何使用AVURLAssetstream式传输video并将caching的数据保存到磁盘

前几天我被要求检查一下从网上下载video有多困难。 我知道这是一件容易的事,因为前一阵子有人告诉我。 所以,我检查,这是超级简单。

问题是,我想保存到磁盘的video,不要强迫用户下载一次又一次。

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

在Stackoverflow许多答案说,这也是不可能的。 特别与video。

我的原始代码播放video:

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() } 

此问题的解决scheme是使用AVAssetExportSessionAVAssetResourceLoaderDelegate

第一步是添加一个通知,以了解video完成时间。 然后我们可以开始将它保存到磁盘。

 override func viewDidLoad() { super.viewDidLoad() NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(playerItemDidReachEnd(_:)), name: AVPlayerItemDidPlayToEndTimeNotification, object: nil) ... } deinit { NSNotificationCenter.defaultCenter().removeObserver(self) } 

我们的function的实现:

 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委托:

 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中用这段代码创build了一个小的演示。

Calm的团队已经开放了我们的实施。 它可以作为一个cocoa豆。 它被称为PersistentStreamPlayer

function包括:

  • audio文件的stream式传输,只要第一个数据可用,就开始播放
  • 一旦缓冲区完成时,也将stream数据保存到文件URL公开timeBuffered,有助于在UI中显示缓冲区进度条
  • 在缓冲器stream停止之后处理重新启动audio文件(例如,慢速networking)
  • 简单的播放,暂停和销毁方法(销毁清除所有内存资源)
  • 不将audio文件数据保存在内存中,以便支持不适合RAM的大文件

你可以在这里find它: https : //github.com/calmcom/PersistentStreamPlayer