swift開(kāi)發(fā)之?dāng)U展實(shí)現(xiàn)命名空間(實(shí)例方法,類方法)

\color{red}{著急的火雞們,最終全部代碼在最下面}

命名空間

對(duì)長(zhǎng)期從事objective-c語(yǔ)言開(kāi)發(fā)的我們來(lái)說(shuō),命名空間可能是一個(gè)比較陌生的名稱。

“命名空間”,簡(jiǎn)單地說(shuō)已艰,就是不允許有相同類名的區(qū)域痊末。從事過(guò)java或者js開(kāi)發(fā)的同學(xué)可能會(huì)有經(jīng)驗(yàn),這類語(yǔ)言的命名空間其實(shí)就是他們的目錄名哩掺,即只要在不同目錄下凿叠,就可以允許有相同的類名。

OC就比較尷尬了,它沒(méi)有命名空間一說(shuō)盒件,也就是全局都不允許有相同的類名蹬碧。那如何保證這一點(diǎn)?蘋(píng)果是建議在類名前加2-3個(gè)唯一的字符來(lái)將自己的類名與其他區(qū)分開(kāi)炒刁,于是就出現(xiàn)了UIView, NSString, MBProgressHUD, CALayer, AFNetworking, SDWebImage等

swift中恩沽,蘋(píng)果終于引入了命名空間一說(shuō),在任意類中打印一下self 會(huì)出現(xiàn)"命名空間.className"翔始,swift中的命名空間的使用不是一個(gè)項(xiàng)目,而是需要跨項(xiàng)目,在一個(gè)項(xiàng)目中,都是一個(gè)命名空間,在同一個(gè)命名空間下,所有全局變量或者函數(shù)共享,不需要import,從swift開(kāi)始,官方更多的建議大家使用pod來(lái)管理第三方框架罗心,不然倒入一個(gè)框架到處都可以用

擴(kuò)展方法前綴

在OC中,蘋(píng)果建議在擴(kuò)展中的方法需要增加前綴城瞎,原因是防止與自帶方法或者其他庫(kù)的擴(kuò)展中方法重名渤闷,事實(shí)上也應(yīng)該這么做,因?yàn)槲覀冇星败囍b脖镀,往往這類由于重寫(xiě)了方法造成的閃退飒箭,一旦xcode不能正常捕捉錯(cuò)誤,將很難排查蜒灰。

swift擴(kuò)展中弦蹂,同樣需要關(guān)心方法覆蓋的問(wèn)題,對(duì)于原生類自帶的方法卷员,我們可以覆蓋重復(fù)定義盈匾,并且最終調(diào)用走的是擴(kuò)展中的方法,但是擴(kuò)展中的方法不能重復(fù)定義毕骡,xcode會(huì)檢測(cè)并報(bào)錯(cuò)

自定義命名空間

綜上所述削饵,我們自己模擬出類似“命名空間”,是個(gè)不錯(cuò)的選擇未巫,原因如下:

1.防止擴(kuò)展中的方法或?qū)傩愿采w了原來(lái)已有的窿撬,造成無(wú)法預(yù)期的錯(cuò)誤

2.有了命名空間,我們就不需要加前綴這種影響美觀的操作叙凡,代碼可讀性更高

3.有了命名空間劈伴,開(kāi)發(fā)過(guò)程中,尤其對(duì)于新人握爷,可一眼看出方法或?qū)傩允菍儆谠愖詭У倪€是擴(kuò)展的跛璧,防止長(zhǎng)時(shí)間使用造成下意識(shí)的認(rèn)知疲勞

Swift擴(kuò)展模擬“命名空間”

首先,我們要知道swift中幾個(gè)概念:

協(xié)議:與OC中協(xié)議類似新啼,都是定義一套遵守者需要實(shí)現(xiàn)的規(guī)則追城,但是與OC不同的是,在swift中我們也可以對(duì)協(xié)議進(jìn)行擴(kuò)展燥撞,最終效果是所有遵守該協(xié)議的類都會(huì)增加協(xié)議被擴(kuò)展的內(nèi)容

泛型:swift提供了“泛型”來(lái)最大程度使函數(shù)座柱、變量迷帜、容器等靈活化,如果你在架構(gòu)一個(gè)應(yīng)用或者sdk色洞,那么泛型可以提供最大的便利性戏锹。swift中的標(biāo)準(zhǔn)庫(kù)都是通過(guò)泛型定義的,例如Array可以塞進(jìn)Int火诸,也可以塞進(jìn)String

