iOS学习

Swift 4 迁移踩坑

2017-10-12  本文已影响152人  切一斤桃花卖酒钱
文档
  1. private 属性作用域扩大到 extension , 防止 fileprivate 的滥用

  2. 可以把类型和协议用 & 组合在一起作为一个类型使用,优化代码结构

  3. Associated Type 可以追加 Where 约束语句

  4. keyPath:

    Swift 3

    let barKeyPath = #keyPath(Foo.bar)
    var bar = foo.value(forKeyPath: barKeyPath)
    foo.setValue("BAR", forKeyPath: barKeyPath)
    
    

    Swift 4

    let barKeyPath = \Foo.bar
     var bar = foo[keyPath: barKeyPath]
     foo[keyPath: barKeyPath] = "BAR"
    

    swift 4: 更安全,控制 value 的 anyObject 类型,并且不需要类继承 NSObject

  1. 下标支持泛型

  2. 以及一些系统优化: unicode count 计算,字符串处理速度

  3. String.Charactor.startIndex -> String.startIndex 去掉 charactor 中间属性,直接取 string 的属性

  4. String 当做 Collection 来用(即 String 拥有 Collection 的属性) eg: string.reversed() string.filter string.map …

  5. 多行字符串字面量,省去繁琐换行操作

     let joke = """
        Q: Why does \(name) have \(n) \(character)'s in their name?
        A: I don't know, why does \(name) have \(n) \(character)'s in their name?
        Q: Because otherwise they'd be called \(punchline).
        """
    
  6. Encoding and Decoding

    让归档更简单只能, 继承 Codeable 协议的对象都可 实现 let encoded = try? JSONEncoder().encode(swift) 将 对象归档为Data,相反解档


看完文档,正式开始

迁移顺序


Error
  1. 报错 Ambiguous reference to member '+' + 号表达不明确,将表达式拆分

  2. accesses to 0x7f887f904430, but modification requires exclusive access 可能会有多个原因,如果是 KVO 报此错误,是因为 Swift 4 使用了新的 KVO 方法。 参照 Swift 新的 KVO 方法
    或: github 实例

    // MARK: - Property
    
    fileprivate var _observation: NSKeyValueObservation?
    
    
    //MARK: - Lifecycle
    
    _observation = LCKXMPPActivity.sharedInstance.observe(\.status,   options: [.initial, .old]) { (child, change) in
         //
     }
    

statusLCKXMPPActivity 中声明为 @objc dynamic 的属性

@objc open dynamic var status: LCKXMPPStatus

遇到一个很很很奇怪的问题, _observation 对象如果不声明为全局的话, observe 方法并不会被 status 的的改变而触发

  1. 属性字符串 NSAttriburedString
Swift 3

   let attributes = [NSStrikethroughStyleAttributeName : NSUnderlineStyle.styleSingle.rawValue,
   NSForegroundColorAttributeName: UIColor.kTitleColor4] as [String : Any]

Swift 4

   let attributes: [NSAttributedStringKey: Any] =  [NSAttributedStringKey.strikethroughStyle: NSUnderlineStyle.styleSingle.rawValue,
   NSAttributedStringKey.foregroundColor: UIColor.kTitleColor4]     

attributedStringkey 统一通过 NSAttributedStringKey 调用,并且 NSAttributedString(string: <#T##String#>, attributes: <#T##[NSAttributedStringKey : Any]?#>) 接受的类型是 [NSAttributedStringKey : Any] 类型

warning
  1. swap(&a, &b) --> swapAt(Index:Int, Index: Int)

  2. init(colorLiteralRed:green:blue:alpha:) --> init(red:green:blue:alpha:)

  3. redundant conformance of XXX to protocol YYY 协议声明重复,查看当前所继承的协议是否有重复,或者查看子类是否又继承了一遍

  4. The use of Swift 3 @objc inference in Swift 4 mode is deprecated.
    首先在设置里如图所示:

    inference.png
    Swift 4 中不再默认[推断]为 OC 的方法加上 @objc 属性,所以需要 swift 调用 OC 的地方手动加上 objc.
    参见资料: stackoverflow evgenii
  5. Multiple Closures with Trailing Closure Violation: Trailing closure syntax should not be used when passing more than one closure argument. (multiple_closures_with_trailing_closure)

    warning: 
    
    ObjectiveManager.shared.finishObjective(objectiveID: "2", keyResults: [], success: {
          //
        }) { _ in
          //
        }
    
    success:
    
    ObjectiveManager.shared.finishObjective(objectiveID: "2", keyResults: [], success: {
      //
    }, failure: { _ in
      //
    })
    

    尾随闭包的问题,参见资料: Apple

  6. Declarations in extensions cannot override yet
    override 用户添加新方法,但不能修改原方法,所以讲 override 的方法不能放在 extension 中. 父方法前加 @objc

  7. This block declaration is not a prototype
    修改 strict prototypes set no 或者 OC 的 block 一个一个慢慢改吧,这里是混编 OC 的问题。
    参照资料 stackoverflow

  8. 当 OC 调用 swift 时报错 No known class method for selector 所调用的方法前加 @objc

  9. 'substring(from:)' is deprecated: Please use String slicing subscript with a 'partial range from' operator.
    Stringsubstring属性在 Swift 4 新的实现

    let newStr = str.substring(to: index) -->  let newStr = String(str[..<index])
    

如果是某一范围 str[..<str.index(str.startIndex, offsetBy: 8)]
参照资料: stackoverflow

  1. tableview height for header & footer 不调用, 因为estimatedSectionHeaderHeight 在 iOS 11 不再有默认值。解决方案: 设置 tableview.estimatedSectionHeaderHeight = 0 或者
    实现 viewForFooterInSection 代理返回 nil, footer 同理

有些问题可能忘写了,以及不清楚的地方请留言讨论

持续更新,可以收藏关注先。。。

下一篇 iOS 11 以及 iPhone X 适配(11 已完成, X 还在挣扎中)

最后

写这篇博客的时候,并没有女朋友喂我水果吃 = =

上一篇下一篇

猜你喜欢

热点阅读