苹果公司的新的编程语言 Swift 高级语言(十一)--初始化类的析构函数的一个实例

时间:2024-03-22 18:03:44

一 、实例的初始化

         实例的初始化是准备一个类、结构或枚举的实例以便使用的过程。

初始化包含设置一个实例的每个存储属性为一个初始值,以及运行不论什么其他新的实例可以使用之前须要的设置或初始化。

一个类、结构或枚举能定义一个初始化方法来设置它的特性,用来确保它的实例的全部属性都有有效的初始值。

通过调用类、结构或枚举提供的初始化方法来运行实例的初始化过程。

类的实例也能实现一个析构,用来在类的实例释放之前运行不论什么特定的清除过程来释放分配的专有资源。

1 、 初始化方法的定义

初始化方法定义与实例方法的定义形式类似。能够包括參数,也能够不包括參数。

包括參数时也能够为參数指定本地和外部參数名字。

但初始化方法的名字固定使用init,且不返回值。

最简单的初始化方法是不带參数的init方法。

struct Fahrenheit {

var temperature:Double

init() {

temperature =32.0

}

}

var f =Fahrenheit()

以上定义了一个名字为Fahrenheit的结构。包括一个初始化方法init,用来在该结构实例化时为它唯一的存储属性temperature分配一个初始值32.0 。

除了在初始化方法中为实例的属性设置初始值外,还能在属性定义时为其提供一个默认值。

在属性定义时为其提供默认值是更好的初始化属性的方法,一方面语法更加简单,还有一方面还能够依据提供的默认值的类型判断属性的类型,并且还能够更好的利用默认初始化和初始化继承的特点。

以下相同的结构定义。比上面的结构定义语法更简单。

struct Fahrenheit {

var temperature
=32.0

}

你能使用输入參数和可选的属性类型来定义一个或多个定制的初始化方法,多个定制的初始化方法依据每一个初始化方法提供的參数的名称和类型加以区分和被调用。

例如以下样例展示了怎样使用參数为一个结构定义不同的初始化方法,以及实例化时怎样依据初始化方法提供的參数的不同名字来调用对应的初始化方法。

struct Celsius {

var temperatureInCelsius:Double
=0.0

init(fromFahrenheitfahrenheit:Double)
{

temperatureInCelsius = (fahrenheit
-32.0) /1.8

}

init(fromKelvinkelvin:Double)
{

temperatureInCelsius =kelvin
-273.15

}

}

let boilingPointOfWater
=Celsius(fromFahrenheit:212.0)

// boilingPointOfWater.temperatureInCelsius is 100.0

let freezingPointOfWater
=Celsius(fromKelvin:273.15)

因为初始化方法的參数名字和类型扮演的重要角色,因此假设你在一个初始化方法的定义中没有提供一个外部名字,则Swift会自己主动为它的每个參数提供一个自己主动分配的名字,由Swift自己主动产生的外部參数名字与本地參数名字同样。

与实例方法同样,当然假设你不想为初始化方法的參数提供外部參数名字。能够使用’_’作为參数的外部名字。

常量属性的值可以在初始化期间被改动,但对于类的实例的常量属性,仅能在引入该属性的类的初始化方法中改动,而不能被其子类改动。

2、 默认初始化方法

假设一个结构或基类没有提供定制的初始化方法,且其全部的属性都提供有默认值,则Swift可以为其提供一个默认的初始化方法,默认的初始化方法设置新创建的实例的全部属性的值为它们的默认值。例如以下样例所看到的:

class ShoppingListItem
{

var name:String?

var quantity
=1

var purchased
=false

}

var item =ShoppingListItem()

因为类ShoppingListItem的全部的三个属性都提供有默认值。且ShoppingListItem是一个基类,也没有为其提供定制的初始化方法。因此Swift为其提供了一个默认初始化方法,默认初始化方法不带參数。

3、 结构类型的成员逐一初始化方法(memberwise Initializers)

假设一个结构的全部的存储属性都提供有默认值,且未定义不论什么自己的定制初始化方法。则除了能够使用Swift为其提供的默认初始化方法外,也自己主动接收一个成员逐一初始化方法。

成员逐一初始化方法是初始化新结构实例的成员属性的一种快捷方式,新实例的全部属性的初始值通过名字逐一传送给该初始化方法然后被设置。例如以下所看到的:

struct Size
{

var width
=0.0,height =0.0

}

let twoByTwo =Size(width:2.0,height:2.0)

在该样例中,结构Size包括仅有的两个存储属性width和height,并都带有默认值,而其本身也没有提供初始化方法,因此其自己主动接收一个为init(width:height:)的成员逐一初始化方法。

可以使用它来初始化一个新的Size实例。其属性的初始值通过成员逐一初始化方法的參数逐一提供。

对于值类型(结构和枚举)。你能在自定义的初始化方法内部使用self.init调用同样类型的其他初始化方法。

4、 类的初始化

