我正在读取带有内容的视频数据,AVAssetReader
并通过再次读取内容并修改时间戳来“循环”内容。相关代码为:
private func readFrame() -> CMSampleBuffer? {
let buffer = output?.copyNextSampleBuffer()
guard buffer != nil else {
do {
reader?.cancelReading()
try loop()
} catch let e {
return nil
}
return readFrame()
}
// if we've looped, add previous loop times
// MEMORY LEAK IS IN HERE
if CMTimeCompare(offsetTime, .zero) > 0 {
var newTiming = [CMSampleTimingInfo(
duration: CMSampleBufferGetDuration(buffer!),
presentationTimeStamp: CMTimeAdd(CMSampleBufferGetOutputPresentationTimeStamp(buffer!), offsetTime),
decodeTimeStamp: CMTimeAdd(CMSampleBufferGetOutputDecodeTimeStamp(buffer!), offsetTime)
)]
CMSampleBufferCreateCopyWithNewTiming(allocator: kCFAllocatorDefault, sampleBuffer: buffer!, sampleTimingEntryCount: 1, sampleTimingArray: &newTiming, sampleBufferOut: &loopedBuffer)
return loopedBuffer
}
return buffer
}
private func loop() throws {
guard (reader?.status == .some(.completed)) else {
throw FileError.cannotLoop(reader?.error)
}
offsetTime = CMTimeAdd(offsetTime, assetDuration)
try createReader() // creates a new reader and calls reader.startReading()
}
一旦视频完成其第一个循环(即一次offsetTime
大于零),内存就会开始爆炸。注释掉创建具有更新时序的新缓冲区的代码可以修复内存泄漏(但这不可行,因为我需要更新的时序)。
大概CMSampleBufferCreateCopyWithNewTiming
是为每个缓冲区分配新空间,并且该缓冲区永远不会被清理。我已经阅读了autoreleasepool
一个潜在的选项,但是这个缓冲区被传递了很多(它通过一个自定义的发布者)。有没有办法消除内存泄漏?我可以为 1 个缓冲区分配空间并在每一帧上一次又一次地覆盖它 - 这可能吗?