简介
单例模式是设计模式中最简单的形式之一。这一模式的目的是使得类的一个对象成为系统中的唯一实例。要实现这一点,可以从客户端对其进行实例化开始。因此需要用一种只允许生成对象类的唯一实例的机制,“阻止”所有想要生成对象的访问。使用工厂方法来限制实例化过程。这个方法应该是静态方法(类方法),因为让类的实例去生成另一个唯一实例毫无意义。
要点
显然单例模式的要点有三个;一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。
从具体实现角度来说,就是以下三点:一是单例模式的类只提供私有的构造函数,二是类定义中含有一个该类的静态私有对象,三是该类提供了
singleton
class ClassVariableTester @@class_count = 0 def initialize @instance_count = 0 end def increment @@class_count = @@class_count + 1 @instance_count = @instance_count + 1 end def to_s "class count :#{@@class_count} -- instance count :#{@instance_count}" end end cv1 = ClassVariableTester.new cv1.increment cv1.increment puts("cv1:#{cv1}") cv2 = ClassVariableTester.new puts("cv2:#{cv2}") #cv1:class count :2 -- instance count :2 #cv2:class count :2 -- instance count :0
当创建了第二个对象时,@@class_count 为2,二@instance_count为0,因为类变量被所有实例所共享,党cv1.increment调用了两次以后@@class_count为2,创建第二个ClassVariableTester对象cv2的时候,共享了@@class_count,所以此时的@@class_count仍为2。
而实例变量只能为当前对象服务,所以实例对象cv2的@@instance_count为0
类变量的这种特性是一种单例模式
class SimpleLogger @@instance = SimpleLogger.new def self.get_instance @@instance end private_class_method :new end sl1 = SimpleLogger.get_instance sl2 = SimpleLogger.get_instance puts sl1 == sl2
结果为:true 。
采用一个类变量来保存仅有的一个类的实例,同时需要一个类方法返回这个单例实例。
但是通过SimpleLogger.new还是可以创建另一个实例对象,因此需要把着个new方法设为私有的。
sl3 = SimpleLogger.new private method `new' called for SimpleLogger:Class (NoMethodError) require 'singleton' class SimpleLogger include Singleton end #puts SimpleLogger.new sl1 = SimpleLogger.instance sl2 = SimpleLogger.instance puts sl1 == sl2
结果为:true
Ruby类库中提供了singleton,来简化单例类的创建。
混入Singleton,就省略了创建类变量,初始化单例实例,创建类级别的instance方法,以及将new设为私有。
通过SimpleLogger.instance来获取日志器的单例。
但是两种方式还是又差异的。
第一种方式称之为“勤性单例(eager instantiation)”。
在确实需要之前就创建了实例对象。
第二种方式称之为“惰性单例(lazy instantiation)”
在调用instance时才会去创建 。
但是这个Singleton不能真正的阻止任何事情,可以用过public_class_method改变new方法的为公用的。
打开类,设置new方法为public之后,就可以用SimpleLogger.new来创建对象了。
class SimpleLogger public_class_method :new end puts SimpleLogger.new
再来分两种情况:
(一)使用全局变量,尽量不要使用全局变量,因为全局变量是程序紧密的耦合在一起,
其实单例模式和全局变量的作用是一样的,
$logger = SimpleLogger.new
(二)使用类作为单例,
class SimpleLogger WARNING = 1 INFO = 2 def initialize(file) @@log = File.open(file, "w") @@level = WARNING end def self.warning(msg) puts @@level > WARNING @@log.puts(msg) if @@level > WARNING @@log.flush end def self.level @@level end def self.level=(new_level) @@level = new_level end end SimpleLogger.new("test.txt") puts SimpleLogger.level SimpleLogger.level = SimpleLogger::INFO puts SimpleLogger.level SimpleLogger.warning("warning")
实例
require 'rubygems' require 'watir' require 'singleton' class AutoTest include Singleton def OpenUrl(url) @browser= Watir::Browser.new @browser.goto(url) @url=url end def set_textarea(text) @browser.text_field(:id,'kw').set(text) end def click @browser.button(:id,'su').click end end test,test2 = AutoTest.instance test.OpenUrl('http://www.baidu.com') test.set_textarea('aslandhu') test.click
这里虽然创建了两个AutoTest实例,但是第二个实例其实为nil,也就是说并没有创建成功。
require 'rubygems' require 'watir' require 'singleton' require 'thread' class TestOneObj end class <<TestOneObj include Singleton def instance @browser= Watir::Browser.new self end def openurl(url) @browser.goto(url) end def set_textarea(text) @browser.text_field(:id,'kw').set(text) end def click @browser.button(:id,'su').click end end test = TestOneObj.instance test2 = TestOneObj.instance p test.inspect p test2.inspect test.openurl('www.baidu.com') test2.set_textarea('aslandhu') test.click
上面这段代码试图创建两个Browser对象,但事实上创建的两个对象均为同一个。虽然打开了两个IE窗口,但是对象还是一个,即test与test2是同一个对象。
本文向大家介绍举例讲解C#编程中对设计模式中的单例模式的运用,包括了举例讲解C#编程中对设计模式中的单例模式的运用的使用技巧和注意事项,需要的朋友参考一下 单例模式的介绍 说到单例模式,大家第一反应应该就是——什么是单例模式?,从“单例”字面意思上理解为——一个类只有一个实例,所以单例模式也就是保证一个类只有一个实例的一种实现方法罢了,下面给出单例模式的一个官方定义:确保一个类只有一个实例,并提供
本文向大家介绍详解C++设计模式编程中对访问者模式的运用,包括了详解C++设计模式编程中对访问者模式的运用的使用技巧和注意事项,需要的朋友参考一下 访问者模式(visitor),表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。访问者模式适用于数据结构相对稳定的系统。它把数据结构和作用于结构上的操作之间的耦合解脱开,使得操作集合可以相对自由地
本文向大家介绍设计模式中的观察者模式在Ruby编程中的运用实例解析,包括了设计模式中的观察者模式在Ruby编程中的运用实例解析的使用技巧和注意事项,需要的朋友参考一下 观察者模式(有时又被称为发布/订阅模式)是软件设计模式的一种。 在此种模式中,一个目标对象管理所有相依于它的观察者对象,并且在它本身的状态改变时主动发出通知。 这通常透过呼叫各观察者所提供的方法来实现。 实现观察者模式的时候要注意,
本文向大家介绍详解组合模式的结构及其在Ruby设计模式编程中的运用,包括了详解组合模式的结构及其在Ruby设计模式编程中的运用的使用技巧和注意事项,需要的朋友参考一下 定义:也叫合成模式,或者部分-整体模式,主要是用来描述部分与整体的关系,定义,将对象组合成树形结构以表示“部分-整体”的层次结构,使得用户对单个对象和组合对象的使用具有一致性。 类图: 角色说明: Componnent抽象构件角色:
本文向大家介绍Ruby设计模式编程中对外观模式的应用实例分析,包括了Ruby设计模式编程中对外观模式的应用实例分析的使用技巧和注意事项,需要的朋友参考一下 何为外观模式? 外观模式为子系统中一组不同的接口提供统一的接口。外观定义了上层接口,通过降低复杂度和隐藏子系统间的通信以及依存关系,让子系统更加易于使用。 比方说子系统中有一组不同的类,其中一些彼此依赖。这让客户端难以使用子系统
本文向大家介绍实例解析Ruby设计模式编程中Strategy策略模式的使用,包括了实例解析Ruby设计模式编程中Strategy策略模式的使用的使用技巧和注意事项,需要的朋友参考一下 今天你的leader兴致冲冲地找到你,希望你可以帮他一个小忙,他现在急着要去开会。要帮什么忙呢?你很好奇。 他对你说,当前你们项目的数据库中有一张用户信息表,里面存放了很用户的数据,现在需要完成一个选择性查询用户信息