Objective-C中的单例模式

 ​    ​单例模式算是设计模式中比较简单的一种吧,设计模式不是只针对某种编程语言,在C++, Java, PHP等其他OOP语言也有设计模式,笔者初接触设计模式是通过《漫谈设计模式》了解的。这本书中是用java写的,个人感觉拜读完这本书以后虽然有不理解的地方但还是收获蛮大的。上面提到依赖注入,控制翻转的时候,没大看懂,当学习到Strut,Spring, Hibernate的东西的时候才略懂略懂。不过在23种设计模式里面单例模式还是算比较好理解的, 那么在OC中又是怎么来表示单例模式的呢?下面会结合着lusashi的代码,理解一下OC中得单例模式。

    ​    ​首先得了解什么是单例模式,用大白话说,单例模式就是在程序中这个类只对应着一个实例,这就是单例模式,单例模式一般用全局静态对象来实现。下面我们会建立一个生成单例的类SingletonClass,在实现文件中定义各种方法来实现我们的单例模式。

 

    ​    ​1.单例模式一般用全局静态对象来实现,所以我们在SingletonClass.m中定义一个静态全局变量是少不了的

//定义静态全局变量
static SingletonClass *single = nil;

    ​    ​2.上面的静态变量是定义在实现文件中的所以是私有的,要想获取该类的实例得有个getInstance方法来获取实例,在给静态变量分配内存空间之前首先要判断是否已经分配过啦,确保单例,如果分配过了就不分配了。

//获取静态全局对象
+(id)getInstance
{
    //如果没有生成对象,则为静态全局变量分配内存
    if (single == nil) {
        single = [[SingletonClass alloc] init];
    }
    return single;
}

    ​    ​

    ​    ​3.为了防止用户通过alloc和new来实例化对象,因此我们要对类方法allcoWithZone进行重写

//防止通过alloc或者new来创建新的对象我们要重写allocWithZone
+(id)allocWithZone:(NSZone *)zone
{
    if (single == nil) {
        single = [[super allocWithZone:zone] init];
    }
    return single;
}

 

    ​    ​4.为了防止用户把单例进行深浅拷贝,我们需要重写copyWithZone方法和mutableCopyWithZone方法,在重写方法之前我们的单例类必须遵循协议NSCoping和NSMutableCoping协议

    ​    ​遵循协议代码如下:

@interface SingletonClass : NSObject<NSCopying, NSMutableCopying>

//单例中获取单例对象的方法
+(id) getInstance;

//单例测试方法
-(void) singletonFunction;

@end

 

    ​    ​重写copyWithZone方法

//为了防止通过copy来创建新的实例我们要重写copyWithZone;
-(id)copyWithZone:(NSZone *)zone
{
    return self;
}

 

    ​    ​重写mutableCopyWithZone方法

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

 

    ​5.防止用户把创建的单例dealloc,我们需要重写retainCount方法

//重写retainCount方法,防止被dealloc,返回最大值
-(NSUInteger) retainCount
{
    return NSUIntegerMax;
}

 

    ​6. 重写release,autorelease, retain方法

//重写retain,引用计数不变
-(id) retain
{
    return self;
}

//重写release
-(oneway void) release
{
}

//重写autorelease
-(id) autorelease
{
    return self;
}

 

    ​至此我们的单例模式基本创建完毕,下面开始我们的测试吧;

    ​在main函数中的代码如下:

//单例模式的测试
SingletonClass *single1 = [SingletonClass getInstance];
SingletonClass *single2 = [SingletonClass new];
SingletonClass *single3 = [[SingletonClass alloc] init];
SingletonClass *single4 = [single1 copy];
SingletonClass *single5 = [single1 mutableCopy];
SingletonClass *single6 = [single1 retain];
[single1 release];

[single1 singletonFunction];
NSLog(@"single_retainCount = %lu", single1.retainCount);

//输出地址
NSLog(@"getInstance     single1_P = %p", single1);
NSLog(@"new             single2_P = %p", single2);
NSLog(@"allo            single3_P = %p", single3);
NSLog(@"copy            single4_P = %p", single4);
NSLog(@"mutableCopy     single5_P = %p", single5);
NSLog(@"retain          single6_P = %p", single6);

 

    ​    ​运行结果如下:

2014-08-07 16:04:44.207 Memory[20664:303] singleton Ps: 我是单例模式中得测试方法!!
2014-08-07 16:04:44.207 Memory[20664:303] single_retainCount = 18446744073709551615
2014-08-07 16:04:44.207 Memory[20664:303] getInstance     single1_P = 0x100204690
2014-08-07 16:04:44.208 Memory[20664:303] new             single2_P = 0x100204690
2014-08-07 16:04:44.208 Memory[20664:303] alloC            single3_P = 0x100204690
2014-08-07 16:04:44.208 Memory[20664:303] copy            single4_P = 0x100204690
2014-08-07 16:04:44.209 Memory[20664:303] mutableCopy     single5_P = 0x100204690
2014-08-07 16:04:44.209 Memory[20664:303] retain          single6_P = 0x100204690

​    ​    ​单例的地址是不变的。

     上面是在非ARC模式下得单例模式,那么在ARC模式下我们应如何实现我们的单例模式呢,我们下面就会给出ARC下的单例模式,用下面的方法,因没有重写alloc,copy等方法,通过alloc还是可以给该对象分配一个新对象的,上面是线程不安全的,下面是线程安全的:

