版本記錄
版本號 | 時間 |
---|---|
V1.0 | 2017.08.01 |
前言
我是swift2.0的時候開始接觸的,記得那時候還不是很穩定,公司的項目也都是用oc做的,并不對swift很重視,我自己學了一段時間,到現在swift3.0+已經出來了,自己平時也不寫,忘記的也差不多了,正好項目這段時間已經上線了,不是很忙,我就可以每天總結一點了,希望對自己對大家有所幫助。在總結的時候我會對比oc進行說明,有代碼的我會給出相關比對代碼。
1. swift簡單總結(一)—— 數據簡單值和類型轉換
2. swift簡單總結(二)—— 簡單值和控制流
3. swift簡單總結(三)—— 循環控制和函數
4. swift簡單總結(四)—— 函數和類
5. swift簡單總結(五)—— 枚舉和結構體
6. swift簡單總結(六)—— 協議擴展與泛型
7. swift簡單總結(七)—— 數據類型
8. swift簡單總結(八)—— 別名、布爾值與元組
9. swift簡單總結(九)—— 可選值和斷言
10. swift簡單總結(十)—— 運算符
11. swift簡單總結(十一)—— 字符串和字符
12. swift簡單總結(十二)—— 集合類型之數組
13. swift簡單總結(十三)—— 集合類型之字典
14. swift簡單總結(十四)—— 控制流
15. swift簡單總結(十五)—— 控制轉移語句
16. swift簡單總結(十六)—— 函數
17. swift簡單總結(十七)—— 閉包(Closures)
18. swift簡單總結(十八)—— 枚舉
19. swift簡單總結(十九)—— 類和結構體
20. swift簡單總結(二十)—— 屬性
21. swift簡單總結(二十一)—— 方法
22. swift簡單總結(二十二)—— 下標腳本
23. swift簡單總結(二十三)—— 繼承
24. swift簡單總結(二十四)—— 構造過程
25. swift簡單總結(二十五)—— 構造過程
26. swift簡單總結(二十六)—— 析構過程
27. swift簡單總結(二十七)—— 自動引用計數
28. swift簡單總結(二十八)—— 可選鏈
29. swift簡單總結(二十九)—— 類型轉換
30.swift簡單總結(三十)—— 嵌套類型
擴展 - Extensions
擴展就是向一個已有的類、結構體或枚舉類型添加新功能,這包括在沒有權限獲取原始源代碼的情況下擴展類型的能力(即逆向建模),擴展和OC
中的分類類似,不過不同的是,swift
的擴展是沒有名字的。
在swift
中的擴展可以:
- 添加計算型屬性和計算靜態屬性
- 定義實例方法和類型方法
- 提供新的構造器
- 定義下標
- 定義和使用新的嵌套類型
- 使一個已有類型符合某個協議
注意:如果你定義了一個已有類型添加新功能,那么這個新功能對該類型的所有已有實例中都是可用的,即使他們是在你的這個擴展的前面定義的。
本篇主要從以下幾個方面進行說明。
- 擴展語法
- 計算型屬性
- 構造器
- 方法
- 下標
- 嵌套類型
擴展語法 - Extension Syntax
下面先看一下如何聲明一個擴展。
extension SomeType {
//加到SomeType的新功能代碼
}
一個擴展可以擴展一個已有來行,使其能夠適配一個或多個協議,當這種情況發生時,協議的名字應該完全按照類或結構體的名字方式進行書寫。
extension SomeType : SomeProtocol, AnotherProtocol {
//加到SomeType的新功能代碼
}
按照這種方式添加的協議遵循者(protocol comformance)
被稱為在擴展中添加協議遵循者。
計算性屬性 - Computed Properties
擴展可以向已有類型添加計算性實例屬性和計算型類型屬性,下面就是向swift
內部的Double
類型加入幾個計算型實例屬性。
下面看一下代碼。
class JJPracticeVC: UIViewController {
override func viewDidLoad()
{
super.viewDidLoad()
view.backgroundColor = UIColor.lightGray
let oneInch = 25.4.mm
print("One inch is \(oneInch) meters")
let threeFeet = 3.ft
print("Three feet is \(threeFeet) meters")
}
}
extension Double {
var km : Double {
return self * 1_000.0
}
var m : Double {
return self
}
var cm : Double {
return self/100.0
}
var mm : Double {
return self / 1_000.0
}
var ft : Double {
return self/3.28084
}
}
下面看輸出結果
One inch is 0.0254 meters
Three feet is 0.914399970739201 meters
這些屬性是只讀的計算型屬性,所以考慮他們不用get
關鍵字,它們的返回值是Double
類型。
構造器 - Initializers
擴展可以向已有類型添加新的構造器,這可以讓你擴展其他類型,將你自己的定制類型作為構造參數,或者提供該類型的原始實現中沒有包含的額外初始化選項。
擴展能向類中添加新的便利構造器,但是它們不能向類中添加新的指定構造器或析構函數,指定構造器和析構函數必須總是由原始的類實現來提供。
下面的例子給出的就是描述幾何矩形的定制結構體Rect
,下面看一下簡單的代碼。
struct Size {
var width = 0.0, height = 0.0
}
struct Point {
var x = 0.0, y = 0.0
}
struct Rect {
var origin = Point()
var size = Size()
}
因為結構體Rect
提供了其所有屬性的默認值,所以正如默認構造器中描述的,它可以自動接受一個默認的構造器和一個成員級構造器,這些構造器可以用于構造新的Rect
實例。
let defaultRect = Rect()
let aRect = Rect(origin: Point(x : 2.0, y : 2.0 ), size: Size(width: 5.0, height: 5.0))
你可以提供一個額外的使用特殊中心點和大小的構造器來擴展Rect
結構體。
extension Rect{
init(center : Point, size : Size) {
let originX = center.x - size.width * 0.5
let originY = center.y - size.height * 0.5
self.init(origin: Point(x : originX, y : originY), size: size)
}
}
這個新的構造器首先根據提供的center
和size
值計算一個合適的點,然后調用該結構體自動的成員構造器init(origin: size:)
,該構造器將新的原點和大小存到了合適的屬性中。
class JJPracticeVC: UIViewController {
override func viewDidLoad()
{
super.viewDidLoad()
view.backgroundColor = UIColor.lightGray
let centerRect = Rect(center: Point(x : 4.0, y : 4.0), size: Size(width: 3.0, height: 3.0))
print(centerRect)
}
}
下面看輸出結果
Rect(origin: JJSwift.Point(x: 2.5, y: 2.5), size: JJSwift.Size(width: 3.0, height: 3.0))
注意:如果你使用擴展提供了一個新的構造器,你依舊有責任保證構造過程能夠讓所有勢力完全初始化。
方法 - Methods
擴展可以向已有類型添加新的實例方法和類型方法,下面的例子向Int
類型添加一個新的實例方法repetitions
。
extension Int{
func repetitions(task : () -> ()) {
for i in 0..<self {
task()
}
}
}
這個repetitions
方法使用了一個() -> ()
類型的單參數,表明函數沒有參數而且沒有返回值,定義該擴展后,你就可以對任意整數調用repetitions
方法,進行多次執行某個任務。
下面我們就調用下試試。
class JJPracticeVC: UIViewController {
override func viewDidLoad()
{
super.viewDidLoad()
view.backgroundColor = UIColor.lightGray
3.repetitions {
print("Hello")
}
}
}
下面看輸出結果
Hello
Hello
Hello
上面其實就是尾隨閉包。
1. 修改實例方法 - Mutating Instance Methods
通過擴展添加的實例方法也可以修改該實例本身,結構體和枚舉類型中修改self
或其屬性的方法必須將該實例方法標注為mutating
,正如來自原始實現的修改方法一樣。
下面的例子給Int
類型添加了一個新的名字為square
的修改方法,來實現一個原始值的平方計算。
class JJPracticeVC: UIViewController {
override func viewDidLoad()
{
super.viewDidLoad()
view.backgroundColor = UIColor.lightGray
var someInt = 3
someInt.square()
print(someInt)
}
}
extension Int{
mutating func square(){
self = self * self
}
}
下面看輸出結果
9
下標 - Subscripts
擴展可以向一個已有類型添加新下標,看下面的例子,向Int
添加了一個整型下標,該下標[n]
返回十進制數字從右向左的第n
個數字。例如123456789
中 [0]
返回9
。
下面看一下代碼。
class JJPracticeVC: UIViewController {
override func viewDidLoad()
{
super.viewDidLoad()
view.backgroundColor = UIColor.lightGray
print(123456789[0])
}
}
extension Int{
subscript( digitIndex : Int) -> Int{
var index = digitIndex
var decimalBase = 1
while index > 0 {
decimalBase *= 10
index -= 1
}
return (self/decimalBase) % 10
}
}
下面看輸出結果
9
如果該Int
值沒有足夠的位數,下標越界,那么上述實現的下標會返回0
,因為它會在數字左邊自動補0
。
print(123456789[10])
下面看輸出結果
0
嵌套類型 - Nested Types
擴展可以向已有的類、結構體和枚舉添加新的嵌套類型。
extension Character{
enum Kind {
case Vowel, Consonaut, Other
}
var kind : Kind{
switch String(self).lowercased() {
case "a", "e", "i", "o", "u":
return .Vowel
case "b", "c", "d", "f", "g", "h", "j", "k", "l", "m","n", "p", "q", "r", "s", "t", "v", "w", "x", "y", "z":
return .Consonaut
default:
return .Other
}
}
}
該例子向Character
添加了新的嵌套枚舉,這個名為kind
的枚舉表示特定字符的類型,這個例子還添加了一個新的計算實例屬性kind
,用來返回合適的kind
枚舉成員。
下面我們調用一下。
class JJPracticeVC: UIViewController {
override func viewDidLoad()
{
super.viewDidLoad()
view.backgroundColor = UIColor.lightGray
printLetterKinds(word: "Hello")
}
}
func printLetterKinds(word : String){
print("\(word) is made up of the following kinds of letters")
for character in word.characters {
switch character.kind {
case .Vowel:
print("vowel")
case .Consonaut:
print("consonaut")
default:
print("other")
}
}
}
下面看輸出結果
Hello is made up of the following kinds of letters
consonaut
vowel
consonaut
consonaut
vowel
這個很好理解,相信大家都可以看的明白。
后記
未完,待續~~~