importFoundation


//***********************************************************************************************

//1.Automatic Reference Counting(自动引用计数)

//_______________________________________________________________________________________________

//介绍

//Swift使用自动引用计数(ARC)来跟踪并管理应用使用的内存。但是在少数的情况下,为了自动的管理内存空间,ARC需要了解你的代码片段之间的关系的更多信息

//引用计数只应用在类的实例,也就是说只应用在引用类型,不能用在结构体或者枚举中


//2.How ARC WorksARC如何工作)

//说明

//每次创建一个实例,ARC会自动的分配一个内存块用来存储这个实例的相关信息,当实例不再被使用时,ARC会释放这个实例使用的内存,ARC会跟踪实例的属性或者方法,直到所有的属性和方法都不使用的时候,ARC才会释放掉相关的内存


//3.ARC in ActionARC实践)

//_______________________________________________________________________________________________

//实例代码演示ARC如何工作

classPerson{

letname:String

init(name:String){

self.name= name

println("\(name)is being initialized")

}

deinit{

\(name)is being deinitialized")

}

}

varreference1:Person?

varreference2:varreference3:Person?


reference1=Person(name:"Listo") //reference1Person实例的强引用,所以不会被销毁

reference2=reference1

reference3reference2 //如果将这个Person实例赋值给另外的两个变量,那么将建立另外两个指向这个实例的强引用

reference1=nil

nil //此时如果销毁其中的两个引用,只要还有一个对Person类实例的引用存在,ARC不会销毁Person实例


//直到第三个强引用被破坏,ARC销毁Person类实例


//4.Strong Reference Cycles Between Class Instances(类实例之间的强引用环)

//上面演示的是ARC追踪Person实例的引用数量,并且在它不被使用时销毁实例

//一个类的实例永远不会有0个强引用。在两个类实例彼此保持对方的强引用,使得每个实例都使对方保持有效时会发生这种情况,这就是强引用环,这是需要解决的


//代码演示强引用环的产生

classPerson1{

name= name

}

varapartment:Apartment? //每个Person类的实例拥有一个被初始化为nilapartment可选属性,因为一个人并不一定拥有一座公寓

)

}

}


classApartment{

letnumber:Int

init(number:Int) {

number= number

}

tenant:Person1? //每个Apartment类的实例拥有一个被初始化为niltenant可选属性,因为一座公寓不一定有居民

("Apartment #numbervarListo:Person1?

varnumber87:Apartment?


Listo"Listo Han") //此时Listo.name = "Listo",Listo.apartment = nil

number87(number:87) //此时number87.number = 87,number87.tenant = nil


Listo!.apartment=number87

!.tenant //这时将两个实例关联起来,一个人拥有一所公寓,一所公寓拥有一个人,便形成了强引用环


//这样即使破坏了Listonumber87所持有的强引用,引用计数也不会是0,因此ARC不会销毁这两个实例


//5.Resolving Strong Reference Cycles Between Class Instances(解决实例间的强引用环问题)

//解决方法

//Swift中提供两种方法结局强引用环:弱引用和无主引用

//弱引用和无主引用允许引用环中的一个实例引用另外一个实例,但不是强引用。因此实例可以互相引用但是不会产生强引用环

//对于生命周期中引用会变为nil的实例,使用弱引用;对于初始化时赋值之后引用再也不会赋值为nil的实例,使用无主引用


//弱引用原理

//弱引用不会增加实例的引用计数,因此不会阻止ARC销毁被引用的实例。这种特性使得引用不会变成强引用环。声明属性或者变量的时候,关键字weak表明引用为弱引用


//使用实例代码演示弱引用(弱引用面向的是可选类型,分开销毁)

classPerson2{

String) {

name= name

}

varapartment:Apartment2?

classApartment2{

weakPerson2? //声明Apartment2类中的tenant属性为弱引用

varPin:Person2?

varnumber99:Apartment2?


Pin=Person2(name:"Pin s")

number99=Apartment299)


Pinapartmentnumber99 //Person2实例仍然是Apartment2实例的强引用,Apartment2实例是Person2实例的弱引用

!.tenant=Pin


Pin=nil

number99=nil //此时两个实例都被释放掉


//代码演示无主引用(无主引用面向的是非可选类型,一次销毁)

//在这个模型中,消费者不一定有信用卡,但是每张信用卡一定对应一个消费者。鉴于这种关系,Customer类有一个可选类型属性card,CreditCard类的customer属性则是非可选类型的

classCustomer{

varcard:CreditCard? 每个用户不一定都有信用卡

name= name

}

classCreditCard{

uNownedletcustomer:Customer 使用uNowned来声明无主引用

Int,customer:Customer){

number= number

customer= customer

}

("Card #varListo1:Customer?

Listo1=Customer"listo han")

Listo1!.card=CreditCard(number:12344,135)">Listo1!)


Listo1=nil 此时两个实例都被销毁


//5.UNowned References and Implicitly Unwrapped Optional Properties(无主引用以及隐式展开的可选属性)

//PersonApartment的例子说明了下面的场景:两个属性的值都可能是nil,并有可能产生强引用环。这种场景下适合使用弱引用

//CustomerCreditCard的例子则说明了另外的场景:一个属性可以是nil,另外一个属性不允许是nil,并有可能产生强引用环。这种场景下适合使用无主引用

//存在第三种场景:两个属性都必须有值,且初始化完成后不能为nil。这种场景下,要一个类用无主引用属性,另一个类用隐式展开的可选属性


//代码演示第三种情况

classCountry{

letcapitalCity:City!

String,capitalName:name= name

capitalCity=City(name: capitalName,country:self)

}

}


