Swift 面向協(xié)議開發(fā)筆記

1.控件動畫的封裝

比如UIButton,UILabel,UITableViewCell當用戶觸碰這些控件的時候萌抵,我們的需求是要給用戶展示彈跳動畫。如果每個控件都去實現(xiàn)這個動畫舔稀,那么就會出現(xiàn)代碼重復乳丰,并且用繼承去實現(xiàn)這個功能會有明顯的缺點,接下來我們用面向協(xié)議來封裝一下:

import UIKit
//添加一個動畫協(xié)議
protocol AnimationBeat {  
}
//添加擴展
extension AnimationBeat where Self: UIView {
    //隨便添加一個彈跳動畫
    func animationBeat() {
        self.transform = CGAffineTransform()
        UIView.animateKeyframes(withDuration: 0.3, delay: 0, options: UIViewKeyframeAnimationOptions(rawValue: 0), animations: {
            UIView.addKeyframe(withRelativeStartTime: 0, relativeDuration: 1 / 3.0, animations: {
                self.transform = CGAffineTransform(scaleX: 1.1, y: 1.1)
            })
            UIView.addKeyframe(withRelativeStartTime: 1 / 3.0, relativeDuration: 1 / 3.0, animations: {
                self.transform = CGAffineTransform(scaleX: 0.9, y: 0.9)
            })
            UIView.addKeyframe(withRelativeStartTime: 2 / 3.0, relativeDuration: 1 / 3.0, animations: {
                self.transform = CGAffineTransform(scaleX: 1.0, y: 1.0)
            })
        }, completion: nil)
    }
}

只要某個控件需要這個動畫内贮,那么它就可以遵守上面我們添加的協(xié)議产园,當前控件就會有這個彈跳動畫的能力,如下:

//MyLabel遵守AnimationBeat協(xié)議
  import UIKit
  class MyLabel: UILabel, AnimationBeat {
  }

//接下來我們就可以在控制器中調用動畫的方法了
  let myLable = MyLabel()
  //調用協(xié)議中的動畫
  myLable.animationBeat()

2.加載 Xib 的封裝

對于一個項目來說肯定避免不了會用到Xib夜郁,如果項目用到了50個Xib什燕,就會寫出類似50個重復的代碼,如下:

extension ShopView {
    class func loadXib() -> ShopView {
        return Bundle.main.loadNibNamed("\(self)", owner: nil, options: nil)?.first as! ShopView
    }
}

因為加載Xib的方法寫的太麻煩竞端,所以封裝一個類方法屎即,方便外界調用,但是上面這個方法會在很多地方重復出現(xiàn)事富。那我們就可以用協(xié)議來解決這個重復代碼的問題技俐,如下:

protocol LoadXibProtocol {
}
//為上面定義的協(xié)議添加一個擴展
extension LoadXibProtocol where Self: UIView {
    ///提供加載XIB方法
    static func loadXib(xibStr: String? = nil) -> Self {  
        return Bundle.main.loadNibNamed(xibStr ?? "\(self)", owner: nil, options: nil)?.last as! Self
    }
}

我們創(chuàng)建一個Swift文件專門添加協(xié)議,在這個文件中添加上面這段代碼统台,之后每個View只要遵守這個LoadXibProtocol協(xié)議就能調用loadXib方法雕擂,如下:

import UIKit
///遵守LoadXibProtocol協(xié)議
class ShopView: UIView, LoadXibProtocol {
}

那接下來我們使用就非常簡單了,并且提供了2種加載Xib的方式贱勃,看實際情況來使用:

import UIKit
class ViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        //第一種方法
        let shopView = ShopView.loadXib()
        //第二種方法
        let shopView = ShopView.loadXib(xibStr: "ShopView")
        view.addSubview(shopView)
    }
}

3.Cell 的注冊標識封裝

在實際開發(fā)中井赌,很多地方都會用到UITableView谤逼、UICollectionView,并且每次使用都要自己注冊下Cell仇穗,有時候還想不起來怎么寫注冊標識流部,寫多了的話還是比較煩的。接下來我們就用面向協(xié)議來封裝Cell注冊標識仪缸,以后就不用再手寫注冊標識了贵涵,如下:

import UIKit
//定義一個協(xié)議
protocol RegisterRepeat {
}

//為協(xié)議擴展identifier屬性和xib屬性
extension RegisterRepeat {
    static var identifier: String {
        return "\(self)"
    }
    static var xib: UINib? {
        return nil
    }
}

//為UITableView擴展一個方法
extension UITableView {
    func lr_registerCell<T: UITableViewCell>(cell: T.Type) where T: RegisterRepeat {
        if let xib = T.xib {
          // T遵守了RegisterRepeat協(xié)議,所以通過T就能取出identifier這個屬性
            register(xib, forCellReuseIdentifier: T.identifier)
        }else {
            register(cell, forCellReuseIdentifier: T.identifier)
        }
    }
    
    func lr_dequeueReusableCell<T: UITableViewCell>(indexPath: IndexPath) -> T where T: RegisterRepeat {
        return dequeueReusableCell(withIdentifier: T.identifier, for: indexPath) as! T
    }
}

首先我們定義了一個RegisterRepeat協(xié)議恰画,為這個協(xié)議擴展了2個屬性宾茂,identifier就是自動返回遵守當前協(xié)議的Cell標識,xib就是如果從xib加載Cell時需要用到的一個屬性拴还。接下來我們會為UITableView擴展2個方法跨晴,lr_registerCell是注冊Cell的方法,在這個方法中我們添加T這個泛型(這個T必須傳入的是UITableViewCell類型)片林,并且這個泛型T需要遵守RegisterRepeat這個協(xié)議端盆。最后我們通過判斷是否是從Xib中加載,然后在進行Cell注冊费封。同理lr_dequeueReusableCell就是取出這個泛型T焕妙,并且需要遵守RegisterRepeat協(xié)議。接下來我們的自定義Cell只要遵守這個RegisterRepeat協(xié)議就可以了弓摘,如下:

//第一個自定義的Cell焚鹊,代碼加載(非Xib加載)
import UIKit
class ShopViewCell: UITableViewCell, RegisterRepeat {
}

//第二個自定義的Cell,Xib加載
import UIKit
class HomeViewCell: UITableViewCell, RegisterRepeat {
    //因為這個cell是從Xib中加載韧献,需要重寫RegisterRepeat協(xié)議中的xib屬性末患,給它賦值。
    static var xib: UINib? {
        return UINib(nibName: "HomeViewCell", bundle: nil)
    }
}

接下來我們就可以在控制器實現(xiàn)下我們擴展的方法:

import UIKit
class ViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        let myTableView = UITableView(frame: view.bounds, style: .plain)
        myTableView.delegate = self
        myTableView.dataSource = self
        // myTableView.register(UITableViewCell.self, forCellReuseIdentifier: "myTableViewID")
        //通過協(xié)議代碼加載Cell
        // myTableView.lr_registerCell(cell: ShopViewCell.self)
        //通過協(xié)議Xib加載Cell
        myTableView.lr_registerCell(cell: HomeViewCell.self)
        view.addSubview(myTableView)
    }  
}

extension ViewController: UITableViewDelegate, UITableViewDataSource {
    
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 10
    }
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        // let cell = tableView.dequeueReusableCell(withIdentifier: "myTableViewID", for: indexPath)
        // 通過協(xié)議取出Cell
        let cell = tableView.lr_dequeueReusableCell(indexPath: indexPath) as HomeViewCell
        cell.textLabel?.text = "\(indexPath.row)"
        return cell
    }
    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return 40
    }
}

從上面的代碼來看锤窑,我們注冊的Cell方法再也不用去關心如何去寫標識符璧针,你只要把遵守RegisterRepeat協(xié)議的Cell類名給我,我就會自動幫你添加標識符渊啰。同理取Cell的時候也不用關心Cell的標識符了探橱。