因为类涉及继承,因此一个类的初始化过程比較复杂。须要在初始化过程完毕对类的全部存储属性(包含从它的超类链继承的随意属性)的初始化。Swift为了确保正确完毕一个类的初始化。定义了两类类的初始化方法:指派初始化方法和便利初始化方法。

4.1、 类的指派初始化方法和便利初始化方法

类的指派初始化方法是一个类的主要初始化方法。

一个类的指派初始化方法初始化该类本身引入的全部的属性。然后向上调用其直接超类的初始化方法来继续超类琏的初始化过程。一个类必须有至少一个指派初始化方法。通常一个类仅有一个指派初始化方法。

类的便利初始化方法是类的次要初始化方法。你能为类定义一个便利初始化方法来调用同样类的其他初始化方法。一个类能够不必提供便利初始化方法。

便利初始化方法作为一种便利的初始化方法,主要用来方便创建某些特定的实例或者特定输入值类型的类的实例,因此在便利初始化方法中调用指派初始化方法时,对于指派初始化方法的一些參数被设置为默认值。

类的指派初始化方法和便利初始化方法之间的调用必须遵守例如以下规则:

1)指派初始化方法必须调用它的直接超类的指派初始化方法。

2)便利初始化方法必须调用同样类的另外的能够使用的初始化方法。

3) 便利初始化方法最后必须调用一个指派初始化方法。

类的指派初始化方法语法与值类型的简单初始化方法同样:

init(parameters) {

statements

}

便利初始化方法的语法除了使用conveniencekeyword来标示外,语法与指派初始化方法同样。

convenience init(parameters)
{

statements

}

4.2、 类的两阶段初始化

在Swift中类的初始化分成两个阶段完毕,在第一个阶段,每个存储属性被引入它的类的初始化方法分配一个初始值。一旦每个存储属性的初始状态确定。開始第二个初始化阶段:在新的实例可以使用之前,进一步定制它的存储属性。

为了正确完毕一个类的两阶段初始化过程,Swift进行例如以下安全检查:

1) 指派初始化方法必须在调用它的直接超类的初始化方法之前确保由它的类引入的全部的属性被初始化。

2) 指派初始化方法在为一个继承的属性分配一个值之前必须向上调用它的直接超类的初始化方法;

3) 便利初始化方法必须在为不论什么属性分配值之前调用其他的初始化方法;

4) 一个初始化方法在第一个阶段完毕前不能调用其他不论什么实例方法,读不论什么实例属性的值或者引用self,这是由于这时实例还处于内存不确定状态。

4.3、 初始化方法的继承和重写

与Objective-C语言不同,Swift 中子类默认不继承它的超类的初始化方法。但同意在子类中重写超类的初始化方法。与方法、属性、下标不同。初始化方法的重写不须要写一个override标识。

4.4、 自己主动初始化继承

在一些条件下,超类的初始化方法也能被它的子类自己主动继承。

规则1:假设一个子类未定义不论什么指派初始化方法,则自己主动继承它的超类的全部指派初始化方法。即对于指派初始化方法继承而言,没有则继承。

规则2:假设一个子类提供了其超类的全部指派初始化方法的实现,能够是通过规则1继承来的。也能够是提供自己特定的实现,那么它自己主动继承其超类的全部便利初始化方法。即对于便利初始化方法继承而言假设已实现超类的全部指派初始化方法,则自己主动继承全部便利初始化方法。

5、 使用闭合或功能为属性设置默认值。

       
      你能使用一个闭合或全局功能来为一个存储属性提供特定的默认值。

在该属性所属类型的新的实例被初始化时,为一个属性提供默认值的闭合或全局功能被调用,且返回一个值作为该属性的默认值。

例如以下所看到的:

   class SomeClass
{

let someProperty:SomeType
= {

return someValue

}()

}

              该例使用一个闭合为someProperty属性提供默认值。该闭合以一个圆括号结束。说明该闭合可以被运行。

须要注意:在使用一个闭合为一个属性分配默认值时。由于在闭合运行时,该实例还没有初始化完毕,因此在闭合内部不能存取实例的其他不论什么属性值,即使那些属性带有默认值,也不能使用隐含的self属性。也不能调用不论什么实例的方法。

二、析构

在一个实例不再须要时,Swift会自己主动释放它。

Swift使用automatic reference counting (ARC)来实现一个实例的内存管理。

因此当实例释放时。用户不须要手工清除它所使用的系统资源。

但是当你的实例包括和使用了自己分配的资源时。你可能须要运行一些额外的清除工作。比如创建一个打开一个文件的类。就可能须要在类的实例释放前关闭该文件。

每一个类可以定义至少一个析构。析构语法例如以下:

deinit {

// perform the deinitialization

}

一个实例的析构在实例释放之前自己主动被调用。超类的析构被子类继承,超类的析构可以在子类的析构实现的最后被自己主动调用。

超类的析构总是被调用,即使一个没有提供自己的析构的子类。

在一个实例的析构完毕之前该实例还没有被释放,因此在析构内部可以存取实例的全部属性。

版权所有所有。请明确注明转载的链接和出处,谢谢!

版权声明:本文博客原创文章,博客,未经同意,不得转载。