问题描述
我正在使用 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 个缓冲区分配空间并在每一帧上一次又一次地覆盖它 - 这可能吗?
解决方法
暂无找到可以解决该程序问题的有效方法,小编努力寻找整理中!
如果你已经找到好的解决方法,欢迎将解决方案带上本链接一起发送给小编。
小编邮箱:dio#foxmail.com (将#修改为@)