簡介
單例模式是設計模式中最簡單的形式之一。這一模式的目的是使得類的一個對象成為系統中的唯一執行個體。要實現這一點,可以從用戶端對其進行執行個體化開始。因此需要用一種只允許產生對象類的唯一執行個體的機制,“阻止”所有想要產生對象的訪問。使用Factory 方法來限制執行個體化過程。這個方法應該是靜態方法(類方法),因為讓類的執行個體去產生另一個唯一執行個體毫無意義。
要點
顯然單例模式的要點有三個;一是某個類只能有一個執行個體;二是它必須自行建立這個執行個體;三是它必須自行向整個系統提供這個執行個體。
從具體實現角度來說,就是以下三點:一是單例模式的類只提供私人的建構函式,二是類定義中含有一個該類的靜態私人對象,三是該類提供了
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 endendtest,test2 = AutoTest.instancetest.OpenUrl('http://www.baidu.com')test.set_textarea('aslandhu')test.click
這裡雖然建立了兩個AutoTest執行個體,但是第二個執行個體其實為nil,也就是說並沒有建立成功。
require 'rubygems'require 'watir'require 'singleton'require 'thread'class TestOneObj endclass <<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 endendtest = TestOneObj.instancetest2 = TestOneObj.instancep test.inspectp test2.inspecttest.openurl('www.baidu.com')test2.set_textarea('aslandhu')test.click
上面這段代碼試圖建立兩個Browser對象,但事實上建立的兩個對象均為同一個。雖然開啟了兩個IE視窗,但是對象還是一個,即test與test2是同一個對象。