Swift专题讲解十六——ARC在Swift中的应用

一、引言

ARC(自动引用计数)是Objective-C和Swift中用于解决内存管理问题的方案。在学习Objective-C编程时经常会学习到一个关于ARC的例子:在一个公用的图书馆中,每次进入一人就将卡插入,走的时候将自己的卡拔出拿走。图书馆系统会判定只要有卡插入,就将图书馆的灯打开,当所有卡都被取走后,将图书馆的灯关掉。这个例子对应于Objective-C中的对象声明周期管理十分贴切。每当一个对象增加一个引用时,其引用计数会加1,当一个引用被取消时,对象的引用计数减1,当引用计数减为0时,说明此对象将不再有任何引用,对象会被释放掉,让出内存。Swift也采用同样的方式进行内存管理。

注意:在Swift中只有引用类型有自动引用计数,结构体、枚举这类值类型是没有引用计数的。关于引用计数的示例代码如下:

class MyClass {
    deinit{
        print("MyClass deinit")
    }
}
var cls1:MyClass? = MyClass()
var cls2:MyClass? = cls1
var cls3:MyClass? = cls2
cls2 = nil
cls1 = nil
//执行下面代码后才会打印“MyClass deinit”
cls3 = nil

二、循环引用的处理方法

在开发中,开发者一不小心就会写出产生循环引用的代码,在上面的示例中可以看出,除非实例的引用全部解除,否则实例将不会调用析构方法,内存不会被释放,如果在写代码时,A引用了B,同样B也引用了A,那么实际上现在A和B的引用计数都是2,将A和B都置为nil后,A和B实例依然保有1个引用计数,都不会被释放,实例如下:

class MyClassOne {
    var cls:MyClasstwo?
    deinit{
        print("ClassOne deinit")
    }
}
class MyClasstwo {
    var cls:MyClassOne?
    deinit{
        print("Classtwo deinit")
    }
}
var obj1:MyClassOne? = MyClassOne()
var obj2:MyClasstwo? = MyClasstwo()
obj1?.cls = obj2
obj2?.cls = obj1
obj1=nil
obj2=nil
//没有打印析构函数的调用信息

对于上面的情况,可以将属性声明称weak类型来防止这种循环引用,weak的作用在于只是弱引用实例,原实例的引用计数并不会加1,示例如下:

//关于弱引用的演示
class MyClassthree{
    weak var cls:MyClassFour?
    deinit{
        print("Classthree deinit")
    }
}
class MyClassFour {
    var cls:MyClassthree?
    deinit{
        print("ClassFour deinit")
    }
}
var obj3:MyClassthree? = MyClassthree()
var obj4:MyClassFour? = MyClassFour()
obj3?.cls = obj4
obj4?.cls = obj3
obj4=nil
//此时obj3中的cls也为nil
obj3?.cls

若引用的实例被释放后,其在另一个实例中的引用也将被置为nil,所以weak只能用于optional类型的属性,然而在开发中还有一种情况,某个类必须保有另一个类的示例,这个实例不能为nil,但是这个属性又不能影响其原始实例的释放,这种情况也会造成循环引用,示例如下:

class MyClassFive{
    var cls:MyClassSix
    init(param:MyClassSix){
        cls = param
    }
    deinit{
        print("ClassFive deinit")
    }
}
class MyClassSix{
    var cls:MyClassFive?
    deinit{
        print("ClassSix deinit")
    }
}
var obj6:MyClassSix? = MyClassSix()
var obj5:MyClassFive? = MyClassFive(param: obj6!)
obj6?.cls = obj5
obj5=nil
obj6=nil
//没有打印任何信息

上面的示例也会造成循环引用,然而MyClassFive类中的cls属性为常量不可为nil,不可使用weak弱引用来做Swift中又提供了一个关键字uNowned无主引用来处理这样的问题,示例如下:

class MyClassFive{
    uNowned var cls:MyClassSix
    init(param:MyClassSix){
        cls = param
    }
    deinit{
        print("ClassFive deinit")
    }
}
class MyClassSix{
    var cls:MyClassFive?
    deinit{
        print("ClassSix deinit")
    }
}
var obj6:MyClassSix? = MyClassSix()
var obj5:MyClassFive? = MyClassFive(param: obj6!)
obj6?.cls = obj5
obj5=nil
obj6=nil

关于弱引用和无主引用,其区别主要是在于:

1.弱引用用于解决Optional值的引起的循环引用。

2.无主引用用于解决非Optional值引起的循环引用。

3.个人以为,弱引用可用下图表示:

4.无主引用可用如下图表示:

若将上面的代码修改如下,程序会直接崩溃:

class MyClassFive{
    uNowned var cls:MyClassSix
    init(param:MyClassSix){
        cls = param
    }
    deinit{
        print("ClassFive deinit")
    }
}
class MyClassSix{
    var cls:MyClassFive?
    deinit{
        print("ClassSix deinit")
    }
}
var obj6:MyClassSix? = MyClassSix()
var obj5:MyClassFive? = MyClassFive(param: obj6!)
obj6?.cls = obj5
obj6=nil
obj5?.cls

