Swift中類的使用
主要內(nèi)容
- 類的介紹和定義
- 類的屬性
- 類的構(gòu)造函數(shù)
一. 類的介紹和定義
- Swift也是一門面向?qū)ο箝_發(fā)的語言
- 面向?qū)ο蟮幕A(chǔ)是類,類產(chǎn)生了對(duì)象
- 在Swift中如何定義類呢?
- class是Swift中的關(guān)鍵字,用于定義類
class 類名 : SuperClass {
// 定義屬性和方法
}
- 注意:
- 定義的類,可以沒有父類.那么該類是rootClass
- 通常情況下,定義類時(shí).繼承自NSObject(非OC的NSObject)
二. 如何定義類的屬性
類的屬性介紹
- Swift中類的屬性有多種
- 存儲(chǔ)屬性:存儲(chǔ)實(shí)例的常量和變量
- 計(jì)算屬性:通過某種方式計(jì)算出來的屬性
- 類屬性:與整個(gè)類自身相關(guān)的屬性
存儲(chǔ)屬性
- 存儲(chǔ)屬性是最簡單的屬性嫉到,它作為類實(shí)例的一部分沃暗,用于存儲(chǔ)常量和變量
- 可以給存儲(chǔ)屬性提供一個(gè)默認(rèn)值,也可以在初始化方法中對(duì)其進(jìn)行初始化
- 下面是存儲(chǔ)屬性的寫法
- age和name都是存儲(chǔ)屬性,用來記錄該學(xué)生的年齡和姓名
- chineseScore和mathScore也是存儲(chǔ)屬性,用來記錄該學(xué)生的語文分?jǐn)?shù)和數(shù)學(xué)分?jǐn)?shù)
class Student : NSObject {
// 定義屬性
// 存儲(chǔ)屬性
var age : Int = 0
var name : String?
var chineseScore : Double = 0.0
var mathScore : Double = 0.0
}
// 創(chuàng)建學(xué)生對(duì)象
let stu = Student()
// 給存儲(chǔ)屬性賦值
stu.age = 10
stu.name = "why"
stu.chineseScore = 89.0
stu.mathScore = 98.0
計(jì)算屬性
- 計(jì)算屬性并不存儲(chǔ)實(shí)際的值何恶,而是提供一個(gè)getter和一個(gè)可選的setter來間接獲取和設(shè)置其它屬性
- 計(jì)算屬性
一般
只提供getter方法
- 如果只提供getter孽锥,而不提供setter,則該計(jì)算屬性為只讀屬性,并且可以省略get{}
- 下面是計(jì)算屬性的寫法
- averageScore是計(jì)算屬性,通過chineseScore和mathScore計(jì)算而來的屬性
- 在setter方法中有一個(gè)newValue變量,是系統(tǒng)指定分配的
class Student : NSObject {
// 定義屬性
// 存儲(chǔ)屬性
var age : Int = 0
var name : String?
var chineseScore : Double = 0.0
var mathScore : Double = 0.0
// 計(jì)算屬性
var averageScore : Double {
get {
return (chineseScore + mathScore) / 2
}
// 沒有意義,因?yàn)橹螳@取值時(shí)依然是計(jì)算得到的
// newValue是系統(tǒng)分配的變量名,內(nèi)部存儲(chǔ)著新值
set {
self.averageScore = newValue
}
}
}
// 獲取計(jì)算屬性的值
print(stu.averageScore)
類屬性
- 類屬性是與類相關(guān)聯(lián)的细层,而不是與類的實(shí)例相關(guān)聯(lián)
- 所有的類和實(shí)例都共有一份類屬性.因此在某一處修改之后,該類屬性就會(huì)被修改
- 類屬性的設(shè)置和修改,需要通過類來完成
- 下面是類屬性的寫法
- 類屬性使用static來修飾
- courseCount是類屬性,用來記錄學(xué)生有多少門課程
class Student : NSObject {
// 定義屬性
// 存儲(chǔ)屬性
var age : Int = 0
var name : String?
var chineseScore : Double = 0.0
var mathScore : Double = 0.0
// 計(jì)算屬性
var averageScore : Double {
get {
return (chineseScore + mathScore) / 2
}
// 沒有意義.newValue是系統(tǒng)分配的變量名,內(nèi)部存儲(chǔ)著新值
set {
self.averageScore = newValue
}
}
// 類屬性
static var corseCount : Int = 0
}
// 設(shè)置類屬性的值
Student.corseCount = 3
// 取出類屬性的值
print(Student.corseCount)
監(jiān)聽屬性的改變
- 在OC中我們可以重寫set方法來監(jiān)聽屬性的改變
- Swift中可以通過屬性觀察者來監(jiān)聽和響應(yīng)屬性值的變化
- 通常是監(jiān)聽存儲(chǔ)屬性和類屬性的改變.(對(duì)于計(jì)算屬性惜辑,我們不需要定義屬性觀察者,因?yàn)槲覀兛梢栽谟?jì)算屬性的setter中直接觀察并響應(yīng)這種值的變化)
- 我們通過設(shè)置以下觀察方法來定義觀察者
- willSet:在屬性值被存儲(chǔ)之前設(shè)置疫赎。此時(shí)新屬性值作為一個(gè)常量參數(shù)被傳入盛撑。該參數(shù)名默認(rèn)為newValue,我們可以自己定義該參數(shù)名
- didSet:在新屬性值被存儲(chǔ)后立即調(diào)用捧搞。與willSet相同抵卫,此時(shí)傳入的是屬性的舊值狮荔,默認(rèn)參數(shù)名為oldValue
- willSet與didSet只有在屬性第一次被設(shè)置時(shí)才會(huì)調(diào)用,在初始化時(shí)介粘,不會(huì)去調(diào)用這些監(jiān)聽方法
- 監(jiān)聽的方式如下:
class Person : NSObject {
var name : String? {
// 可以給newValue自定義名稱
willSet (new){ // 屬性即將改變,還未改變時(shí)會(huì)調(diào)用的方法
// 在該方法中有一個(gè)默認(rèn)的系統(tǒng)屬性newValue,用于存儲(chǔ)新值
print(name)
print(new)
}
// 可以給oldValue自定義名稱
didSet (old) { // 屬性值已經(jīng)改變了,會(huì)調(diào)用的方法
// 在該方法中有一個(gè)默認(rèn)的系統(tǒng)屬性oldValue,用于存儲(chǔ)舊值
print(name)
print(old)
}
}
var age : Int = 0
var height : Double = 0.0
}
let p : Person = Person()
// 在賦值時(shí),監(jiān)聽該屬性的改變
// 在OC中是通過重寫set方法
// 在swift中,可以給屬性添加監(jiān)聽器
p.name = "why"
//p.name = "yz"
類的構(gòu)造函數(shù)
構(gòu)造函數(shù)的介紹
- 構(gòu)造函數(shù)類似于OC中的初始化方法:init方法
- 默認(rèn)情況下載創(chuàng)建一個(gè)類時(shí),必然會(huì)調(diào)用一個(gè)構(gòu)造函數(shù)
- 即便是沒有編寫任何構(gòu)造函數(shù)殖氏,編譯器也會(huì)提供一個(gè)默認(rèn)的構(gòu)造函數(shù)。
- 如果是繼承自NSObject,可以對(duì)父類的構(gòu)造函數(shù)進(jìn)行重寫
構(gòu)造函數(shù)的基本使用
構(gòu)造函數(shù)的基本使用
- 類的屬性必須有值
- 如果不是在定義時(shí)初始化值,可以在構(gòu)造函數(shù)中賦值
class Person: NSObject {
var name : String
var age : Int
// 重寫了NSObject(父類)的構(gòu)造方法
override init() {
name = ""
age = 0
}
}
// 創(chuàng)建一個(gè)Person對(duì)象
let p = Person()
初始化時(shí)給屬性賦值
- 很多時(shí)候,我們?cè)趧?chuàng)建一個(gè)對(duì)象時(shí)就會(huì)給屬性賦值
- 可以自定義構(gòu)造函數(shù)
- 注意:如果自定義了構(gòu)造函數(shù),會(huì)覆蓋init()方法.即不在有默認(rèn)的構(gòu)造函數(shù)
class Person: NSObject {
var name : String
var age : Int
// 自定義構(gòu)造函數(shù),會(huì)覆蓋init()函數(shù)
init(name : String, age : Int) {
self.name = name
self.age = age
}
}
// 創(chuàng)建一個(gè)Person對(duì)象
let p = Person(name: "why", age: 18)
字典轉(zhuǎn)模型(初始化時(shí)傳入字典)
- 真實(shí)創(chuàng)建對(duì)象時(shí),更多的是將字典轉(zhuǎn)成模型
- 注意:
- 去字典中取出的是NSObject,任意類型.
- 可以通過as!轉(zhuǎn)成需要的類型,再賦值(不可以直接賦值)
class Person: NSObject {
var name : String
var age : Int
// 自定義構(gòu)造函數(shù),會(huì)覆蓋init()函數(shù)
init(dict : [String : NSObject]) {
name = dict["name"] as! String
age = dict["age"] as! Int
}
}
// 創(chuàng)建一個(gè)Person對(duì)象
let dict = ["name" : "why", "age" : 18]
let p = Person(dict: dict)
字典轉(zhuǎn)模型(利用KVC轉(zhuǎn)化)
- 利用KVC字典轉(zhuǎn)模型會(huì)更加方便
- 注意:
- KVC并不能保證會(huì)給所有的屬性賦值
- 因此屬性需要有默認(rèn)值
- 基本數(shù)據(jù)類型默認(rèn)值設(shè)置為0
- 對(duì)象或者結(jié)構(gòu)體類型定義為可選類型即可(可選類型沒有賦值前為nil)
class Person: NSObject {
// 結(jié)構(gòu)體或者類的類型,必須是可選類型.因?yàn)椴荒鼙WC一定會(huì)賦值
var name : String?
// 基本數(shù)據(jù)類型不能是可選類型,否則KVC無法轉(zhuǎn)化
var age : Int = 0
// 自定義構(gòu)造函數(shù),會(huì)覆蓋init()函數(shù)
init(dict : [String : NSObject]) {
// 必須先初始化對(duì)象
super.init()
// 調(diào)用對(duì)象的KVC方法字典轉(zhuǎn)模型
setValuesForKeysWithDictionary(dict)
}
}
// 創(chuàng)建一個(gè)Person對(duì)象
let dict = ["name" : "why", "age" : 18]
let p = Person(dict: dict)
類的析構(gòu)函數(shù)
析構(gòu)函數(shù)
- Swift 會(huì)自動(dòng)釋放不再需要的實(shí)例以釋放資源
- Swift 通過自動(dòng)引用計(jì)數(shù)(ARC)處理實(shí)例的內(nèi)存管理
- 當(dāng)引用計(jì)數(shù)為0時(shí),系統(tǒng)會(huì)自動(dòng)調(diào)用析構(gòu)函數(shù)(不可以手動(dòng)調(diào)用)
- 通常在析構(gòu)函數(shù)中釋放一些資源(如移除通知等操作)
- 析構(gòu)函數(shù)的寫法
deinit {
// 執(zhí)行析構(gòu)過程
}
示例練習(xí)
class Person {
var name : String
var age : Int
init(name : String, age : Int) {
self.name = name
self.age = age
}
deinit {
print("Person-deinit")
}
}
var p : Person? = Person(name: "why", age: 18)
p = nil
----------
import UIKit
/*
析構(gòu)函數(shù) 相當(dāng)于OC 中的 dealloc 函數(shù)
*/
class Person {
var name : String = ""
var age : Int = 0
//重寫析構(gòu)函數(shù), 監(jiān)聽對(duì)象的銷毀
deinit {
print("-----definite--")
}
}
var p : Person? = Person()
p = nil