詳解Ruby設(shè)計(jì)模式編程中對(duì)單例模式的運(yùn)用
簡(jiǎn)介
單例模式是設(shè)計(jì)模式中最簡(jiǎn)單的形式之一。這一模式的目的是使得類的一個(gè)對(duì)象成為系統(tǒng)中的唯一實(shí)例。要實(shí)現(xiàn)這一點(diǎn),可以從客戶端對(duì)其進(jìn)行實(shí)例化開始。因此需要用一種只允許生成對(duì)象類的唯一實(shí)例的機(jī)制,“阻止”所有想要生成對(duì)象的訪問(wèn)。使用工廠方法來(lái)限制實(shí)例化過(guò)程。這個(gè)方法應(yīng)該是靜態(tài)方法(類方法),因?yàn)樽岊惖膶?shí)例去生成另一個(gè)唯一實(shí)例毫無(wú)意義。
要點(diǎn)
顯然單例模式的要點(diǎn)有三個(gè);一是某個(gè)類只能有一個(gè)實(shí)例;二是它必須自行創(chuàng)建這個(gè)實(shí)例;三是它必須自行向整個(gè)系統(tǒng)提供這個(gè)實(shí)例。
從具體實(shí)現(xiàn)角度來(lái)說(shuō),就是以下三點(diǎn):一是單例模式的類只提供私有的構(gòu)造函數(shù),二是類定義中含有一個(gè)該類的靜態(tài)私有對(duì)象,三是該類提供了
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
當(dāng)創(chuàng)建了第二個(gè)對(duì)象時(shí),@@class_count 為2,二@instance_count為0,因?yàn)轭愖兞勘凰袑?shí)例所共享,黨cv1.increment調(diào)用了兩次以后@@class_count為2,創(chuàng)建第二個(gè)ClassVariableTester對(duì)象cv2的時(shí)候,共享了@@class_count,所以此時(shí)的@@class_count仍為2。
而實(shí)例變量只能為當(dāng)前對(duì)象服務(wù),所以實(shí)例對(duì)象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
結(jié)果為:true 。
采用一個(gè)類變量來(lái)保存僅有的一個(gè)類的實(shí)例,同時(shí)需要一個(gè)類方法返回這個(gè)單例實(shí)例。
但是通過(guò)SimpleLogger.new還是可以創(chuàng)建另一個(gè)實(shí)例對(duì)象,因此需要把著個(gè)new方法設(shè)為私有的。
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
結(jié)果為:true
Ruby類庫(kù)中提供了singleton,來(lái)簡(jiǎn)化單例類的創(chuàng)建。
混入Singleton,就省略了創(chuàng)建類變量,初始化單例實(shí)例,創(chuàng)建類級(jí)別的instance方法,以及將new設(shè)為私有。
通過(guò)SimpleLogger.instance來(lái)獲取日志器的單例。
但是兩種方式還是又差異的。
第一種方式稱之為“勤性單例(eager instantiation)”。
在確實(shí)需要之前就創(chuàng)建了實(shí)例對(duì)象。
第二種方式稱之為“惰性單例(lazy instantiation)”
在調(diào)用instance時(shí)才會(huì)去創(chuàng)建 。
但是這個(gè)Singleton不能真正的阻止任何事情,可以用過(guò)public_class_method改變new方法的為公用的。
打開類,設(shè)置new方法為public之后,就可以用SimpleLogger.new來(lái)創(chuàng)建對(duì)象了。
class SimpleLogger public_class_method :new end puts SimpleLogger.new
再來(lái)分兩種情況:
(一)使用全局變量,盡量不要使用全局變量,因?yàn)槿肿兞渴浅绦蚓o密的耦合在一起,
其實(shí)單例模式和全局變量的作用是一樣的,
$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")
實(shí)例
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
這里雖然創(chuàng)建了兩個(gè)AutoTest實(shí)例,但是第二個(gè)實(shí)例其實(shí)為nil,也就是說(shuō)并沒(méi)有創(chuàng)建成功。
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
上面這段代碼試圖創(chuàng)建兩個(gè)Browser對(duì)象,但事實(shí)上創(chuàng)建的兩個(gè)對(duì)象均為同一個(gè)。雖然打開了兩個(gè)IE窗口,但是對(duì)象還是一個(gè),即test與test2是同一個(gè)對(duì)象。
- 設(shè)計(jì)模式中的觀察者模式在Ruby編程中的運(yùn)用實(shí)例解析
- 實(shí)例解析Ruby設(shè)計(jì)模式開發(fā)中對(duì)觀察者模式的實(shí)現(xiàn)
- 深入剖析Ruby設(shè)計(jì)模式編程中對(duì)命令模式的相關(guān)使用
- Ruby設(shè)計(jì)模式編程中對(duì)外觀模式的應(yīng)用實(shí)例分析
- 詳解組合模式的結(jié)構(gòu)及其在Ruby設(shè)計(jì)模式編程中的運(yùn)用
- 設(shè)計(jì)模式中的模板方法模式在Ruby中的應(yīng)用實(shí)例兩則
- 實(shí)例解析Ruby設(shè)計(jì)模式編程中Strategy策略模式的使用
- 實(shí)例講解Ruby使用設(shè)計(jì)模式中的裝飾器模式的方法
- Ruby設(shè)計(jì)模式編程中使用Builder建造者模式的實(shí)例
- Ruby設(shè)計(jì)模式編程之適配器模式實(shí)戰(zhàn)攻略
- Ruby使用設(shè)計(jì)模式中的代理模式與裝飾模式的代碼實(shí)例
- Ruby中使用設(shè)計(jì)模式中的簡(jiǎn)單工廠模式和工廠方法模式
- 解析proxy代理模式在Ruby設(shè)計(jì)模式開發(fā)中的運(yùn)用
相關(guān)文章
使用Ruby re模塊創(chuàng)建復(fù)雜的正則表達(dá)式
復(fù)雜的正則表達(dá)式很難構(gòu)建,甚至很難閱讀。Ruby的Re模塊可以幫助你利用簡(jiǎn)單的表達(dá)式構(gòu)建復(fù)雜的正則表達(dá)式2014-03-03ruby實(shí)現(xiàn)github第三方認(rèn)證
GitHub在用戶認(rèn)證過(guò)程中采用了雙匙機(jī)制,在雙匙加密機(jī)制中,只有合法用戶才擁有私匙,只要GitHub在收到請(qǐng)求時(shí)可以證明提交請(qǐng)求的客戶端上擁有該私匙,即可以確認(rèn)該操作是由合法用戶發(fā)起的。我們通過(guò)ruby來(lái)簡(jiǎn)單模擬下吧。2015-06-06關(guān)于Ruby on Rails路由配置的一些建議
這篇文章主要介紹了關(guān)于Ruby on Rails路由配置的一些建議,作者提出了相關(guān)代碼編寫時(shí)一些值得注意的地方,需要的朋友可以參考下2015-08-08Ruby中的Range對(duì)象學(xué)習(xí)筆記
這篇文章主要介紹了Ruby中的Range對(duì)象學(xué)習(xí)筆記,本文講解了Range對(duì)象的定義、Range對(duì)象的一些使用技巧等內(nèi)容,需要的朋友可以參考下2014-11-11Ruby on Rails實(shí)現(xiàn)最基本的用戶注冊(cè)和登錄功能的教程
這里我們主要以has_secure_password的用戶密碼驗(yàn)證功能為中心,來(lái)講解Ruby on Rails實(shí)現(xiàn)最基本的用戶注冊(cè)和登錄功能的教程,需要的朋友可以參考下2016-06-06Ruby與Ruby on Rails框架環(huán)境搭建的簡(jiǎn)明教程
這篇文章主要介紹了Ruby與Ruby on Rails框架環(huán)境搭建的簡(jiǎn)明教程,包括RubyGems的升級(jí)與OpenSSL的支持等配置,需要的朋友可以參考下2016-05-05