泛型約束:顧名思義锦针,就是通過(guò)泛型,來(lái)約束協(xié)議遵守者的類型

正式開(kāi)始惭蹂,我們的思路是通過(guò)擴(kuò)展模擬出“命名空間”伞插,其實(shí)這不是正兒八經(jīng)的命名空間,只是期望通過(guò)一個(gè)特殊的符號(hào)盾碗,將我們自己擴(kuò)展的方法屬性等和官方的以及第三方的區(qū)分開(kāi)來(lái)媚污,類似于:


self.circleView.wm.moveToBottom()

加入circleView是一個(gè)UIView實(shí)例,這里的wm就是我們所說(shuō)的特殊符號(hào)廷雅,其實(shí)也就是一個(gè)屬性耗美,moveToBottom就是我們自己擴(kuò)展出的方法。

看到這里航缀,第一個(gè)問(wèn)題就拋出來(lái)了商架,如何給circleView擴(kuò)展一個(gè)名叫wm的屬性。很多聰明火雞們就馬上會(huì)想到兩種方式芥玉,一種是擴(kuò)展UIView蛇摸,增加一個(gè)屬性;另一種是使UIView遵守一個(gè)協(xié)議灿巧,通過(guò)擴(kuò)展協(xié)議來(lái)增加一個(gè)屬性赶袄。

假設(shè),我們擴(kuò)展的屬性類型是:


public class NameSpace {

}

方法一:


extension UIView {

  public var wm: NameSpace {

        get {

            return NameSpace()

        }

    }

}

方法二:


/// 命名空間協(xié)議

public protocol NameSpaceProtocol {

    public var wm: NameSpace { get }

}

/// 擴(kuò)展協(xié)議

extension NameSpaceProtocol {

    public var wm: NameSpace {

        get {

            return NameSpace()

        }

    }

}

/// UIView實(shí)現(xiàn)協(xié)議

extension UIView: NameSpaceProtocol {



}

我們將這兩種方式做個(gè)比較抠藕,結(jié)論還是顯而易見(jiàn)的饿肺,方式二的好處有:

1.我們?cè)跀U(kuò)展每個(gè)類的時(shí)候,不需要像方式一那樣都聲明一個(gè)mw的屬性盾似,而是只要實(shí)現(xiàn)NameSpaceProtocol就可以了

2.對(duì)于子類敬辣,如果我們不希望其有這個(gè)屬性,那么方式一就無(wú)解了零院,方式二則可以利用泛型約束的方式溉跃,可以隨心所欲的控制

3.方式二寫(xiě)法更多的采用了swift獨(dú)有的特性,風(fēng)格上更加優(yōu)雅告抄,簡(jiǎn)單說(shuō)就是更裝*

我們?cè)诖嘶A(chǔ)上撰茎,在對(duì)NameSpace進(jìn)行擴(kuò)展,就實(shí)現(xiàn)了最終想要的效果


extension NameSpace {

  public func moveToBottom() {



}

}

/// 此時(shí)玄妈,UIView已經(jīng)達(dá)到了想要的效果

let circleView = UIView()

circleView.wm.moveToBottom()

第二個(gè)問(wèn)題就來(lái)了乾吻,這里我們真正擴(kuò)展的其實(shí)是NameSpace,我們這里目標(biāo)只有UIView拟蜻,如果接下來(lái)還要給Date, Int, String等等擴(kuò)展绎签,實(shí)際上都是對(duì)NameSpace擴(kuò)展,那么如果不做區(qū)分酝锅,那在其中一個(gè)類調(diào)用方法時(shí)诡必,Xcode會(huì)提示出所有,包括其他目標(biāo)擴(kuò)展出的方法搔扁,事實(shí)上真的去調(diào)用非本目標(biāo)的方法爸舒,編譯也是不會(huì)報(bào)錯(cuò)的,但是這不是我們想要的稿蹲。于是扭勉,我們引入泛型約束來(lái)做區(qū)分:


/// 命名空間

public final class NameSpace<T> {



}

/// 擴(kuò)展UIView

extension NameSpace where T == UIView {

}

這下舒服了,在使用過(guò)程中不是對(duì)UIView的擴(kuò)展不會(huì)出現(xiàn)在快捷提示苛聘。這里也做了個(gè)小優(yōu)化涂炎,就是不希望NameSpace再做它用,所以加了個(gè)final描述一下

