Swift2转Swift3

接触swift 已经有一年多的时间了,由最初的OC代码转为 swift 代码,然后从 swift 2.3 转为 swift 3。每次的转换都感觉是将项目整个的翻新了一遍,每次的转换代码都是一次改朝换代。

以下是在代码改朝换代的时候的一些心得:

在将 OC 代码转换为 swift 代码的时候,我当时使用的是 xcode7.3。xcode7.3在我的印象中,编写OC代码就是联想功能最差的一个。 所以:

  • 一、在更新swift的时候,在swift文件中,几乎是不会联想的,动则就是整个屏幕中的代码全是白颜色,然后,类名,方法名,都是硬敲出来的。
  • 二、既然是改写 swift 代码,那么就是对swift 不是太了解
  • 三、针对一些第三方库,期望改为swift版。例如:Masnory -> SnapKit
  • 四、cocoapod 推荐使用 useasframework 的方式集成
  • 五、在 swift 2.3 转 swift 3.0 的时候,block(闭包)里面的参数都不要形参,所以参数的前面都是要加上 _ 来防止错误。并且 block 非option 的都要加上 @escaping 来修饰
  • 六、在swift中,在iOS8 机型中,所有的控制器在使用 xib 的情况下,都要对初始化方法 init(nibName:nibBundle)这个方法重写,否则崩溃
  • 七、增加桥接文件

以上几点都是在转码的时候耗费时间比较长、存在坑的。下面说几点在转码过程中总结的一些经验

  • 一、项目中所使用的到工具类、工厂类、公共类、网络请求的封装等等,就是指一些公共模块。建议在转码初期,先将这些文件转为 swift。既然能称为工具类,那么一般都是解耦的,所以说,可以新建一个swift项目,将这些工具类转为swift文件。这样做的目的是: 一:工具类的使用量非常大,所以很有必要 swift2.0 / 3.0 化 二:在工具类使用如此频繁的情况下,如果你的xcode不会联想,这样将会使多么令人头疼的事情啊!在这点印象颇深
  • 二、更新第三方库为 swift 版,因为虽说允许OC swift混编,但是在类型这个方面兼容性并不是太好。例如:OC中一些 NSArray 的地方,也许我们清楚里面存放的是字符串,但是如果没有显示指定 NSArray <NSString* > * 的话,那么在swift 中使用起来,你只能得到 Any 类型,使用的时候还要类型绑定,这个属于类型兼容性。再者一点就是 OC 的方法在联想方面差的要命
  • 三、cocoapod 使用 useasframework,swift中比较注重 module 的概念,这个也是趋势,所以同样是混编,但是仍然要更改为包的形式
  • 四、关于block 形参的问题,这个需要我们有耐心的一个一个更改
  • 五、在 swift 3.0 中返回值没有使用那么会报一个警告,添加一个@discardresult 在方法的前面,放置警告
  • 六、针对第三点中的 包 的概念,我们会发现,例如在使用 snapkit 的时候,只要使用 snp 的地方都要 import SnapKit (当然这个主要是针对swift2.3 -> swift 3.0 并且没有使用 useasframework )。这个时候会发现每个文件都 import SnapKit 这样来一下,是多么痛苦的事情。 下面是我当时新建的 mac 工程整个项目添加 import SnapKit 的方法
func importSnapKit(path: String) {
    
    let manager = FileManager.default
    
    guard let subPaths = try? manager.subpathsOfDirectory(atPath: path) else {
        return
    }
    
    for subPath in subPaths {
        
        let realPath = path + "/\(subPath)"
        
        var isDirectory: ObjCBool = true
        
        if manager.fileExists(atPath: realPath, isDirectory: &isDirectory) {
            
            if realPath.contains("SnapKit") { // 过滤自身
                continue
            }
            
            if !isDirectory.boolValue {
                
                alterSnapKitContent(path: realPath)
            } else {
                
                importSnapKit(path: realPath)
            }
        }
    }
}

