iOS中的單例模式

来源:http://www.cnblogs.com/chao8888/archive/2016/06/14/5585845.html
-Advertisement-
Play Games

單例模式 概念相關 單例模式 在程式運行過程,一個類只有一個實例 使用場合 在整個應用程式中,共用一份資源(這份資源只需要創建初始化1次) static static關鍵字會在聲明變數的時候分配記憶體,在程式運行期間只分配一次記憶體。之後再訪問時,實際都是在訪問原先分配的記憶體 如果使用static來修飾 ...


單例模式


概念相關

  • 單例模式
  • 在程式運行過程,一個類只有一個實例
  • 使用場合
  • 在整個應用程式中,共用一份資源(這份資源只需要創建初始化1次)

static


  • static關鍵字會在聲明變數的時候分配記憶體,在程式運行期間只分配一次記憶體。之後再訪問時,實際都是在訪問原先分配的記憶體
  • 如果使用static來修飾局部變數,那麼局部變數在代碼塊結束後將不會回收,下次使用保持上次使用後的值。
  • 如果使用static來修飾全局變數,那麼表示該全局變數只在本文件中有效,外界無法使用extern來引用。static變數的作用域被限制在定義變數的當前文件中,其它文件是不能訪問的。

ARC實現單例


步驟

01 在類的內部提供一個static修飾的全局變數
02 提供一個類方法,方便外界訪問
03 重寫+allocWithZone方法,保證永遠都只為單例對象分配一次記憶體空間
04 嚴謹起見,重寫-copyWithZone方法和-MutableCopyWithZone方法

源碼


static ZCTools *_instance;

+ (instancetype)shareTools
{
    return [[self alloc]init];
}

+ (instancetype)allocWithZone:(struct _NSZone *)zone
{
    @synchronized(self) {
        if (_instance == nil) {
            _instance = [super allocWithZone:zone];
        }
    }
    return _instance;
}

- (nonnull id)copyWithZone:(nullable NSZone *)zone
{
    return _instance;
}

- (nonnull id)mutableCopyWithZone:(nullable NSZone *)zone
{
    return _instance;
}

代碼分析

//提供一個static修飾的全局變數,強引用著已經實例化的單例對象實例
static ZCTools *_instance;

//類方法,返回一個單例對象
+ (instancetype)shareTools
{
     //註意:這裡建議使用self,而不是直接使用類名Tools(考慮繼承)
    return [[self alloc]init];
}

//保證永遠只分配一次存儲空間
+(instancetype)allocWithZone:(struct _NSZone *)zone
{
    //使用GCD中的一次性代碼
//    static dispatch_once_t onceToken;
//    dispatch_once(&onceToken, ^{
//        _instance = [super allocWithZone:zone];
//    });

    //使用加鎖的方式,保證只分配一次存儲空間
    @synchronized(self) {
        if (_instance == nil) {
            _instance = [super allocWithZone:zone];
        }
    }
    return _instance;
}
/*
1. mutableCopy 創建一個新的可變對象,並初始化為原對象的值,新對象的引用計數為 1;
2. copy 返回一個不可變對象。分兩種情況:(1)若原對象是不可變對象,那麼返回原對象,並將其引用計數加 1 ;(2)若原對象是可變對象,那麼創建一個新的不可變對象,並初始化為原對象的值,新對象的引用計數為 1。
*/
//讓代碼更加的嚴謹
- (nonnull id)copyWithZone:(nullable NSZone *)zone
{
//    return [[self class] allocWithZone:zone];
    return _instance;
}

- (nonnull id)mutableCopyWithZone:(nullable NSZone *)zone
{
    return _instance;
}

MRC實現單例


步驟

01 在類的內部提供一個static修飾的全局變數
02 提供一個類方法,方便外界訪問
03 重寫+allocWithZone方法,保證永遠都只為單例對象分配一次記憶體空間
04 嚴謹起見,重寫-copyWithZone方法和-MutableCopyWithZone方法
05 重寫release方法
06 重寫retain方法
07 建議在retainCount方法中返回一個最大值

