美文网首页
Swift-12.深浅拷贝与闭包引起的循环强引用

Swift-12.深浅拷贝与闭包引起的循环强引用

作者: 顺手给我new一个对象 | 来源:发表于2017-01-30 15:26 被阅读84次

    本章包含内容:

    • 深浅拷贝
    • 字符串的拷贝
    • 集合类的拷贝
    • 隐式强引用-集合,timer、元组
    • 闭包属性引起的循环强引用
    • 解决闭包属性引起的循环强引用

    一、深浅拷贝

    直接给出结论:

    • 值类型的赋值属于深拷贝(重新申请一份内存,把原内存上的内容拷贝一份到新的内存中)
    • 引用类型的赋值属于浅拷贝(指向同一份内存)

    从上面给出的结论,我们可以通过以下代码验证:

    //结构体、值类型
    struct Deep {
        var copy: Int = 0
    }
    
    //类、引用类型
    class Shallow {
        var copy: Int = 0
    }
    
    var d0 = Deep()
    var d1 = d0
    d1.copy = 10
    //此时时深拷贝,d0和d1修改的是不同的内存地址
    print(d0.copy)//输出 0
    print(d1.copy)//输出 10
    
    
    var s0 = Shallow()
    var s1 = s0
    //此时时浅拷贝,s0和s1修改的是同一份内存地址
    s1.copy = 9
    print(s0.copy)//输出9
    print(s1.copy)//输出9
    

    二、字符串的拷贝

    直接给出结论:

    • String类型的赋值属于深拷贝(struct类型)
    • NSMutableString类型的赋值属于浅拷贝(class类型)

    从上面给出的结论,我们可以通过以下代码验证:

    var swiftStr: String = "Hello"
    var swiftStr1 = swiftStr
    
    swiftStr1 += " world"
    print(swiftStr) //输出:Hello
    print(swiftStr1)//输出:Hello world
    
    var ocStr = NSMutableString(string: "Hello")
    var ocStr1 = ocStr
    ocStr1.insert(" world", at: ocStr.length)
    print(ocStr) //输出:Hello world
    print(ocStr1)//输出:Hello world
    

    三、集合类的拷贝

    集合的拷贝如下:

    //struct类型
    var array:Array<Int> = [1,2,3]
    var array1 = array
    
    array1 += [4,5,6]
    
    //深拷贝
    print(array) //输出:[1, 2, 3]
    print(array1)//输出:[1, 2, 3, 4, 5, 6]
    
    //struct类型
    var dict:Dictionary<Int,String> = [1:"a",2:"b"]
    var dict1 = dict
    dict[3] = "c"
    //深拷贝
    print(dict) //输出:[2: "b", 3: "c", 1: "a"]
    print(dict1)//输出:[2: "b", 1: "a"]
    
    
    //class类型
    var ocArray = NSMutableArray(array:[1,2,3])
    var ocArray1 = ocArray
    ocArray1.add(5)
    //浅拷贝
    print(ocArray)
    /*
     输出:
     (
     1,
     2,
     3,
     5
     )
    
     */
    print(ocArray1)
    /*
     输出:
     (
     1,
     2,
     3,
     5
     )
     */
    

    数组的拷贝过程:

    • (1)根据被拷贝数组的大小来创建一个新的数组对象,新的容量跟原始数组大小相同
    • (2)将原始数组中的每一个元素一次拷贝到新的数组对象中

    以上拷贝过程,可以通过以下代码验证:

    //结构体、值类型
    struct Deep {
        var copy: Int = 0
    }
    
    //类、引用类型
    class Shallow {
        var copy: Int = 0
    }
    
    var de0 = Deep()
    var de1 = Deep()
    //数组的元素都是值类型
    var dearray = [de0, de1]
    
    var sh0 = Shallow()
    var sh1 = Shallow()
    //数组的元素都是引用类型
    var sharray = [sh0,sh1]
    
    
    //深拷贝
    var dearray1 = dearray
    var sharray1 = sharray
    
    //当将数组中的某个元素替换,或者改变了数组的大小,不会影响另外一个数组。
    /* 深拷贝
    dearray1.removeLast()
    print(dearray1.count)//输出:1
    print(dearray.count)//输出:2
    */
    
    //深拷贝
    dearray1[0] = Deep(copy: 3)
    print(dearray1[0].copy) //输出3
    print(dearray[0].copy)  //输出1
    
    //此时是浅拷贝
    sharray[0].copy = 99
    print(sharray[0].copy) //输出:99
    print(sharray1[0].copy)//输出:99
    

    四、隐式强引用

    以下是显示强引用

    class Student {
        var name: String
        init(name: String) {
            self.name = name
        }
        
        func show() {
            print("name = \(name)")
        }
        
        deinit {
            print("\(name) deinit")
        }
    }
    
    //此时,stu0和stu1都引用了同一个对象
    var stu0: Student? = Student(name: "Tom") //引用计数+1
    var stu1 = stu0 //引用计数+1
    stu0 = nil //用用计数-1
    stu1 = nil //用用计数-1
    

    与之对应的,以下是隐式强引用:

    class Student {
        var name: String
        init(name: String) {
            self.name = name
        }
        
        func show() {
            print("name = \(name)")
        }
        
        deinit {
            print("\(name) deinit")
        }
    }
    
    //此时,stu0和stu1都引用了同一个对象
    /*
    var stu0: Student? = Student(name: "Tom") //引用计数+1
    var stu1 = stu0 //引用计数+1
    stu0 = nil //用用计数-1
    stu1 = nil //用用计数-1
    */
    
    //放到外面,此时构造出的两个对象,构造完成即销毁
    /*
    Student(name: "zhangsan") //输出:Tom deinit
    Student(name: "lisi")  //输出:zhangsan deinit
    */
    
    //将对象放到数组内,因数组内的元素是引用类型,那么数组会强引用该对象
    var stuarray:[Student] = [Student(name: "zhangsan"), Student(name: "lisi")]
    stuarray[0].name = "zhangsanfeng"
    //(1)当某个对象不在属于数组时,该对象的引用对象会减1
    //(2)数组本身被销毁的时候,它包含的所有对象不再属于它,因此如果对象是引用数据类型,它的计数将会减1
    stuarray.remove(at: 0)//此时zhangsanfeng对象将被释放
    

    五、闭包属性引起的循环强引用

    class CycleRef {
        //闭包是引用类型
        lazy var closure:()->Void = {
            print("closure!")
        }
        deinit {
            print("deinit")
        }
    }
    
    var cr:CycleRef? = CycleRef()
    cr!.closure()//在这时候才初始化closure:(),输出:closure!
    cr = nil //输出:deinit
    

    将以上代码修改如下将引起循环强引用:

    class CycleRef {
        var a: Int = 9
        //闭包是引用类型
        lazy var closure:()->Void = {
            print("a = \(self.a)")
        }
        deinit {
            print("deinit")
        }
    }
    
    var cr:CycleRef? = CycleRef() //引用计数+1
    cr!.closure()//在这时候才初始化closure:(),输出:closure! 引用计数+1
    cr = nil //没有输出deinit,代表此时对象没有被释放,内存泄漏了
    

    六、解决闭包属性引起的循环强引用

    [unowned self] in 解决闭包属性引起的循环强引用:

    class CycleRef {
        var a: Int = 9
        //如果闭包属性中没有直接或者间接访问self,就不会产生循环强引用
        //闭包是引用类型
        lazy var closure:()->Void = {
            //默认闭包会对它访问的对象执行强引用
            [unowned self] in //添加这句代码
            print("a = \(self.a)")
        }
        deinit {
            print("deinit")
        }
    }
    
    var cr:CycleRef? = CycleRef() //引用计数+1
    cr!.closure()//在这时候才初始化closure:(),输出:closure! 引用计数+1
    cr = nil //因为在闭包中添加了[unowned self] in这句代码,此时输出:deinit,代表此时对象被释放
    

    [weak self] in 解决闭包属性引起的循环强引用:

    class CycleRef {
        var a: Int = 9
        //如果闭包属性中没有直接或者间接访问self,就不会产生循环强引用
        //闭包是引用类型
        lazy var closure:()->Void = {
            //默认闭包会对它访问的对象执行强引用
            [weak self] in //添加这句代码
            print("a = \(self!.a)")
        }
        deinit {
            print("deinit")
        }
    }
    
    var cr:CycleRef? = CycleRef() //引用计数+1
    cr!.closure()//在这时候才初始化closure:(),输出:closure! 引用计数+1
    cr = nil //因为在闭包中添加了[weak self] in这句代码,此时输出:deinit,代表此时对象被释放
    

    因为weak一般修饰的是可选类型,所以我们在很多时候会使用** [unowned self] in**

    相关文章

      网友评论

          本文标题:Swift-12.深浅拷贝与闭包引起的循环强引用

          本文链接:https://www.haomeiwen.com/subject/eichittx.html