+ (id)sharedSingleton {
    static MySingleton *sharedSingleton = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        sharedSingleton = [[self alloc] init];
    });
    return sharedSingleton;
}
时间: 2024-09-19 08:58:42

Objective-C中的单例模式的相关文章

《Android 源码设计模式解析与实战》——第2章,第2.7节Android源码中的单例模式

2.7 Android源码中的单例模式在Android系统中,我们经常会通过Context获取系统级别的服务,如WindowsManagerService.ActivityManagerService等,更常用的是一个LayoutInflater的类,这些服务会在合适的时候以单例的形式注册在系统中,在我们需要的时候就通过Context的getSystemService(String name)获取.我们以LayoutInflater为例来说明,平时我们使用LayoutInflater较为常见的地

在php中实现单例模式

<?php /** * 设计模式之单例模式 * $_instance必须声明为静态的私有变量 * 构造函数和析构函数必须声明为私有,防止外部程序new * 类从而失去单例模式的意义 * getInstance()方法必须设置为公有的,必须调用此方法 * 以返回实例的一个引用 * ::操作符只能访问静态变量和静态函数 * new对象都会消耗内存 * 使用场景:最常用的地方是数据库连接. * 使用单例模式生成一个对象后, * 该对象可以被其它众多对象所使用. */ class Example { /

.Net中利用单例模式的提升系统整体性能

今天走在路上突然想起一个问题,之前没有深刻的注意到. 问题:如果在WebForm页面周期中如果 一个对象频繁创建,请求结束后又进行销毁肯定会对系统产生一定的性能影响,我们都知道,当我们每次new 一个对象时,其都在内存中指定的堆上分配一块空间,那么创建这个对象然后销毁(Response.End()执行后) 这个对象肯定要消耗服务器的时间及空间(尤其是并行请求数量很大的时候,撑破内存也不是不可能),因为 每个请求都会建这个对象.然后大家有没有注意到,有些类可能是这样的,举个例子: public c

iOS App开发中使用设计模式中的单例模式的实例解析_IOS

一.单例的作用顾名思义,单例,即是在整个项目中,这个类的对象只能被初始化一次.它的这种特性,可以广泛应用于某些需要全局共享的资源中,比如管理类,引擎类,也可以通过单例来实现传值.UIApplication.NSUserDefaults等都是IOS中的系统单例. 二.单例模式的两种写法 1,常用写法 #import "LGManagerCenter.h" static LGManagerCenter *managerCenter; @implementation LGManagerCen

PHP中数据库单例模式的实现代码分享_php实例

首先我们要知道明确单例模式这个概念,那么什么是单例模式呢? 单例模式顾名思义,就是只有一个实例. 作为对象的创建模式, 单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类我们称之为单例类. 单例模式的要点有三个: 一是某个类只能有一个实例: 二是它必须自行创建这个实例: 三是它必须自行向整个系统提供这个实例. 下面我们讨论下为什么要使用PHP单例模式? 多数人都是从单例模式的字面上的意思来理解它的用途, 认为这是对系统资源的节省, 可以避免重复实例化, 是一种"计划

简单讲解在Java编程中实现设计模式中的单例模式结构_java

1. 模式介绍 模式的定义 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 模式的使用场景 确保某个类有且只有一个对象的场景,例如创建一个对象需要消耗的资源过多,如要访问 IO 和数据库等资源. 2. UML类图 角色介绍: (1)Client : 高层客户端. (2)Singleton : 单例类. 3. 模式的简单实现 public class Singleton { private static Singleton intance; private Singleton(

java中的单例模式

    Java Singleton模式主要作用是保证在Java应用程序中,一个类Class只有一个实例存在. 使用Singleton的好处还在于可以节省内存,因为它限制了实例的个数,有利于Java垃圾回收.    我们在浏览BBS.SNS网站的时候,常常会看到"当前在线人数"这样的一项内容.对于这样的一项功能,我们通常的做法是把当前的在线人数存放到一个内存.文件或者数据库中,每次用户登录的时候,就会马上从内存.文件或者数据库中取出,在其基础上加1后,作为当前的在线人数进行显示,然后再

详解Ruby设计模式编程中对单例模式的运用_ruby专题

简介      单例模式是设计模式中最简单的形式之一.这一模式的目的是使得类的一个对象成为系统中的唯一实例.要实现这一点,可以从客户端对其进行实例化开始.因此需要用一种只允许生成对象类的唯一实例的机制,"阻止"所有想要生成对象的访问.使用工厂方法来限制实例化过程.这个方法应该是静态方法(类方法),因为让类的实例去生成另一个唯一实例毫无意义. 要点      显然单例模式的要点有三个:一是某个类只能有一个实例:二是它必须自行创建这个实例:三是它必须自行向整个系统提供这个实例.      

浅析Java设计模式编程中的单例模式和简单工厂模式_java

单例模式动机 有时候只有一个类的实例是很重要的.比如,一个系统应该只有一个窗口管理实例. 单例模式是最简单设计模式:类负责实例化自己,确保只有一个实例,并且提供一个访问这个实例的入口. 目的 1. 确保只有一个实例被创建. 2. 提供访问这个实例的入口. 使用final确保被创建一次,private的构造函数确保不被实例化.public的getInstance方法确保外部能够访问.下面是饿汉模式: public class Singleton { private static final Sin