升级到Swift 4.0可能遇到的坑总结

2020-01-08 23:47:50于丽

3-3. "Method 'initialize()' defines Objective-C class method 'initialize', which is not permitted by Swift"

==报错原因: 在于已经废弃的initialize方法,示例如下==

方法交叉(Method Swizzling)

有时为了方便,也有可能是解决某些框架内的 bug,或者别无他法时,需要修改一个已经存在类的方法的行为。方法交叉可以让你交换两个方法的实现,相当于是用你写的方法来重载原有方法,并且还能够是原有方法的行为保持不变。


extension UIViewController {
 public override class func initialize() {//此处报错
  
 //此处省略100行代码
  
 }
}

initialize该方法已经被Swift4.0废弃

在Swift3.0还勉强可以使用,但是会有警告;但是在4.0已经被完全废弃

==替代方法:==

在 app delegate 中实现方法交叉

像上面通过类扩展进行方法交叉,而是简单地在 app delegate 的 application(_:didFinishLaunchingWithOptions:) 方法调用时调用该方法


extension UIViewController {
 public override class func initializeOnceMethod() {
  
 //此处省略100行代码
  
 }
}

//在AppDelegate的方法中调用:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey : Any]? = nil) -> Bool {
 //此处省略100行代码
 UIViewController.initializeOnceMethod()

}

3-4. 'dispatch_once' is unavailable in Swift: Use lazily initialized globals instead

报错原因: dispatch_once在Swift4.0也已经被废弃


extension UITableView {
 struct once{
  static var onceTaken:Int = 0
 }
 dispatch_once(&once.onceTaken) { () -> Void in
 //在这里dispatch_once就会报错
  //此处省略1000000行代码 
 }
}

解决方法: 通过给DispatchQueue添加扩展实现


extension DispatchQueue {
 private static var _onceTracker = [String]()
 public class func once(token: String, block: () -> ()) {
  objc_sync_enter(self)
  defer {
   objc_sync_exit(self)
  }
  if _onceTracker.contains(token) {
   return
  }
  _onceTracker.append(token)
  block()
 }
 
 func async(block: @escaping ()->()) {
  self.async(execute: block)
 }
 
 func after(time: DispatchTime, block: @escaping ()->()) {
  self.asyncAfter(deadline: time, execute: block)
 }
}

使用字符串token作为once的ID,执行once的时候加了一个锁,避免多线程下的token判断不准确的问题。
使用的时候可以传token


 DispatchQueue.once(token: "tableViewOnce") {
  print( "Do This Once!" ) 
 }

或者使用UUID也可以:


private let _onceToken = NSUUID().uuidString
 
DispatchQueue.once(token: _onceToken) { 
 print( "Do This Once!" ) 
}