classCity{

letcountry:Country

Country){

country= country

}

}


varcountry =Country(name:"Canada",capitalName:"ottawa")

("country.)'s captial city is calledcapitalCity) //使用隐式展开的可选值满足了两个类的初始化函数的要求。初始化完成后,capitalCity属性就可以做为非可选值类型使用,却不会产生强引用环


//6.Strong Reference Cycles for Closures(闭包产生的强引用环)

//将一个闭包赋值给类实例的某个属性,并且这个闭包使用了实例,这样也会产生强引用环


//代码演示闭包产生强引用环

classHTMLElement{

String=""

lettext:String?

@lazyvarasHTML: () ->String= { 属性引用闭包

iflettext =self.text{

return"<\(self.name)>\(text)</\(>"

}

else{

/>"

}

}

String? =nil){

text= text

}

varparagraph:HTMLElement? =HTMLElement(name:"p",text:"hello,world")

(paragraph!.asHTML())


paragraph //这时声明paragraphnil,不会被释放掉,因为闭包使用了实例的属性,所以不释放


//解决闭包产生的强引用环

//在定义闭包时同时定义占有列表作为闭包的一部分,可以解决闭包和类实例之间的强引用环。占有列表定义了闭包内占有一个或者多个引用类型的规则。和解决两个类实例间的强引用环一样,声明每个占有的引用为弱引用或无主引用,而不是强引用。根据代码关系来决定使用弱引用还是无主引用

classHTMLElement1{

String= {

[uNownedself]in //在这里声明占有列表来解决闭包的强引用,占有列表放在闭包的参数之前,也就是闭包最开始的地方

self.text{

>"

}

else{

/>"

}

}

text= text

}

varparagraph1:HTMLElement1? =HTMLElement1(name:(paragraph1())


paragraph1=nil 此时实例被释放


转载:http://blog.csdn.net/u013096857/article/details/37872083

寒城攻略:Listo 教你 25 天学会 Swift 语言 - 18 Automatic Reference Counting的更多相关文章

  1. three.js模拟实现太阳系行星体系功能

    这篇文章主要介绍了three.js模拟实现太阳系行星体系功能,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下

  2. HTML5页面无缝闪开的问题及解决方案

    这篇文章主要介绍了HTML5页面无缝闪开方案,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  3. ios – Swift中的UIView动画不起作用,错误的参数错误

    我正在尝试制作动画并使用下面的代码.我得到“无法使用类型’的参数列表调用’animateWithDuration'(FloatLiteralConvertible,延迟:FloatLiteralConvertible,选项:UIViewAnimationoptions,动画:()–>()–>$T4,完成:(Bool)–>(Bool)–>$T5)’“错误.这意味着我使用了错误的参数.我错了.请

  4. ios – 无法识别的选择器发送到实例NSTimer Swift

    解决方法让updateTime成为一个类方法.如果它是在一个纯粹的Swift类中,你需要在@objc前面说明该方法的声明,如:

  5. ios – 为什么,将nil作为参数从Objc C发送到swift类初始化器,用新对象替换nil参数

    除非属性本身被声明为nonnull:

  6. ios – 在Swift中对MKCircle进行子类化

    我想通过添加另一个String属性来继承MKCircle,我们称之为“代码”.这个属性不是可选的和常量的,所以我必须从初始化器设置它,对吧?有没有办法定义一个单一的便利初始化器,在这种情况下需要3个参数?本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  7. ios – 类型推断(自动类型检测)如何在swift中工作?

    LLVM如何检测变量是一个字符串?

  8. ios – Swift可选项:语言问题,还是做错了什么?

    应该有可选的类型;type是但是,如果我这样做,它的工作原理:它似乎是基本的替代,但我可能会遗漏一些语言的细微差别.谁能对此有所了解?之后就像暧昧一样,更多,这是我的解决方案:这适用于所有非对象Swift对象,包括Swift字符串,数字等.感谢Viktor提醒我String不是Swift中的对象.如果您知道值的类型,您可以替换任何?使用适当的可选类型,如String?

  9. ios – 覆盖Swift中的超类委托

    我正在开发一个包含两个UIViews的Swift(v1.2)项目.MyView和MyViewSubclass.MyView有一个委托,我想在MyViewSubclass中覆盖它作为一个子协议,类似于UITableViews有一个UITableViewDelegate,它也符合超级uiscrollviewdelegate.我的第一个想法是覆盖超类属性,但这会导致编译器错误,因为子类不能覆盖具有不同类

  10. ios – 使用捕获列表中的无主内容导致崩溃,即使块本身也不会执行

    欣赏有关如何调试此内容的任何提示或有关导致崩溃的原因的解释……

随机推荐

  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,所以编译器会报错,现在来一一解决。

返回
顶部