寒城攻略:Listo 教你 25 天学会 Swift 语言 - 18 Automatic Reference Counting

前端之家收集整理的这篇文章主要介绍了寒城攻略:Listo 教你 25 天学会 Swift 语言 - 18 Automatic Reference Counting前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

importFoundation@H_301_3@


@H_301_3@

//***********************************************************************************************@H_301_3@

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

//_______________________________________________________________________________________________@H_301_3@

//介绍@H_301_3@

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

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


@H_301_3@

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

//说明@H_301_3@

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


@H_301_3@

//3.ARC in ActionARC实践)@H_301_3@

//_______________________________________________________________________________________________@H_301_3@

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

classPerson{@H_301_3@

letname:String@H_301_3@

init(name:String){@H_301_3@

self.name= name@H_301_3@

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

}@H_301_3@

@H_301_3@

deinit{@H_301_3@

\(name)is being deinitialized")@H_301_3@

}@H_301_3@

}@H_301_3@

varreference1:Person?@H_301_3@

varreference2:varreference3:Person?@H_301_3@


@H_301_3@

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

reference2=reference1@H_301_3@

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

reference1=nil@H_301_3@

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


@H_301_3@

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


@H_301_3@

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

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

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


@H_301_3@

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

classPerson1{@H_301_3@

name= name@H_301_3@

}@H_301_3@

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

)@H_301_3@

}@H_301_3@

}@H_301_3@


@H_301_3@

classApartment{@H_301_3@

letnumber:Int@H_301_3@

init(number:Int) {@H_301_3@

number= number@H_301_3@

}@H_301_3@

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

("Apartment #numbervarListo:Person1?@H_301_3@

varnumber87:Apartment?@H_301_3@


@H_301_3@

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

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


@H_301_3@

Listo!.apartment=number87@H_301_3@

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


@H_301_3@

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


@H_301_3@

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

//解决方法@H_301_3@

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

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

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


@H_301_3@

//弱引用原理@H_301_3@

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


@H_301_3@

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

classPerson2{@H_301_3@

String) {@H_301_3@

name= name@H_301_3@

}@H_301_3@

varapartment:Apartment2?@H_301_3@

classApartment2{@H_301_3@

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

varPin:Person2?@H_301_3@

varnumber99:Apartment2?@H_301_3@


@H_301_3@

Pin=Person2(name:"Pin s")@H_301_3@

number99=Apartment299)@H_301_3@


@H_301_3@

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

!.tenant=Pin@H_301_3@


@H_301_3@

Pin=nil@H_301_3@

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


@H_301_3@

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

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

classCustomer{@H_301_3@

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

name= name@H_301_3@

}@H_301_3@

classCreditCard{@H_301_3@

unownedletcustomer:Customer 使用unowned来声明无主引用@H_301_3@

Int,customer:Customer){@H_301_3@

number= number@H_301_3@

customer= customer@H_301_3@

}@H_301_3@

("Card #varListo1:Customer?@H_301_3@

Listo1=Customer"listo han")@H_301_3@

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


@H_301_3@

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


@H_301_3@

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

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

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

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


@H_301_3@

//代码演示第三种情况@H_301_3@

classCountry{@H_301_3@

letcapitalCity:City!@H_301_3@

String,capitalName:name= name@H_301_3@

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

}@H_301_3@

}@H_301_3@


@H_301_3@

classCity{@H_301_3@

letcountry:Country@H_301_3@

Country){@H_301_3@

country= country@H_301_3@

}@H_301_3@

}@H_301_3@


@H_301_3@

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

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


@H_301_3@

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

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


@H_301_3@

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

classHTMLElement{@H_301_3@

String=""@H_301_3@

lettext:String?@H_301_3@

@H_301_3@

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

iflettext =self.text{@H_301_3@

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

}@H_301_3@

else{@H_301_3@

/>"@H_301_3@

}@H_301_3@

}@H_301_3@

@H_301_3@

String? =nil){@H_301_3@

text= text@H_301_3@

}@H_301_3@

@H_301_3@

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

(paragraph!.asHTML())@H_301_3@


@H_301_3@

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


@H_301_3@

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

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

classHTMLElement1{@H_301_3@

String= {@H_301_3@

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

self.text{@H_301_3@

>"@H_301_3@

}@H_301_3@

else{@H_301_3@

/>"@H_301_3@

}@H_301_3@

}@H_301_3@

text= text@H_301_3@

}@H_301_3@

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


@H_301_3@

paragraph1=nil 此时实例被释放@H_301_3@


@H_301_3@

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

猜你在找的Swift相关文章