如何在 Swift 中使扩展中的计时器无效

问题描述

我试图在扩展中添加计时器属性代码如下

 @objc public extension RCTTouchHandler {
  static let kSessionTimer = "sessionTimer"
  
   var sessionTimer: Timer {
    get {
      return objc_getAssociatedobject(self,RCTTouchHandler.kSessionTimer) as? Timer ?? Timer.scheduledTimer(timeInterval: RCTTouchHandler.kSessionExpiredDuration,target: self,selector: #selector(touchSessionExpiration),userInfo: nil,repeats: false)
    }

    set(timer) {
      objc_setAssociatedobject(self,RCTTouchHandler.kSessionTimer,self,.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
    }
  }

  @objc func touchSessionExpiration() {
    print("session expired")
  }

  deinit {
    self.sessionTimer.invalidate()
  }
}

那么,sessionTimer对这个对象有强引用,同时这个对象强保留sessionTimer。为了避免保留周期和内存泄漏,我应该在对象被取消初始化时使计时器无效。但是,在尝试使此扩展中的 deinit 函数中的计时器无效时,我遇到了以下错误

Deinitializers may only be declared within a class 

根据Swift Doc

扩展可以向类添加新的便利初始化器,但不能向类添加新的指定初始化器或析构器。指定的初始化器和析构器必须始终由原始类实现提供。

所以我想知道如何在此扩展中正确地使计时器无效?

解决方法

如您所说,您不能在扩展中实现 deinit { }


您可以创建自己的对象并在那里实现 deinit { },而不是在任何地方使用 RCTTouchHandler,而是使用您自己的对象:

@objc extension RCTTouchHandler {
    static let kSessionTimer = "sessionTimer"
}

public class MyTouchHandler: RCTTouchHandler {
    private var timer: Timer? = nil

    var sessionTimer: Timer {
        get {
            guard let timer = timer else {
                let newTimer = Timer.scheduledTimer(timeInterval: RCTTouchHandler.kSessionExpiredDuration,target: self,selector: #selector(touchSessionExpiration),userInfo: nil,repeats: false)
                self.timer = newTimer
                return newTimer
            }
            return timer
        }

        set(timer) {
            self.timer = timer
        }
    }

    @objc func touchSessionExpiration() {
        print("session expired")
    }

    deinit {
        self.timer?.invalidate()
    }
}

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...