为什么AVPlayer首先下载而不是直播流?

12

这是我第一次使用AVPlayer,我希望使用HTTP请求进行直播流播放mp3文件。我使用的AVPlayer可以很好地播放mp3文件,但问题在于AVPlayer需要先下载mp3文件才能播放它。我不知道为什么AVPlayer需要下载文件然后再播放,而不是直接实时播放。我的Xcode版本是8.2.1,我正在使用Swift 3。

这是我的代码片段。

var audioPlayer = AVPlayer()
var avplayerItem : AVPlayerItem?

override func viewDidLoad() {
    super.viewDidLoad()

    let fileurl:URL = URL(string : "http://www.noiseaddicts.com/samples_1w72b820/2514.mp3")!
    avplayerItem = AVPlayerItem(url : fileurl)
    audioPlayer = AVPlayer(playerItem : avplayerItem)
    audioPlayer.rate = 1.0
    audioPlayer.play()
}

如果AVPlayer需要在播放之前下载整个文件,那么对于音频直播流,我该怎么办?
谢谢。

1
直播需要服务器支持,至少要有部分下载的能力。其他播放器是否允许使用该URL进行直播? - Luaan
你怎么知道 AVPlayer 正在下载整个文件? - Rhythmic Fistman
3
首先,游戏耗时太长,在此期间,玩家当前的物品和当前时间都是0。当它开始播放时,如果我关闭了互联网连接,它仍然会一直播放到结束。 - Shariif Islam
这里也有同样的问题,有更新吗? - johnny
2个回答

1

你基本上是使用mp3作为容器在互联网上传输音频,因此播放器必须全部下载它,如果你想要高效或避免下载延迟,你必须使用HLSDASHCMAFWebRTC来传输它,这样播放器将通过块下载它,并在下载时播放,类似于Spotify或Apple Music的做法。


0
关于编辑额外的内容: Luaan 也是对的。"直播需要服务器支持-至少需要能够进行部分下载。有没有其他播放器可以使用那个URL进行直播?" 你也应该尝试使用一个未定义时长的音频URL。
你是否尝试过将其视为一个未定义时长的音频流来处理?
  avplayerItem = CachingPlayerItem(url: url, recordingName: recordingName ?? "default.mp3")

CachingPlayerItem将从URL准备一个AVURLAsset。

 init(url: URL, customFileExtension: String?, recordingName: String) {
    guard let components = URLComponents(url: url, resolvingAgainstBaseURL: false),
    let scheme = components.scheme,
    var urlWithCustomScheme = url.withScheme(cachingPlayerItemScheme) else {
    fatalError("Urls without a scheme are not supported")
    }
    self.recordingName = recordingName
    self.url = url
    self.initialScheme = scheme
    if let ext = customFileExtension {
    urlWithCustomScheme.deletePathExtension()
    urlWithCustomScheme.appendPathExtension(ext)
    self.customFileExtension = ext
    }
    let asset = AVURLAsset(url: urlWithCustomScheme)
    asset.resourceLoader.setDelegate(resourceLoaderDelegate, queue: DispatchQueue.main)
    super.init(asset: asset, automaticallyLoadedAssetKeys: nil)
    resourceLoaderDelegate.owner = self
    addObserver(self, forKeyPath: "status", options: NSKeyValueObservingOptions.new, context: nil)
    NotificationCenter.default.addObserver(self, selector: #selector(playbackStalledHandler), name:NSNotification.Name.AVPlayerItemPlaybackStalled, object: self)
    }

在初始化AVPlayerItem之后,从中创建一个AVPlayer对象,以便启动从给定URL加载音频流的过程:

player = AVPlayer(playerItem: playerItem)
player.automaticallyWaitsToMinimizeStalling = false

func resourceLoader(_ resourceLoader: AVAssetResourceLoader, shouldWaitForLoadingOfRequestedResource loadingRequest: AVAssetResourceLoadingRequest) -> Bool {
if playingFromData {
// Nothing to load.
} else if session == nil {
// If we're playing from a url, we need to download the file.
// We start loading the file on first request only.
guard let initialUrl = owner?.url else {
fatalError("internal inconsistency")
}
startDataRequest(with: initialUrl)
}
pendingRequests.insert(loadingRequest)
processPendingRequests()
return true
}

资源安装完成后:

func startDataRequest(with url: URL) {
var recordingName = "default.mp3"
if let recording = owner?.recordingName{
recordingName = recording
}
fileURL = try! FileManager.default.url(for: .documentDirectory, in: .userDomainMask, appropriateFor: nil, create: false)
.appendingPathComponent(recordingName)
let configuration = URLSessionConfiguration.default
configuration.requestCachePolicy = .reloadIgnoringLocalAndRemoteCacheData
session = URLSession(configuration: configuration, delegate: self, delegateQueue: nil)
session?.dataTask(with: url).resume()
outputStream = OutputStream(url: fileURL, append: true)
outputStream?.schedule(in: RunLoop.current, forMode: RunLoopMode.defaultRunLoopMode)
outputStream?.open()
}

然后,开始将数据字节接收到委托函数中:

func urlSession(_ session: URLSession, dataTask: URLSessionDataTask, didReceive data: Data)

现在您已经开始接收实时音频流。

func urlSession(_ session: URLSession, dataTask: URLSessionDataTask, didReceive data: Data) {
let bytesWritten = data.withUnsafeBytes{outputStream?.write($0, maxLength: data.count)}
print("bytes written :\(bytesWritten!) to \(fileURL)")
}

现在创建一个OutputStream对象,打开它,然后将我们在上面的委托函数中接收到的字节追加进去,就完成了,我们保存了实时音频流的所需部分。
摘自:Medium文章,Mohan Pandey

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