Swift、Objective-C 單例模式 (Singleton)

来源:http://www.cnblogs.com/silence-cnblogs/archive/2017/04/27/6776217.html
-Advertisement-
Play Games

Swift、Objective C 單例模式 (Singleton) 本文的單例模式分為 嚴格單例模式 和 不嚴格單例模式 。單例模式要求一個類有一個實例,有公開介面可以訪問這個實例。嚴格單例模式,要求一個類只有一個實例;不嚴格單例模式,可以創建多個實例。 有的類只能有一個實例,例如 UIAppli ...


Swift、Objective-C 單例模式 (Singleton)

本文的單例模式分為嚴格單例模式不嚴格單例模式。單例模式要求一個類有一個實例,有公開介面可以訪問這個實例。嚴格單例模式,要求一個類只有一個實例;不嚴格單例模式,可以創建多個實例。

有的類只能有一個實例,例如 UIApplication,通過 shared 屬性訪問唯一的實例,屬於嚴格單例模式。有用戶登錄功能的 App 中,如果當前用戶的數據模型與其他用戶的數據模型不同,那麼當前用戶的類也應該用嚴格單例模式。在邏輯上,當前用戶只有一個,只能有一個實例;這樣可以在各個地方訪問當前用戶的數據。如果當前用戶的數據模型與其他用戶的數據模型相同,則應用不嚴格單例模式。可以給其他用戶創建實例,同時也可以在各個地方訪問當前用戶的數據。

Swift 實現

嚴格單例模式

大多數 Objective-C 的類都繼承自 NSObject,而 Swift 的類可以繼承自 NSObject 或者不繼承。

繼承自 NSObject

class SingletonClass: NSObject {

    static let shared = SingletonClass()
    
    // Make sure the class has only one instance
    // Should not init or copy outside
    private override init() {}
    
    override func copy() -> Any {
        return self // SingletonClass.shared
    }
    
    override func mutableCopy() -> Any {
        return self // SingletonClass.shared
    }
    
    // Optional
    func reset() {
        // Reset all properties to default value
    }
}

靜態屬性 shared 持有唯一的實例,對外公開。

重載 init() 方法,使其對外不可見,不可以在外部調用,防止在外部創建實例。

重載 copy()、mutableCopy() 方法,返回 self,防止在外部複製實例。這裡也可以返回 SingletonClass.shared,效果是一樣的,因為只有一個實例。只有 shared 能調用 copy()、mutableCopy() 方法,那麼 self 就是 shared。寫 self,代碼比較簡潔。

單例一旦創建,一直持有,不能手動銷毀,但可以重置數據。如果需要的話,可以添加一個重置數據的方法 reset()。例如,當前用戶退出登錄,需要把當前用戶實例的所有屬性重置為預設值,防止數據錯誤。

不繼承自 NSObject

class SingletonClass2 {
    
    static let shared = SingletonClass2()
    
    // Make sure the class has only one instance
    // Should not init outside
    private init() {}
    
    // Optional
    func reset() {
        // Reset all properties to default value
    }
}

不繼承自 NSObject 的類沒有 copy()、mutableCopy() 方法,不需要重載。其他同上。

不嚴格單例模式

把重載的 init() 方法去掉,或者把 private 去掉,即可創建多個實例。如果繼承自 NSObject,重載 copy()、mutableCopy() 方法:創建新實例,傳遞數據給新實例,返回新實例。其他與嚴格單例模式相同。

不嚴謹的重置數據方法

如果單例有很多屬性,重置數據需要把每個屬性都變成預設值,則 reset() 方法要寫很多。有一種不嚴謹的重置數據方法:重新生成一個實例並賦值給持有實例的靜態變數。

class SingletonClass3 {
    
    private static var _shared = SingletonClass3()
    
    static var shared: SingletonClass3 {
        return _shared
    }
    
    private init() {}
    
    // Not safe
    // We can obtain more than one instance outside with this function
    func reset() {
        SingletonClass3._shared = SingletonClass3()
    }
}

如果在外部訪問單例都通過 shared 屬性,這麼寫不會出錯。然而,如果外部持有單例,就有可能出錯。

let s = SingletonClass3.shared
s.reset()
print(s === SingletonClass3.shared) // false

以上會輸出 false,s 在重置之後,和 SingletonClass3.shared 不是同一個對象。因此,這樣的重置數據方法不嚴謹。還是要老老實實把每個屬性賦為預設值。

Objective-C 實現

嚴格單例模式

.h 文件

@interface SingletonClassOC : NSObject

+ (nonnull instancetype)shared;

@end

.m 文件

@implementation SingletonClassOC

static id _shared;

+ (nonnull instancetype)shared {
    if (!_shared) {
        _shared = [[self alloc] init];
    }
    return _shared;
}

+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _shared = [super allocWithZone:zone];
    });
    return _shared;
}

- (instancetype)init {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _shared = [super init];
    });
    return _shared;
}

- (id)copyWithZone:(NSZone *)zone {
    return self; // _shared
}

- (id)mutableCopyWithZone:(NSZone *)zone {
    return self; // _shared
}

// Optional
- (void)reset {
    // Reset all properties to default value
}