上面所举的例子满足了两种情况,一种是两类实例引用的属性都是Optional值的时候使用weak来解决循环引用,一种是两类实例有一个为非Optional值的时候使用uNowned来解决循环引用,然而还有第三种情况,两类实例引用的属性都为非Optional值的时候,可以使用无主引用与隐式拆包结合的方式来解决,这也是无主引用最大的应用之处,示例如下:

class MyClassSeven{
    uNowned var cls:MyClassEight
    init(param:MyClassEight){
        cls = param
    }
    deinit{
        print("ClassSeven deinit")
    }
}
class MyClassEight{
    var cls:MyClassSeven!
    init(){
        cls = MyClassSeven(param:self)
    }
    deinit{
        print("ClassEight deinit")
    }
}
var obj7:MyClassEight? = MyClassEight()
obj7=nil

除了在两个类实例间会产生循环引用,在闭包中,也可能出现循环引用,当某个类中包含一个闭包属性,同时这个闭包属性中又使用了类实例,则会产生循环引用,示例如下:

class MyClassNine {
    var name:String = "HS"
    lazy var closure:()->Void = {
        //闭包中使用引用值会使引用+1
        print(self.name)
    }
    deinit{
        print("ClassNine deinit")
    }
}
var obj9:MyClassNine? = MyClassNine()
obj9?.closure()
obj9=nil
//不会打印析构信息

Swift中提供了闭包的捕获列表来对引用类型进行弱引用或者无主引用的转换:

class MyClassNine {
    var name:String = "HS"
    lazy var closure:()->Void = {
        [uNowned self]()->Void in
        print(self.name)
    }
    deinit{
        print("ClassNine deinit")
    }
}
var obj9:MyClassNine? = MyClassNine()
obj9?.closure()
obj9=nil

捕获列表以中括号标识,多个捕获参数则使用逗号分隔。

专注技术,热爱生活,交流技术,也做朋友。

——珲少 QQ群:203317592

Swift专题讲解十六——ARC在Swift中的应用的更多相关文章

  1. 使用canvas压缩图片大小的方法示例

    这篇文章主要介绍了使用canvas压缩图片大小的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. html5写一个BUI折叠菜单插件的实现方法

    这篇文章主要介绍了html5写一个BUI折叠菜单插件的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  3. ios – 如何从Swift中的Closure返回值?

    ).更改实现以使用存储调用值的成员变量.

  4. ios – 如何正确地将选择器作为参数传递给swift

    最后说我有一个包含B的实例的A类.在A类中,我将一个A作为选择器的函数传递给B的一个方法.B使用这个选择器注册通知.但是,当通知进入时,它无法运行选择器并显示“无法识别的选择器发送到实例”.如果我把所有我想在B班做的事情都移到A班,那就行了.不过,我希望他们分开,好像更有条理.我对Objective-C和Swift来说相当新鲜,所以在这种情况下,我不知道如何传递选择器作为参数.斯威夫特答案会很棒.

  5. xcode – Swift闭包:不能将类型'(_) – > Bool’的值转换为预期的参数类型

    注意:使用Xcode7.3.1解决方法一些闪闪发光的东西正在用Swift的类型推断.给卡一个明确的类型,它将工作.您不需要返回类型或返回:注意:这也是:完全指定.Red枚举值也解决了这个问题:在评论中提到,如果您将卡的定义移动到与过滤器相同的文件中,那么它的工作原理.实际上,如果您将CardType枚举从Card中分离出来,只需将CardType移动到带有过滤器的文件中即可.

  6. ios – Swift – 如何在UITableview中获取所选行的数值

    我在Swift中创建的UITableview上有多个选择,我声明了一个包含所选UITableViewCells的NSIndexPaths的Array.如何将此数组转换为可读术语.例如self.selectedRows是NSloglike:选定项目字符串[{length=2,path=0–1},{length=2,path=0–0},path=0–3}]我希望能够将其转换为:1,2,3.在ObjectiveC中,我通过数组枚举了ObjectsWithOptions,并将数组的id添加到一个可变数组中以获得我

  7. ios – 如何在Swift中使用indicesOfObjectsPassingTest:

    我发现它很混乱,当我第一次在objective-c中学习块时,你如何从声明转换为实际使用方法,而Swift的语法至少是令人困惑的.解决方法这段代码在我的游乐场工作;)希望它有点帮助额外功能定义直列关闭我将上面的示例转换为内联闭包.参数列表和返回类型由术语in分隔.

  8. Swift基础语法汇总

  9. Swift 学习一

    国外开发者最近发现,WWDC2014上苹果发布的新语言Swift,和古老的Scala语言在语法上存在众多的相似之处。Swift语言从语法上来看,几乎是Scala的一个分支,在以下功能上几乎是等同的:类型继承、闭包、元组、协议、扩展、泛型等。不过Swift的运行环境和Scala的区别还是很大,这个概念才是Swift最重要的。但Swift最终编译到机器代码,使用引用计数机制,与Objective-C无缝整合。所以Swift和Scala在代码表象上的相似,应该并不太影响两种语言本质机理上的重大不一致。

  10. Swift基础语法: 24 - Swift的枚举语法, 匹配枚举值和 Switch 语句, 关联值, 原始值

    在前面,我们把函数和闭包解决完毕了,现在让我们来看看Swift中的枚举和Switch:1.枚举语法所谓的枚举其实就是定义一个通用类型的一组相关的值,使你可以在你的代码中以一个安全的方式来使用这些值。

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部