欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Ruby設(shè)計(jì)模式編程中對(duì)單例模式的運(yùn)用

 更新時(shí)間:2016年03月16日 14:45:11   作者:michael_roshen  
這篇文章主要介紹了Ruby設(shè)計(jì)模式編程中對(duì)單例模式的運(yùn)用,講到了包括對(duì)Singleton模塊的使用,需要的朋友可以參考下

簡(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ì)象。

相關(guān)文章

最新評(píng)論