func alterSnapKitContent(path: String) {
    
    guard !path.contains("SnapKit") else { // 过滤自身
        return
    }
    
    guard path.contains(".swift") else {
        return
    }
    
    guard var content = try? String.init(contentsOfFile: path) else {
        print("can't get content at path: \(path)")
        return
    }
    
    guard content.contains("snp.") || content.contains("make.") else {
        return
    }
    
    guard !content.contains("import SnapKit") else {
        return
    }
    
    let containUIKit = content.contains("import UIKit")
    
    let containFoundation = content.contains("import Foundation")
    
    if containUIKit {
        
        content = content.replacingOccurrences(of: "import UIKit", with: "import UIKit\n\nimport SnapKit")
        
        _ = try? content.write(toFile: path, atomically: true, encoding: String.Encoding.utf8)
        
        print("has add:\nimport SnapKit\nin file: \(path)")
    } else if containFoundation {
        
        content = content.replacingOccurrences(of: "import Foundation", with: "import Foundation\n\nimport SnapKit")
        
        _ = try? content.write(toFile: path, atomically: true, encoding: String.Encoding.utf8)
        
        print("has add:\nimport SnapKit\nin file: \(path)")
    } else {
        
        content = content.replacingOccurrences(of: "All rights reserved.\n//", with: "All rights reserved.\n//\n\nimport SnapKit")
        
        _ = try? content.write(toFile: path, atomically: true, encoding: String.Encoding.utf8)
        
        print("has replace all right reversed. ")
        
    }
}

// importSnapKit(path: "/Users/*/Desktop/projectname")

大致思路为: 1、读取项目中的每个文件,当然除了pod、snapkit 文件夹下面的 2、读取每个文件中的内容,判断是否包含snp. 这个字符串,如果存在,则需要导入 import SnapKit 。否则不需要 3、将 import SnapKit 放在 import UIKit 或 import Foundation 或 All rights reserved. 的下面一行 这样等待半分钟,将会自动在需要的文件中 import SnapKit

同样:针对所有的 module 都可以这样导入,只要将限制条件更改为合适的即可

转为Swift 后:

现在我们公司都是使用swift 编程,swift在代码编写方面确实是能够提高效率,尤其是swift 是面向协议编程,其灵活性不可言喻,并且在 swift 的强语言下,swift 项目是相当稳定的。目前 swift 项目唯一不足之处便是xcode 的编译速度问题,编译型语言。我们公司项目是比较大的,每次项目的编译时间在15分钟左右,接下来的任务就是如何降低编译时间。

总体来说推荐大家转为swift编程。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏小詹同学

​我拿 12 年 36 套四级真题做了什么 ?

不会英语的程序员不是好程序员 ?小詹不敢乱立 flag ,但是我知道的是程序员就喜欢自己动手干些实事 ,比如今天教大家自己动手做个有意思的项目——从历年四级英语...

16110
来自专栏杨建荣的学习笔记

任务调度的并行算法

如果串行是肯定不行的。我们可以考虑并行策略,但是开了并行,怎么能够充分利用资源比较好呢。

16830
来自专栏吉浦迅科技

DAY39:阅读扩展数据类型

15120
来自专栏牛客网

阿里技术一面,Java研发岗

之前过了个简单的简历面,过了几天后没打来以为凉了,然后昨晚又接到了电话,括号内容是回答说的,理解有限,不一定都对,欢迎纠正~加油每一个牛友们! 阿里一面: ...

47190
来自专栏熊二哥

UML快速入门

UML(Unified Modeling Language)统一建模语言的概念已经出现了近20年,虽然并不是所有的概念都非常有实践意义,但常见的用例图、类图、序...

27690
来自专栏玉树芝兰

如何用Python读取开放数据?

当你开始接触丰富多彩的开放数据集时,CSV、JSON和XML等格式名词就会奔涌而来。如何用Python高效地读取它们,为后续的整理和分析做准备呢?本文为你一步步...

13320
来自专栏有趣的Python和你

Python数据分析之dataframe的groupbygroupby函数highcharts绘图

14630
来自专栏数据结构与算法

洛谷P2147 [SDOI2008]Cave 洞穴勘测

题目描述 辉辉热衷于洞穴勘测。 某天,他按照地图来到了一片被标记为JSZX的洞穴群地区。经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道...

36190
来自专栏华章科技

最全Python数据工具箱:标准库、第三方库和外部工具都在这里了

导读:Python数据工具箱涵盖从数据源到数据可视化的完整流程中涉及到的常用库、函数和外部工具。其中既有Python内置函数和标准库,又有第三方库和工具。这些库...

2.9K20
来自专栏牛客网

今日头条三面面经

4.       优先队列的底层数据结构?插入和删除一个节点的时间复杂度是多少? 

96820

扫码关注云+社区

领取腾讯云代金券