在Swift开发中,我们对于跨类调用的变量常量,因为并没有OC中使用的全局头文件中写宏的形式,我们一般采用在类外定义全局变量/常量的形式来跨类调用。而问题在于目前写的项目需要在新添加的OC写的功能模块中调用Swift的全局变量,这样编译器是没办法帮你调到的。为了解决这个问题,我考虑来写一个Swift单例来保存全局变量,并由OC的类来调取数据。

  ps:差点忘记文章的初衷了。如果大家有好的用OC调取Swift的全局变量的方法,希望可以提供方法或网址,感激不尽。

  

  在iOS设计中,单例这种设计模式经常用到,也是是设计模式中最简单的一种,甚至有些模式大师都不称其为模式,称其为一种实现技巧。原因是设计模式讲究对象之间的关系的抽象,而单例模式只有自己一个对象,与其他对象之间并不抽象。但不可否认的是,单例已经方方面面的融入到我们的项目中来了,极为方便。

  那我们究竟在什么时候需要使用单例设计模式呢?

  顾名思义,当你只需要一个实例的时候需要使用单例,如UIApplication.sharedApplication() 等 ,就好比windows的任务管理器,回收站等等都是只能同时存在一个。

  

  一.OC中的单例设计模式有三种。简单介绍一下。

  首先,我们要创建一个继承于NSObject的类。在这个类的 .h头文件中,我们声明一个加方法。命名一般如下:

+ (SingleInstance *)sharedInstance;

  随后在.m中实现这个方法。这里有三种形式,相对来说,安全和效率等级越来越高。

  1.直接创建

 static SingleInstance *singleOne;

 + (SingleInstance *)sharedInstance {

     if (!singleOne) {

         singleOne = [[SingleInstance alloc] init];
} return singleOne;
}

  非常好理解。创建一个静态的指针;如果指针没有指向对象,那么创一个对象并让指针持有该对象;最后返回这个指针持有的对象。因为静态,所以指针持有的对象不会被释放,也就保证了每次调取这个方法都只会返回一个对象,实现了单例的目的。

  不难看出,这种做法虽然简单明了,并且逻辑上并没有什么错误,但它并没有考虑线程安全的问题。在我们学习了多线程之后,我们就会考虑这段代码的安全性问题了。假设两条线程同时调用这段代码,并且若此时对象并未被创建,那么会创建几个对象?而若创建了多个对象,很明显与单例设计思想相违背。这样我们就有了以下的做法。

  2.加互斥锁

  根据我们刚才的分析,我们很自然会想到这样的做法:既然要防止多个线程同时访问这段代码,我们加个互斥锁,让线程们排队访问,不就解决了吗?

 static SingleInstance *singleTwo;

 + (SingleInstance *)sharedInstance {

     @synchronized (self) {

         if (!singleTwo) {

             singleTwo = [[SingleInstance alloc] init];
}
} return singleTwo;
}

  更加简单粗暴。一言不合就上锁,这段代码只能排队执行,一个一个来。这样完美的保证了单例只有一个对象的要求。

  但是更大的问题出现了,我确实只执行了一次,但是在第二次使用这个方法的时候,if判断语句还是需要走啊?大家创建完对象仍然一点一点排队走,太影响效率了吧?

  所以,强迫症如我们程序猿,是不允许这样耗费线程资源还执行效率低下的事情发生的。更好的方法来了。

  ps:互斥锁相关再解释一下,与多线程并发有关,意义基本在于多条线程访问同一时间访问同一段代码的时候要按顺序执行,不能并发。这样能有效解决多线程的安全问题。详情请百度。

  3.使用GCD技术

 static SingleInstance *singleThree;

 + (SingleInstance *)sharedInstance {

     static dispatch_once_t onceToken;

     dispatch_once(&onceToken, ^{

         singleThree = [[SingleInstance alloc] init];
}); return singleThree;
}

  使用 dispatch_once 这个函数来创建对象,这样就保证了只有一个对象。问题解决。

  ps:dispatch_once是GCD中的一个对象,这个函数可以保证不管被调用几次,内部block只会执行一次,来保证代码执行的唯一性。

  

  综上,OC使用的单例方式一般就这三种,而我们使用的一般是第三种,安全性更高,执行效率更好。

  

  二.下面看看swift中的几种实现方式:

  1.

     class SingleOne {

         //单例
static let shareSingleOne = SingleOne()
} let singleOne = SingleOne.shareSingleOne

  ps:代码编辑器中没有找到Swift语言,很尴尬..用OC凑活看吧

  一句话搞定,静态常量。所有地方用到的都是同一个。调用直接使用类名调取常量名即可。

  2.

     class SingleTwo {

         //单例
class func shareSingleTwo() -> SingleTwo { struct Singleton{ static var onceToken: dispatch_once_t = static var single: SingleTwo?
} dispatch_once(&Singleton.onceToken, { Singleton.single = shareSingleTwo()
}) return Singleton.single!
}
} let singleTwo = SingleTwo.shareSingleTwo()

  本质上与OC第三种方法一样,都是使用dispatch_once保证其中的代码只执行一次。中间用了一个结构体来保存两个静态变量,也可以不使用这个结构体。

  3.

 import Foundation

 //全局的常量