配置MRC環境

01 註意ARC不是垃圾回收機制,是編譯器特性
02 配置MRC環境:build setting ->搜索automatic ref->修改為NO

源碼

static ZCTools *_instance;

+ (instancetype)shareTools
{
    return [[self alloc]init];
}

+ (instancetype)allocWithZone:(struct _NSZone *)zone
{
    @synchronized(self) {
        if (_instance == nil) {
            _instance = [super allocWithZone:zone];
        }
    }
    return _instance;
}

- (nonnull id)copyWithZone:(nullable NSZone *)zone
{
    return _instance;
}

- (nonnull id)mutableCopyWithZone:(nullable NSZone *)zone
{
    return _instance;
}

- (oneway void)release
{
}

- (instancetype)retain
{
    return _instance;
}

- (NSUInteger)retainCount
{
    return MAXFLOAT;
}

代碼分析

//提供一個static修飾的全局變數,強引用著已經實例化的單例對象實例
static ZCTools *_instance;

//類方法,返回一個單例對象
+ (instancetype)shareTools
{
     //註意:這裡建議使用self,而不是直接使用類名Tools(考慮繼承)

    return [[self alloc]init];
}

//保證永遠只分配一次存儲空間
+ (instancetype)allocWithZone:(struct _NSZone *)zone
{
    //使用GCD中的一次性代碼
//    static dispatch_once_t onceToken;
//    dispatch_once(&onceToken, ^{
//        _instance = [super allocWithZone:zone];
//    });

    //使用加鎖的方式,保證只分配一次存儲空間
    @synchronized(self) {
        if (_instance == nil) {
            _instance = [super allocWithZone:zone];
        }
    }
    return _instance;
}

//讓代碼更加的嚴謹
- (nonnull id)copyWithZone:(nullable NSZone *)zone
{
//    return [[self class] allocWithZone:zone];
    return _instance;
}

- (nonnull id)mutableCopyWithZone:(nullable NSZone *)zone
{
    return _instance;
}

//在MRC環境下,如果用戶retain了一次,那麼直接返回instance變數,不對引用計數器+1
//如果用戶release了一次,那麼什麼都不做,因為單例模式在整個程式運行過程中都擁有且只有一份,程式退出之後被釋放,所以不需要對引用計數器操作
- (oneway void)release
{
}

- (instancetype)retain
{
    return _instance;
}

//慣用法,有經驗的程式員通過列印retainCount這個值可以猜到這是一個單例
- (NSUInteger)retainCount
{
    return MAXFLOAT;
}

通用版本


  • 可以考慮一份單例代碼在ARC和MRC環境下都適用
    (使用條件編譯來判斷當前項目環境是ARC還是MRC)
#if __has_feature(objc_arc)
//如果是ARC,那麼就執行這裡的代碼1
#else
//如果不是ARC,那麼就執行代理的代碼2
#endif
  • 在項目裡面往往需要實現很多的單例,比如下載、網路請求、音樂播放等等,單例是否可用繼承呢?
  • 單例是不可繼承,想一勞永逸可使用帶參數的巨集定義

  • 使用帶參數的巨集完成通用版單例模式代碼

    • 註意條件編譯的代碼不能包含在巨集定義裡面
    • 巨集定義的代碼只用寫一次,之後直接拖到項目中用就OK

#define SingleH(name)  +(instancetype)share##name;

//ARC
#if __has_feature(objc_arc)
#define SingleM(name) static id _instance;\
+(instancetype)allocWithZone:(struct _NSZone *)zone\
{\
    static dispatch_once_t onceToken;\
    dispatch_once(&onceToken, ^{\
        _instance = [super allocWithZone:zone];\
    });\
    return _instance;\
}\
+(instancetype)share##name\
{\
    return [[self alloc]init];\
}\
-(id)copyWithZone:(NSZone *)zone\
{\
    return _instance;\
}\
\
-(id)mutableCopyWithZone:(NSZone *)zone\
{\
    return _instance;\
}

