Swift3.2升级4.0

一, Swift 版本

1, Swift Language ---> 4.0

2, 修改pod库

第三方库暂时使用3.2版本

post_install do |installer|
  installer.pods_project.targets.each do |target|
    target.build_configurations.each do |config|
      config.build_settings['SWIFT_VERSION'] = '3.2'
    end 
  end
end

二, 代码的修改

1,Swfit4.0 对于扩展的属性(包括实例属性,static属性,class属性)只能使用get方法,不能使用set方法

2,Swift4.0中不允许重写扩展中的方法,包括实例方法,static方法,class方法
比如:自定义的协议方法在extension中实现,若某个类遵循了协议,其子类就不能重写该协议的方法
解决方法:在每个需要该协议的类里面都重新遵循该协议,实现协议方法

3,Swift3 使用#selector 指定方法,只有当方法权限为 private时需要加 @objc修饰符,现在swift4.0全部都要加@objc修饰符

4,自定义protocol协议中,有optional修饰的非必须实现方法,需要用@objc 修饰

5, 字体方面的一些重命名

NSFontAttributeName --- .font 
//或者NSAttributedStringKey.font

NSForegroundColorAttributeName --- .foregroundColor
//NSAttributedStringKey.foregroundColor

NSStrikethroughStyleAttributeName --- .strikethroughStyle
//NSAttributedStringKey.strikethroughStyle

//字符串类型的,添加rawValue
NSAttributedStringKey.font.rawValue

//等等等等..........

//大部分类似以下,涉及富文本的方法均已改为了NSAttributedStringKey类型
addAttributes(_ attrs: [NSAttributedStringKey : Any] = [:], range: NSRange)

三,其他报错

1, Closure cannot implicitly capture a mutating self parameter"错误
在struct中,如果我们在闭包中使用self,就会得到此错误
比如:

struct RecordModel {
    /// 定义一个闭包
    var action: (() -> ())?
    var height = 10
    
    self.action = { 
        self.height = 20 
        //Closure cannot implicitly capture a mutating self parameter报错
    }
}

解决方案1:
为closure增加一个inout类型的参数

struct RecordModel {
    /// 定义一个闭包
    var action: ((_ inSelf: inout RecordModel) -> ())?
    var height = 10
    
    self.action = { (inSelf) in
        inSelf.height = 20 
    }
}

根据inout类型的说明,我们知道,实际上这相当于增加了一个隐藏的临时变量,self被复制,然后在closure(闭包)中使用,完成后,再复制回self。也就是说,这个方法有额外的内存开销。如果是struct较大的情形,这么做并不划算

解决方案1:使用UnsafeMutablePointer<Pointee>
这次采用直接指针的方式对于struct来进行操作,采用指针的好处是self不会被多次复制,性能较高。缺点是你需要自行确定你的代码的安全

struct RecordModel {
    /// 定义一个闭包
    var action: (() -> ())?
    var height = 10
    
    let selfPointer = UnsafeMutablePointer(&self)
    self.action = { 
        selfPointer.pointee.height = 20
    }
}

Closure cannot implicitly capture a mutating self parameter错误的原因是在进出closure(闭包)之后,self的一致性没办法得到保证,所以编译器默认不允许在struct的closure(闭包)中使用self。如果我们确定这么做是安全的,就可以通过上面的两种方式解决这个问题。其中,方法二的性能更好一些。

这里可以记一下指针和swift变量之间的关系:

  • UnsafePointer对应let
  • UnsafeMutablePointer对应var
  • AutoreleasingUnsafeMutablePointer对应unowned UnsafeMutablePointer,用于inout的参数类型
  • UnsafeRawPointer对应let Any,raw系列都是对应相应的Any类型
  • UnsafeBufferPointer是non-owning的类型(unowned),用于collection的elements, buffer系列均如此

2, Declarations from extensions cannot be overridden yet 错误
这个错误大致是因为,协议方法是在extension里面的,不能被重写
就是在每一个需要此协议的类里面,重新遵循代理,实现该协议方法

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() {//此处报错
    }
}

替代方法:

  • 在 app delegate 中实现方法交叉
  • 像上面通过类扩展进行方法交叉,而是简单地在 app delegate 的 application(_:didFinishLaunchingWithOptions:) 方法调用时调用该方法
extension UIViewController {
    public override class func initializeOnceMethod() {
    }
}

//在AppDelegate的方法中调用:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey : Any]? = nil) -> Bool {
    UIViewController.initializeOnceMethod()
}

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!" )  
}

另外--->

// NSTextCheckingResult
func rangeAt(_: Int)   --> func range(at: Int)

String的警告
'characters' is deprecated: Please use String or Substring
characters 过期

let string = "abc"
var count = string.characters.count

// 第二行报错
'characters' is deprecated: Please use String or Substring directly

// 对应新方法
count = string.count

warning
Conditional downcast from 'NSData?' to 'Data' is a bridging conversion; did you mean to use 'as'?

// data = NSData(contentsOfFile: path!) as? Data
   data = NSData(contentsOfFile: path!) as Data?

waring
'M_PI' is deprecated: Please use 'Double.pi' or '.pi' to get the value of correct type and avoid

M_PI  --->  Double.pi
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容