Private 权限控制
Swift3中,如果将主体函数的变量定义为private,则其extension无法读取此变量,必须将其改为filePrivate才可以,但是权限又被扩大了。在Swift4中,private变量也可以被其extension读取。
类和接口的组合语法支持
Swift4.0中泛型支持类和接口的组合
编译相关
加快了编译速度
- Swift3.2可以和Swift4进行共存,并在XCode中提供版本选择
- 提前编译Swift-OC Bridge,加快速度
- 不再单独进行Indexing,每次编译时会自动index
- 在Size方面自动过滤掉不用的代码,减少体积
- 可以选择 "Strip Swift Symbol"来达到减少编译后的体积
减少Swift暴露给OC的接口
在之前的Swift 版本中,所有继承自NSObject的类的属性和方法都会被默认加上@objc标识,以便将其暴露给Objective-C,编译器会为这些方法和属性生成可供OC调用的代码,但是实际情况可能只有很少量的几个方法需要暴露给OC,这就造成很大的空间浪费。因此在Swift 4中,编译器大部分情况下不会自动添加@objc标识,如果要暴露给OC,需要手动添加。
Before Swift 4, the compiler made some Swift declarations automatically available to Objective-C. For example, if one subclassed from NSObject, the compiler created Objective-C entry points for all methods in such classes. The mechanism is called @objc inference.
In Swift 4, such automatic @objc inference is deprecated because it is costly to generate all those Objective-C entry points. When "Swift 3 @objc Inference" setting is set to "On", it allows the old code to work. However, it will show deprecation warnings that need to be addressed. It is recommended to "fix" these warnings and switch the setting to "Off", which is the default for new Swift projects.
可以通过以上设置开启/关闭Swift3的 @objc推断
String相关
- Multi-line string literals 多行字符串字面量
- String本身是character的集合(Collection)
所以可以把String当Collection进行操作
- 新增类型Substring
对String进行substring等操作会得到一个类型为Substring的字符串。Substring跟String很类似,他们都实现了StringProtocol。
在 Swift 中,String 的背后有个 Owner Object 来跟踪和管理这个 String,String 对象在内存中的存储由内存起始地址、字符数、指向 Owner Object 指针组成。Owner Object 指针指向 Owner Object 对象,Owner Object 对象持有 String Buffer。当对 String 做取子字符串操作时,子字符串的 Owner Object 指针会和原字符串指向同一个对象,因此子字符串的 Owner Object 会持有原 String 的 Buffer。当原字符串销毁时,由于原字符串的 Buffer 被子字符串的 Owner Object 持有了,原字符串 Buffer 并不会释放,造成极大的内存浪费。
在 Swift 4 中,做取子串操作的结果是一个 Substring 类型,它无法直接赋值给需要 String 类型的地方。必须用 String(<substring>) 包一层,系统会通过复制创建出一个新的字符串对象,这样原字符串在销毁时,原字符串的 Buffer 就可以完全释放了。
- Unicode
改善了在计算Unicode字符长度时的正确性
在 Unicode 中,有些字符是由几个其它字符组成的,比如 é 这个字符,它可以用 \u{E9} 来表示
Associated type constraints 关联类型约束
- protocol中也可以使用where语句对关联类型进行约束了
protocol SomeProtocol where Self: UICollectionViewCell {
}
- Sequence协议有自己的Element associatedtype了,不需要写Iterator.Element了
extension Sequence where Element: Numeric {
var sum: Element {
var result: Element = 0
for element in self {
result += element
}
return result
}
}
[1,2,3,4].sum
强制要求对内存变量的访问具有互斥性
例如,在遍历一个数组的过程中如果同时在循环内对数组进行数据操作,会在编译时报错。