3

我正在编写一个非常简单的应用程序,它有两个视图控制器 - FirstVC 和 SecondVC。在 FirstVC 上,我加载了一个视频(apples.mp4),它在(我的视图)背景中永远循环播放。视频加载由从 FirstVC 的 viewWillAppear() 函数执行的 displayBackgroundVideo() 函数完成。

FirstVC 的代码如下:

override func viewWillAppear(_ animated: Bool) {
    displayBackgroundVideo()
}

func displayBackgroundVideo() {  
    var videoPlayer : AVPlayer?
    var videoPlayerLayer : AVPlayerLayer?

    let bundlePath = Bundle.main.path(forResource: "apples", ofType: "mp4")
    guard bundlePath != nil else { return }

    let url = URL(fileURLWithPath: bundlePath!)
    let item = AVPlayerItem(url: url)

    videoPlayer = AVPlayer(playerItem: item)

    videoPlayerLayer = AVPlayerLayer(player: videoPlayer!)

    videoPlayerLayer?.frame = CGRect(x: -self.view.frame.size.width*2, y: 0, width: self.view.frame.size.width*5, height: self.view.frame.size.height)

    view.layer.insertSublayer(videoPlayerLayer!, at: 0)

    videoPlayer?.playImmediately(atRate: 0.6)

    // Loop video forever
    NotificationCenter.default.addObserver(forName: .AVPlayerItemDidPlayToEndTime, object: videoPlayer?.currentItem, queue: .main)
    { _ in
        videoPlayer?.seek(to: CMTime.zero)
        videoPlayer?.playImmediately(atRate: 0.6)
    }
}

我在 FirstVC 上还有一个按钮,如果你按下它,就会转到 SecondVC。类似地,在 SecondVC 中,我有一个简单的按钮,可以返回到 FirstVC。SecondVC 中没有其他内容。

问题:如果我不断在 FirstVC -> SecondVC -> FirstVC -> SecondVC ->... 之间来回切换,我注意到内存不断增加!但是,如果我在 FirstVC 中注释掉 displayBackgroundVideo() 函数(即不加载视频),那么我会得到预期的来回行为会增加和减少内存(因为我们正在 segueing 和 unwinding成功返回)。

那么,如果我显示视频,为什么我的记忆会爆炸(增加)?我怎么解决这个问题?

4

1 回答 1

1

我相信您在那里造成了内存泄漏

addObserver(forName:object:queue:using:)实际上返回一个不透明的对象来充当观察者,并且通过保留该引用,您可以在 deinit 中使用它来在释放 VC 时实际删除该观察者。

考虑到这一点,我建议更新您的代码以在 viewDidLoad 而不是 viewWillAppear 中移动设置,并添加 deinit 以删除观察者:

class FirstVC: UIViewController {

    private var backgroundVideoPlayerObj: NSObjectProtocol?
    
    override func viewDidLoad(_ animated: Bool) {
        super.viewDidLoad(animated)
        displayBackgroundVideo()
    }

    deinit {            
        if let backgroundVideoPlayerObj = backgroundVideoPlayerObj {
            NotificationCenter.default.removeObserver(backgroundVideoPlayerObj, name: .AVPlayerItemDidPlayToEndTime, object: nil)
        }
    }

    func displayBackgroundVideo() { 
        var videoPlayer : AVPlayer?
        var videoPlayerLayer : AVPlayerLayer?

        ...

        backgroundVideoPlayerObj = NotificationCenter.default.addObserver(forName: .AVPlayerItemDidPlayToEndTime, object: videoPlayer?.currentItem, queue: .main) { _ in
            videoPlayer?.seek(to: CMTime.zero)
            videoPlayer?.playImmediately(atRate: 0.6)
        }
    }

}
于 2021-08-03T22:01:13.763 回答