第三個(gè)問(wèn)題设哗,circleView.wm.moveToBottom()這個(gè)方法唱捣,如果moveToBottom方法中需要訪問(wèn)circleView的方法或?qū)傩栽趺凑课覀冎牢覀儗?shí)際上擴(kuò)展的是NameSpace類网梢,所以我們需要在NameSpace中記錄下來(lái)原來(lái)的對(duì)象就完事了:


/// 命名空間協(xié)議

public protocol NameSpaceProtocol {

    associatedtype TargetType



    /// 實(shí)例變量及方法命名空間

    var wm: NameSpace<TargetType> { get }

}

/// 命名空間

public final class NameSpace<T> {

    internal var base: T

    init(_ base: T) {

        self.base = base

    }

}

/// 擴(kuò)展協(xié)議

extension NameSpaceProtocol {

    public var wm: NameSpace<Self> {

        get {

            return NameSpace<Self>(self)

        }

    }

}

/// 在擴(kuò)展過(guò)程中通過(guò)self.base訪問(wèn)原來(lái)對(duì)象

extension NameSpace where T == UIView {

public func moveToBottom() {

    print("my x is \(self.base.frame.origin.x)")

}

}

寫(xiě)到這里震缭,已經(jīng)讓大部分火雞們滿足了需求,實(shí)際上網(wǎng)上大多數(shù)資料也就到此為止了战虏,但是仍然有部分不滿意拣宰,因?yàn)槲覀円恢弊龅亩际菍?duì)實(shí)例屬性或者是方法做擴(kuò)展,如果是類屬性或者方法活烙,類似于UIColor.wm.color(hexString)這中徐裸,其實(shí)也簡(jiǎn)單,過(guò)程不過(guò)多贅述啸盏,直接貼上\color{red}{最終代碼}


/// 命名空間協(xié)議

public protocol NameSpaceProtocol {

    associatedtype TargetType



    /// 實(shí)例變量及方法命名空間

    var wm: NameSpace<TargetType> { get }



    /// 類變量及方法命名空間

    static var wm: NameSpace<TargetType>.Type { get }

}

/// 命名空間

public final class NameSpace<T> {

    internal var base: T

    internal var BASE: Self.Type

    init(_ base: T) {

        self.base = base

        self.BASE = Self.self

    }

}

/// 擴(kuò)展協(xié)議

extension NameSpaceProtocol {

    public var wm: NameSpace<Self> {

        get {

            return NameSpace<Self>(self)

        }

    }

    public static var wm: NameSpace<Self>.Type {

        get {

            return NameSpace<Self>.self

        }

    }

}

/// 例子:擴(kuò)展UIImage

extension UIImage: NameSpaceProtocol {}

extension NameSpace where T == UIImage {



    /// 根據(jù)顏色生成圖片, 類方法

    public class func image(color: UIColor?, size: CGSize = CGSize(width: 1, height: 1)) -> UIImage {

        UIGraphicsBeginImageContextWithOptions(size, false, UIScreen.main.scale);

        if let color = color, let currentContext = UIGraphicsGetCurrentContext() {

            let fillRect = CGRect(x: 0, y: 0, width: size.width, height: size.height);

            currentContext.setFillColor(color.cgColor)

            currentContext.fill(fillRect)

            let colorImage = UIGraphicsGetImageFromCurrentImageContext()

            UIGraphicsEndImageContext()

            return colorImage ?? UIImage()

        }

        return UIImage()

    }



  /// 寬度重贺,實(shí)例屬性

  public var width: CGFloat {

      return self.base.size.width

}

}

/// 實(shí)際使用

let image = UIImage.wm.image(color: .red, size: CGSize(width: 100, height: 200))

print("the image width is\(image.wm.width)")

注意一:在擴(kuò)展NameSpace之前,我們需要將目標(biāo)實(shí)現(xiàn)一下NameSpaceProtocol協(xié)議回懦,但是實(shí)際開(kāi)發(fā)過(guò)程中你會(huì)發(fā)現(xiàn)有些會(huì)報(bào)警告說(shuō)已經(jīng)實(shí)現(xiàn)過(guò)了气笙,不必驚慌,那是因?yàn)楦割悓?shí)現(xiàn)過(guò)怯晕,子類就不必實(shí)現(xiàn)了潜圃,比如可以將NSObject實(shí)現(xiàn)NameSpaceProtocol協(xié)議,之后UIView等類就不用再寫(xiě)這一步驟了