#else
//MRC
#define SingleM(name) static id _instance;\
+(instancetype)allocWithZone:(struct _NSZone *)zone\
{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [super allocWithZone:zone];\
});\
return _instance;\
}\
+(instancetype)share##name\
{\
return [[self alloc]init];\
}\
-(id)copyWithZone:(NSZone *)zone\
{\
return _instance;\
}\
\
-(id)mutableCopyWithZone:(NSZone *)zone\
{\
return _instance;\
}\
-(oneway void)release\
{\
}\
\
-(instancetype)retain\
{\
    return _instance;\
}\
\
-(NSUInteger)retainCount\
{\
    return MAXFLOAT;\
}
#endif

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

-Advertisement-
Play Games
更多相關文章
  • 1、++i 的用法(以 a=++i ,i=2 為例) 先將 i 值加 1 (也就是 i=i+1 ),然後賦給變數 a (也就是 a=i ), 則最終 a 值等於 3 , i 值等於 3 。 所以 a=++i 相當於 i=i+1 ,a=i2、i++ 的用法(以 a=i++ ,i=2 為例) 先將 i ...
  • 10種排序演算法總結 排序演算法有很多,所以在特定情景中使用哪一種演算法很重要。為了選擇合適的演算法,可以按照建議的順序考慮以下標準: (1)執行時間 (2)存儲空間 (3)編程工作 對於數據量較小的情形,(1)(2)差別不大,主要考慮(3);而對於數據量大的,(1)為首要。 主要排序法有: 一、冒泡(Bu ...
  • 一、標識符和關鍵字 關鍵字: 就是被java語言賦予了特殊含義的單詞。 特點就是所有的關鍵字都是小寫。 標識符: 就是給包、類、介面、方法、變數名起的名字。 規則:1.以數字、字母、_以及$符號組成。 2,不能以數字開頭,不能以空格分開,不能是JAVA中的關鍵字,區別大小寫如Test 和test是2 ...
  • list是一種有序的集合,可以隨時添加和刪除其中的元素。 >>> classmates = ['Michael', 'Bob', 'Tracy'] >>> classmates ['Michael', 'Bob', 'Tracy'] 獲取 >>> classmates[2] 'Tracy' 當索引超 ...
  • 我之前想寫路由器的密碼暴力破解器,我手上只有極路由,發現極路由有安全限制,只能允許連續10次密碼錯誤,所以我改拿博客園練手。 博客園的博客有個功能是給博文設置一個密碼,輸入正確的密碼才能看到文章的內容。經過測試發現這個密碼驗證功能,既沒有驗證碼也沒有提交頻率的限制, 要寫這個針對博客園的密碼暴力破解 ...
  • 本文適用於所有鍵盤類職業,比如程式員,比如作家,比如編輯,比如水軍,等等... 程式員買得起鍵盤嗎?買得起。 程式員買得起好的機械鍵盤嗎?也買得起。 不過大部分of他們是不會很快決定買的,包括我。因為打從一開始接觸電腦,不管是家裡還是學校甚至網吧,我們用的鍵盤也就是隨便玩玩的,並不會多在意,而等到自 ...
  • 1. download cas 4.2.2 from https://github.com/apereo/cas/releases 2. eclipse import cas 4.2.2 eclipse install SpringSource Update Site for Gradle Inte ...
  • 前段時間因為寫一個業務,對mysql的插入量非常大,在idea開發環境,使得idea一直爆出outofmemry提示,並導致old 區一直漲。使用Mongodb 作為存儲後,不僅插入速度大大提升,idea會卡死的頑疾也解決了。 Mongodb 能夠非常簡單的整合進項目 下麵是我的整合過程 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...