swift相关|基础知识

swift:

1.struct和class区别

struct SNode {
    var Data: Int?
}
class CNode {
    var Data: Int?
}

1.property初始化的不同

let snode = SNode(Data: 4) // struct可直接在构造函数中初始化property
print("snode.data:\(String(describing: snode.Data))")
let cnode = CNode()  // class不可直接在构造函数中初始化property
cnode.Data = 5 
print("cnode.data:\(String(describing: cnode.Data))")

2.赋值不同
struct 赋值“=”的时候,会copy一份完整相同的內容给另一個变量 --> 【开辟了新的内存地址】
class 赋值“=”的时候,不会copy一份完整的内容给另一個变量,只是增加了原变量内存地址的引用而已 --> 【没有开辟了新的内存地址】

3.继承
struct不能继承,class可以继承。

4.struct比class更“轻量级”
struct分配在栈中,class分配在堆中。

知识延伸:为什么访问struct比class快?
“堆”和“栈”并不是数据结构上的Heap跟Stack,而是程序运行中的不同内存空间。栈是程序启动的时候,系统事先分配的,使用过程中,系统不干预;堆是用的时候才向系统申请的,用完了需要交还,这个申请和交还的过程开销相对就比较大了。
栈是编译时分配空间,而堆是动态分配(运行时分配空间),所以栈的速度快。

从两方面来考虑:
1.分配和释放:堆在分配和释放时都要调用函数(MALLOC,FREE),比如分配时会到堆空间去寻找足够大小的空间(因为多次分配释放后会造成空洞),这些都会花费一定的时间,而栈却不需要这些。
2.访问时间:访问堆的一个具体单元,需要两次访问内存,第一次得取得指针,第二次才是真正得数据,而栈只需访问一次。

2.map、flatMap和compactMap

map和flatMap

区别一
map 最终将它们组成一个二维的数组;
flatMap 中执行的 closure 返回的是同样的数组,但是 flatMap 将每一个返回的数组都拍扁,取出它的元素,构成一个大的一维数组。

let numbers = [[1, 2, 3, 4], [5, 6], [7]]

let maped = numbers.map { $0 }
let flatMapped = numbers.flatMap { $0 }

print(maped)
print(flatMapped)
/// [[1, 2, 3, 4], [5, 6], [7]]
/// [1, 2, 3, 4, 5, 6, 7]

区别二
map是闭包内return为非可选项,但最终返回值为可选项,即闭包return后又套了一层可选项。
flatMap是闭包内return为可选项,最终返回值也为可选项,即闭包内return后对可选项进行解包,最终又套了一层可选项。

func map( transform: (Wrapped) -> U)  -> U?
func flatMap( transform: (Wrapped) -> U?) -> U?
compactMap函数的应用
  1. 过滤 nil
let nums = [1, nil, 3, nil, 5]
let result = nums.compactMap { (item) -> Int? in
    return item
}
print(result) // [1, 3, 5]

简洁语法,可这样使用

let result = nums.compactMap { return $0 }
print(result) // [1, 3, 5]

2.类型转换

let nums = [1, 2, 3, 4, 5]
let result = nums.compactMap { (item) -> String? in

    return "\(item)"
}
print(result) // ["1", "2", "3", "4", "5"]

简洁语法,可这样使用

let nums = [1, 2, 3, 4, 5]
let result = nums.compactMap { return "\($0)" }
print(result) // ["1", "2", "3", "4", "5"]

3.筛选数据 - 能被4整除的数

let nums = [12, 55, 733, 77, 44]
let result = nums.compactMap { (item) -> Int? in

    if item%4 == 0 {
        return item
    }
    return nil
}
print(result) // [12, 44]

3.public、internal和private

  ◆ privateprivate访问级别所修饰的属性或者方法只能在当前的Swift源文件里可以访问。
  ◆ internal(默认访问级别,internal修饰符可写可不写)internal访问级别所修饰的属性或方法在源代码所在的整个模块都可以访问。如果是框架或者库代码,则在整个框架内部都可以访问,框架由外部代码所引用时,则不可以访问。如果是App代码,也是在整个App代码,也是在整个App内部可以访问。
  ◆ public可以被任何人使用

4.闭包

  1. 非逃逸闭包
    显而易见是非逃逸闭包被限制在函数内,当函数退出的时候,该闭包引用计数不会增加,也就是说其引用计数在进入函数和退出函数时保持不变。

  2. 逃逸闭包
    逃逸闭包恰恰与非逃逸闭包相反,其生命周期长于相关函数,当函数退出的时候,逃逸闭包的引用仍然被其他对象持有,不会在相关函数结束后释放。

Swift 3.x中, 闭包参数默认是非逃逸类型,如果需要其逃逸类型的闭包,记得使用关键字 @escaping
而对于非逃逸型闭包,由于其生命周期确定短于相关函数,编译器可以据此做性能优化。

注意

要谨慎使用@escaping(逃逸闭包),除非明确知道要使用它做什么。

下面是使用逃逸闭包的2个场景:
异步调用: 如果需要调度队列中异步调用闭包, 这个队列会持有闭包的引用,至于什么时候调用闭包,或闭包什么时候运行结束都是不可预知的。
存储: 需要存储闭包作为属性,全局变量或其他类型做稍后使用。

你可能感兴趣的:(swift相关|基础知识)