@end

靜態變數 _shared 持有唯一的實例,通過 shared 方法對外公開。在這方法中,如果發現 _shared 沒有初始化,則進行初始化。方法返回值的 nonnull 表示返回值不為空,這樣寫方便 Swift 調用。不加 nonnull,shared 方法在 Swift 中的返回值是 optional 類型 (SingletonClassOC?),不方便使用;加上 nonnull 則為 SingletonClassOC 類型。

重載 allocWithZone: 和 init 方法,由 dispath_once 保證父類方法只執行一次並給 _shared 賦值,返回 _shared。外部調用初始化方法總是獲得 _shared 持有的唯一實例。

重載 copyWithZone: 和 mutableCopyWithZone: 方法,返回 self,防止在外部複製實例。由於只有一個實例 _shared,只能由 _shared 調用 copy、mutableCopy 方法,那麼 self 就是 _shared。

如果需要,用 reset 方法重置數據。

不嚴格單例模式

.h 文件

@interface SingletonClassOC2 : NSObject

+ (nonnull instancetype)shared;

@end

.m 文件

@implementation SingletonClassOC2

+ (nonnull instancetype)shared {
    static id _shared;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _shared = [[self alloc] init];
    });
    return _shared;
}

- (id)copyWithZone:(NSZone *)zone {
    SingletonClassOC2 *copiedObject = [[self.class allocWithZone:zone] init];
    // Copy data to copiedObject
    return copiedObject;
}

- (id)mutableCopyWithZone:(NSZone *)zone {
    SingletonClassOC2 *copiedObject = [[self.class allocWithZone:zone] init];
    // Copy data to copiedObject
    return copiedObject;
}

- (void)reset {
    // Reset all properties to default value
}

@end

通過 shared 方法返回給外部訪問的實例。靜態變數 _shared 不需要被多個方法使用,因此寫在 shared 方法里即可。由 dispatch_once 保證 _shared 只初始化一次。

沒重載 allocWithZone: 和 init 方法,外部可以創建新實例。init 方法可以根據需要重載,在這裡進行初始化參數等,和非單例模式的類一樣,調用父類方法,最後返回 self。

重載 copyWithZone: 和 mutableCopyWithZone: 方法,在裡面創建新實例,傳遞數據給新實例,返回新實例。外部可以通過 copy 或 mutableCopy 方法複製實例。

SDWebImage 中就用到不嚴格單例模式

NSObject 的類方法 new 相當於 alloc 和 init 方法。

轉載請註明出處:http://www.cnblogs.com/silence-cnblogs/p/6776217.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 收錄待用,修改轉載已取得 "騰訊雲" 授權 最新騰訊雲技術公開課直播,提問騰訊W3C代表,如何從小白成為技術專家? "點擊瞭解活動詳情" 。 作者 |陳澤濱 編輯 | 顧鄉 從事 "iOS" 開發幾年,越來越發現,我們的開發者往往聚焦在程式的開發,如何實現一個功能,如何寫好一行代碼。但對iOS應用開 ...
  • 在AndroidManifest.xml裡面先添加許可權訪問網路的許可權: <uses-permission android:name="android.permission.INTERNET"/> 效果圖如下: 下麵是主要代碼: 1 package com.hb.neting; 2 3 import ...
  •   網上有很多關於實現用xib自定義View,那我為什麼還要寫呢?第一,我用他們的方法都沒有實現。第二,用xib遇到了很多問題,想分享給大家。 用xib自定義View:FHCustomView 1.新建UIView 2.建一個 同名 xib : FHCustomView.xib ...
  • 一、View基礎知識 IDE——Integrated Development Environment 集成開發環境,具有多項功能的綜合性編輯環境 XML——Extensible Markup Language 可拓展標記語言,用來描述Android外觀建立佈局。 View—— 屏幕上用來展示內容的矩 ...
  • 問題情形:1.電量跳變到50%或偏差過大2.電池ID腳接地3.溫度腳來判斷電池是否在位 推測原因:判斷電池在位的方法: 先在設備樹查找bpd是否有值 找不到的話,用代碼的定義(通過溫度腳判斷) 把參數配置到結構體中。計算電池容量: 如果電池不在位並且或為假電池,返回預設電量(50)電池在位檢測函數: ...
  • service粘性等的那4種方式試了,三星的可以,小米老款手機可以,新款不行,華為新款也不行,還有魅族什麼的,都不行,新款的手機上都有一個安全中心,只有在安全中心裡面添加上允許app自啟動才可以 怎麼破? 1.用jpush的這個問題就沒得破了 能做的措施他們都做了 定製的rom會有這個問題 沒得破 ...
  • 使用百度地圖有一段時間了,導航是一個一直困擾我的問題。今天剛發現百度地圖的導航SDK並不是對Android6.0版本不相容。而是對某一部分手機不相容,準確的說是對X64或X86的cpu不相容。下載百度的導航DEMO發現在libs文件加下只有一個armeabi文件夾。這裡面存放的是導航功能所需要的動態 ...
  • Unity編譯時找不到AndroidSDK的問題,Android SDK更新引入的問題。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...