let single = SingleThree() class SingleThree { class var sharedInstance : SingleThree { return single
}
} let singleThree = SingleThree.sharedInstance

  4.

 import Foundation

 class SingleFour {

     static var sharedInstance : SingleFour {

         struct Static {

             static let instance: SingleFour = SingleFour()
} return Static.instance
}
} let singleFour = SingleFour.sharedInstance

  这次我们是在方法内定义静态常量。

  

  大概的创建单例的方式就是这些了。希望大家能有所收获。

Objective-C和Swift实现单例的几种方式的更多相关文章

  1. swift实现单例的四种方式

    单例模式 单例模式是设计模式中最简单的一种,甚至有些模式大师都不称其为模式,称其为一种实现技巧,因为设计模式讲究对象之间的关系的抽象,而单例模式只有自己一个对象. 当你只需要一个实例的时候需要使用单例 ...

  2. Java实现单例的5种方式

    1. 什么是单例模式 单例模式指的是在应用整个生命周期内只能存在一个实例.单例模式是一种被广泛使用的设计模式.他有很多好处,能够避免实例对象的重复创建,减少创建实例的系统开销,节省内存. 2. 单例模 ...

  3. java单例的几种实现方法

    java单例的几种实现方法: 方式1: public class Something { private Something() {} private static class LazyHolder ...

  4. iOS单例的两种实现

    单例模式算是开发中比较常见的一种模式了.在iOS中,单例有两种实现方式(至少我目前只发现两种).根据线程安全的实现来区分,一种是使用@synchronized,另一种是使用GCD的dispatch_o ...

  5. objc单例的两种安全实现方案

    所有转出博客园,请您注明出处:http://www.cnblogs.com/xiaobajiu/p/4122034.html objc的单例的两种安全实现方案 首先应该知道单例的实现有两大类,一个是懒 ...

  6. asp.net mvc表单提交的几种方式

    asp.net MVC中form提交和控制器接受form提交过来的数据 MVC中form提交和在控制器中怎样接受 1.cshtml页面form提交2.控制器处理表单提交数据4种方式方法1:使用传统的R ...

  7. Form 表单提交的几种方式

    简单的总结一下form表单提交的几种方式:1.最简单的方式 就用form的submit提交方式,这种提交方式是不需要回调函数的   这种方式最近到一个form提供action路径后台接受就可以< ...

  8. iOS开发——Swift篇&单例的实现

    Swift实现单例模式 Swift实现单例模式 由于Swift语言弱化了struct和class之间的界限,这里我分别给出自己写的两种的单例实现 class版本: class SwiftSinglet ...

  9. swift学习 - 单例实现(singleton)

    swift中实现单例的方式 class LGConfig: NSObject { static let instance = LGConfig() private override init() { ...

随机推荐

  1. Linux_jdk

    先查看下 yum list java* yum install java-1.7.0-openjdk* -y 环境变量应该是会自动配置的 或者手动配置编辑/etc/profile #vi /etc/p ...

  2. [帖子收集]环境光遮蔽(Ambient Occlusion)

    环境光遮蔽,效果示例图 图片左边是一条龙的简单模型,呈现在一个均匀照明的环境中.尽管模型中有一些明暗不同的区域,但大部分光照都是均匀的.虽然模型有着相当复杂的几何形状,但看上去比较光滑平坦,没有明显的 ...

  3. php添加扩展插件

    给PHP安装扩展的方式有好多 一.重新编译 进入PHP源码目录./configure --prefix=/usr/local/php ...[其他编译参数] 二.通过phpize添加扩展 进入PHP源 ...

  4. UIBezierPath与CAShapeLayer结合画扇形

    /*让半径等于期望半径的一半 lineWidth等于期望半径 就可以画圆*/ 可以看出layer的走势是从圆边的中间一半在圆外 一半在圆内 因此让半径等于期望半径的一半 lineWidth等于期望半径 ...

  5. maven 3.3.9-bin 和 maven 3.3.9-src 的区别 以及 maven安装包的 .tar.gz后缀与.zip 后缀的区别

    (maven 3.3.9-bin)一个是class的文件包,由java文件编译成的,(maven 3.3.9-src )一个是java文件包即是源码(.tar.gz后缀)是linux的压缩包,(.zi ...

  6. Spring 配置文件XML -- <beans>中属性概述

    beans : xml文件的根节点. xmlns : XML NameSpace的缩写,因为XML文件的标签名称都是自定义的,自己写的和其他人定义的标签很有可能会重复命名,而功能却不一样,所以需要加上 ...

  7. linux的用户,群组和其他用户

    1 linux 安全模型:(多任务,多用户的操作系统) 1)使用user和group控制使用者对文件的存储权限. 2)用户使用账户和口令登录linux 3)每个文件都有owner(创建者),owner ...

  8. 中小型公司数据仓库搭建——以mysql为例

    为了方便公司的数据分析平台的独立运行和数据挖掘的探索,今年上半年在公司搭建了支持数据平台和数据挖掘的数据仓库:现就数据仓库的创建工作总结如下,供大家参考: 首先介绍下数据仓库搭建的缘由: 公司创建两年 ...

  9. redis7--hash set的操作

    数据类型Hash(1)介绍hash数据类型存储的数据与mysql数据库中存储的一条记录极为相似Redis本身就类似于Hash的存储结构,分为key-value键值对,实际上它的Hash数据就好像是在R ...

  10. FileZilla客户端源码解析

    FileZilla客户端源码解析 FTP是TCP/IP协议组的协议,有指令通路和数据通路两条通道.一般来说,FTP标准命令TCP端口号是21,Port方式数据传输端口是20. FileZilla作为p ...