參考: Practical Protocol-Oriented-Programming

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市虽抄,隨后出現(xiàn)的幾起案子走搁,更是在濱河造成了極大的恐慌,老刑警劉巖迈窟,帶你破解...
    沈念sama閱讀 211,561評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件私植,死亡現(xiàn)場離奇詭異,居然都是意外死亡车酣,警方通過查閱死者的電腦和手機曲稼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,218評論 3 385
  • 文/潘曉璐 我一進店門索绪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人贫悄,你說我怎么就攤上這事瑞驱。” “怎么了窄坦?”我有些...
    開封第一講書人閱讀 157,162評論 0 348
  • 文/不壞的土叔 我叫張陵唤反,是天一觀的道長。 經(jīng)常有香客問我鸭津,道長彤侍,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,470評論 1 283
  • 正文 為了忘掉前任逆趋,我火速辦了婚禮盏阶,結果婚禮上,老公的妹妹穿的比我還像新娘闻书。我一直安慰自己名斟,他們只是感情好,可當我...
    茶點故事閱讀 65,550評論 6 385
  • 文/花漫 我一把揭開白布魄眉。 她就那樣靜靜地躺著砰盐,像睡著了一般。 火紅的嫁衣襯著肌膚如雪坑律。 梳的紋絲不亂的頭發(fā)上楞卡,一...
    開封第一講書人閱讀 49,806評論 1 290
  • 那天,我揣著相機與錄音脾歇,去河邊找鬼。 笑死淘捡,一個胖子當著我的面吹牛藕各,可吹牛的內容都是我干的。 我是一名探鬼主播焦除,決...
    沈念sama閱讀 38,951評論 3 407
  • 文/蒼蘭香墨 我猛地睜開眼激况,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了膘魄?” 一聲冷哼從身側響起乌逐,我...
    開封第一講書人閱讀 37,712評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎创葡,沒想到半個月后浙踢,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,166評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡灿渴,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,510評論 2 327
  • 正文 我和宋清朗相戀三年洛波,在試婚紗的時候發(fā)現(xiàn)自己被綠了胰舆。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,643評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡蹬挤,死狀恐怖缚窿,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情焰扳,我是刑警寧澤倦零,帶...
    沈念sama閱讀 34,306評論 4 330
  • 正文 年R本政府宣布,位于F島的核電站吨悍,受9級特大地震影響扫茅,放射性物質發(fā)生泄漏。R本人自食惡果不足惜畜份,卻給世界環(huán)境...
    茶點故事閱讀 39,930評論 3 313
  • 文/蒙蒙 一激率、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧拓提,春花似錦勺三、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,745評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至册倒,卻和暖如春蚓挤,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背驻子。 一陣腳步聲響...
    開封第一講書人閱讀 31,983評論 1 266
  • 我被黑心中介騙來泰國打工灿意, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人崇呵。 一個月前我還...
    沈念sama閱讀 46,351評論 2 360
  • 正文 我出身青樓缤剧,卻偏偏與公主長得像,于是被迫代替她去往敵國和親域慷。 傳聞我的和親對象是個殘疾皇子荒辕,可洞房花燭夜當晚...
    茶點故事閱讀 43,509評論 2 348

推薦閱讀更多精彩內容

  • 1.類擴展和分類的區(qū)別 類擴展:沒有名字可以為某個類增加額外的屬性、成員變量和方法 分類:有名字只能擴充方法犹褒,不能...
    彼岸的黑色曼陀羅閱讀 584評論 0 1
  • *面試心聲:其實這些題本人都沒怎么背,但是在上海 兩周半 面了大約10家 收到差不多3個offer,總結起來就是把...
    Dove_iOS閱讀 27,131評論 30 470
  • 1.自定義控件 a.繼承某個控件 b.重寫initWithFrame方法可以設置一些它的屬性 c.在layouts...
    圍繞的城閱讀 3,366評論 2 4
  • 1.昨晚睡得不好抵窒,焦慮感重新涌上心頭。早晨有些沒精神叠骑,難道是因為沒有喝咖啡李皇? 2. 既然想去有道云筆記,不如就去吧...
    一只大白雞閱讀 165評論 0 0
  • 前幾天付材,因為工作需要,我到58招聘網(wǎng)上去篩選簡歷圃阳,說實話厌衔,我有體會到一個做HR的人的心情,總結出了一些簡歷存在的問...
    芥末仙人球閱讀 1,297評論 1 2