注意二:也許有火雞想利用runtime擴(kuò)展屬性舟茶,請(qǐng)注意谭期,在擴(kuò)展NameSpace時(shí)堵第,屬性runtime方式添加時(shí),務(wù)必添加到self.base中:


/// 響應(yīng)對(duì)象

    private var target: ButtonActionTarget? {

        get {

            return objc_getAssociatedObject(self.base, "buttonActionTarget") as? ButtonActionTarget

        }

        set {

            if let newValue = newValue {

                objc_setAssociatedObject(self.base, "buttonActionTarget", newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)

            }

        }

    }

如果你添加到self隧出,會(huì)發(fā)現(xiàn)并不生效踏志,也就是get的時(shí)候一直為nil,這是因?yàn)楸旧鞱ameSpace的作用域并不大胀瞪,因?yàn)槲覀冊(cè)跀U(kuò)展NameSpaceProtocol時(shí)只是臨時(shí)初始化了NameSpace针余,并沒(méi)有引用保存。

然后關(guān)于Self, .self, .Type的理解凄诞,大家可以執(zhí)行查詢圆雁,不過(guò)簡(jiǎn)單來(lái)說(shuō):

Self:用在協(xié)議中,代表的是協(xié)議自身或者實(shí)現(xiàn)者或者子類的類型

.self:用在哪代表的就是什么的自身帆谍,比如用在實(shí)例后面就是實(shí)例本身伪朽,類型后面就是類型本身

.Type:獲取調(diào)用者的類型

最后再次聲明下為什么我們要實(shí)現(xiàn)這個(gè)命名空間的效果:

1.在調(diào)用的時(shí)候,Xcode的快捷提示中不會(huì)顯示目標(biāo)的自帶方法汛蝙,不會(huì)產(chǎn)生混淆驱负,對(duì)新人來(lái)說(shuō)非常友好

2.加了一層命名空間,有效避免覆蓋重寫(xiě)的風(fēng)險(xiǎn)

3.更加優(yōu)雅患雇,許多知名的第三方也都這么做了跃脊,比如RxSwift

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市苛吱,隨后出現(xiàn)的幾起案子酪术,更是在濱河造成了極大的恐慌,老刑警劉巖翠储,帶你破解...
    沈念sama閱讀 218,525評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件绘雁,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡援所,警方通過(guò)查閱死者的電腦和手機(jī)庐舟,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,203評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)住拭,“玉大人挪略,你說(shuō)我怎么就攤上這事√显溃” “怎么了杠娱?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,862評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)谱煤。 經(jīng)常有香客問(wèn)我摊求,道長(zhǎng),這世上最難降的妖魔是什么刘离? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,728評(píng)論 1 294
  • 正文 為了忘掉前任室叉,我火速辦了婚禮睹栖,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘茧痕。我一直安慰自己磨淌,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,743評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布凿渊。 她就那樣靜靜地躺著,像睡著了一般缚柳。 火紅的嫁衣襯著肌膚如雪埃脏。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,590評(píng)論 1 305
  • 那天秋忙,我揣著相機(jī)與錄音彩掐,去河邊找鬼。 笑死灰追,一個(gè)胖子當(dāng)著我的面吹牛堵幽,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播弹澎,決...
    沈念sama閱讀 40,330評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼朴下,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了苦蒿?” 一聲冷哼從身側(cè)響起殴胧,我...
    開(kāi)封第一講書(shū)人閱讀 39,244評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎佩迟,沒(méi)想到半個(gè)月后团滥,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,693評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡报强,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,885評(píng)論 3 336
  • 正文 我和宋清朗相戀三年灸姊,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片秉溉。...
    茶點(diǎn)故事閱讀 40,001評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡力惯,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出召嘶,到底是詐尸還是另有隱情夯膀,我是刑警寧澤,帶...
    沈念sama閱讀 35,723評(píng)論 5 346
  • 正文 年R本政府宣布苍蔬,位于F島的核電站诱建,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏碟绑。R本人自食惡果不足惜俺猿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,343評(píng)論 3 330
  • 文/蒙蒙 一茎匠、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧押袍,春花似錦诵冒、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,919評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至圈盔,卻和暖如春豹芯,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背驱敲。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,042評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工铁蹈, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人众眨。 一個(gè)月前我還...
    沈念sama閱讀 48,191評(píng)論 3 370
  • 正文 我出身青樓握牧,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親娩梨。 傳聞我的和親對(duì)象是個(gè)殘疾皇子沿腰,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,955評(píng)論 2 355