NSTimer循环引用解决方案详解

一、循环引用原理

NSTimer 会强引用它的 target。如果 target(如 ViewController)又直接或间接强引用了 timer,会导致循环引用,内存无法释放。

swift 复制代码
class ViewController: UIViewController {
    var timer: Timer?
    
    override func viewDidLoad() {
        super.viewDidLoad()
        // Timer 强引用 self,self 强引用 Timer → 循环引用
        timer = Timer.scheduledTimer(
            timeInterval: 1,
            target: self,
            selector: #selector(handleTimer),
            userInfo: nil,
            repeats: true
        )
    }
    
    @objc func handleTimer() {
        print("Timer fired")
    }
    
    deinit {
        timer?.invalidate()
        print("Deinit") // 不会执行
    }
}

二、解决方案

1. 使用中间代理对象(Weak Proxy)

创建一个弱引用代理对象,打破 TimerViewController 之间的强引用。

swift 复制代码
class WeakProxy: NSObject {
    weak var target: NSObjectProtocol?
    
    init(target: NSObjectProtocol) {
        self.target = target
        super.init()
    }
    
    // 将方法转发给真正的 target
    override func forwardingTarget(for aSelector: Selector!) -> Any? {
        return target
    }
}

// 在 ViewController 中使用
class ViewController: UIViewController {
    var timer: Timer?
    
    override func viewDidLoad() {
        super.viewDidLoad()
        // Timer 强引用 WeakProxy,WeakProxy 弱引用 self
        timer = Timer.scheduledTimer(
            timeInterval: 1,
            target: WeakProxy(target: self),
            selector: #selector(handleTimer),
            userInfo: nil,
            repeats: true
        )
    }
    
    @objc func handleTimer() {
        print("Timer fired")
    }
    
    deinit {
        timer?.invalidate()
        print("Deinit") // 正常执行
    }
}

2. 使用 Block-Based API(iOS 10+)

利用 scheduledTimer(withTimeInterval:repeats:block:),通过 weak self 避免循环。

swift 复制代码
class ViewController: UIViewController {
    var timer: Timer?
    
    override func viewDidLoad() {
        super.viewDidLoad()
        // Block 内捕获 weak self,避免循环
        timer = Timer.scheduledTimer(
            withTimeInterval: 1,
            repeats: true
        ) { [weak self] _ in
            self?.handleTimer()
        }
    }
    
    func handleTimer() {
        print("Timer fired")
    }
    
    deinit {
        timer?.invalidate()
        print("Deinit") // 正常执行
    }
}

3. 手动销毁 Timer

viewWillDisappeardeinit 中手动销毁 Timer

swift 复制代码
class ViewController: UIViewController {
    var timer: Timer?
    
    override func viewDidLoad() {
        super.viewDidLoad()
        timer = Timer.scheduledTimer(
            timeInterval: 1,
            target: self,
            selector: #selector(handleTimer),
            userInfo: nil,
            repeats: true
        )
    }
    
    @objc func handleTimer() {
        print("Timer fired")
    }
    
    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        timer?.invalidate()
    }
    
    deinit {
        print("Deinit") // 正常执行
    }
}

三、总结

方案 核心思想 优点 缺点
Weak Proxy 中间对象弱引用 target 兼容旧版本 需额外代理类
Block-Based API 利用 weak self 捕获弱引用 代码简洁(iOS 10+) 仅支持 iOS 10+
手动销毁 Timer 在适当时机调用 invalidate() 简单直接 需确保销毁逻辑被执行

选择方案时,优先推荐 Block-Based API (iOS 10+)或 Weak Proxy(兼容旧版本)。

相关推荐
2501_9437823511 分钟前
【共创季稿事节】猜数字游戏:二分法思维与交互式反馈
前端·游戏·microsoft·harmonyos·鸿蒙·鸿蒙系统
GV191rLvq25 分钟前
基于Socket实现的最简单的Web服务器【ASP.NET原理分析】
服务器·前端·asp.net
吠品31 分钟前
LangChain 里 tool_call_id 为空?一次 MCP 工具集成的排查记录
前端
柒和远方1 小时前
Phase 7.4 学习博客:为什么多 API 项目需要 Swagger / OpenAPI
前端·后端·架构
张龙6871 小时前
拼多多开放平台对接踩坑实录:从 CLIENT_ID 配置到 MD5 签名算法的完整填坑指南
前端
GuWenyue1 小时前
提示词彻底过时?一套上下文工程方案,3步让LLM落地生产,代码直接复用
前端·javascript·人工智能
柒和远方1 小时前
Phase 7.3 复盘:后台任务不只是“扔进队列”,还要能被看见
前端·后端·架构
2501_943782351 小时前
【共创季稿事节】 倒计时器:时分秒选择器与定时器的协同工作
前端·华为·harmonyos·鸿蒙·鸿蒙系统
奶油mm1 小时前
公司技术债堆积如山,我一人之力用 Vue3 偷换了整个前端架构
前端·vue.js
用户938515635071 小时前
深入理解 JavaScript 中的 this 与数据存储的奥秘
前端·javascript