校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > 實例解析Ruby設(shè)計模式編程中Strategy策略模式的使用

實例解析Ruby設(shè)計模式編程中Strategy策略模式的使用

熱門標(biāo)簽:滴滴外呼系統(tǒng) 電銷機(jī)器人廠商代理 高德地圖標(biāo)注客服 湖州u友防封電銷卡 地圖標(biāo)注賺錢項目注冊 百度地圖標(biāo)注自定義圖片 白銀外呼paas系統(tǒng) 常德電銷平臺外呼系統(tǒng)軟件價格 徐州網(wǎng)絡(luò)外呼系統(tǒng)哪個好

今天你的leader興致沖沖地找到你,希望你可以幫他一個小忙,他現(xiàn)在急著要去開會。要幫什么忙呢?你很好奇。
他對你說,當(dāng)前你們項目的數(shù)據(jù)庫中有一張用戶信息表,里面存放了很用戶的數(shù)據(jù),現(xiàn)在需要完成一個選擇性查詢用戶信息的功能。他說會傳遞給你一個包含許多用戶名的數(shù)組,你需要根據(jù)這些用戶名把他們相應(yīng)的數(shù)據(jù)都給查出來。
這個功能很簡單的嘛,你爽快地答應(yīng)了。由于你們項目使用的是MySQL數(shù)據(jù)庫,你很快地寫出了如下代碼:

require 'mysql' 
 
class QueryUtil 
  def find_user_info usernames 
    @db = Mysql.real_connect("localhost","root","123456","test",3306); 
    sql = "select * from user_info where " 
    usernames.each do |user| 
      sql  "username = '" 
      sql  user 
      sql  "' or " 
    end 
    puts sql 
    result = @db.query(sql); 
    result.each_hash do |row| 
      #處理從數(shù)據(jù)庫讀出來的數(shù)據(jù) 
    end 
    #后面應(yīng)將讀到的數(shù)據(jù)組裝成對象返回,這里略去 
  ensure 
    @db.close 
  end 
end 

這里根據(jù)傳入的用戶名數(shù)組拼裝了SQL語句,然后去數(shù)據(jù)庫中查找相應(yīng)的行。為了方面調(diào)試,你還將拼裝好的SQL語句打印了出來。
然后,你寫了如下代碼來測試這個方法:

qUtil = QueryUtil.new 
qUtil.find_user_info ["Tom", "Jim", "Anna"] 

現(xiàn)在運(yùn)行一下測試代碼,你發(fā)現(xiàn)程序出錯了。于是你立刻去檢查了一下打印的SQL語句,果然發(fā)現(xiàn)了問題。

select * from user_info where username = 'Tom' or username = 'Jim' or username = 'Anna' or  
拼裝出來的SQL語句在最后多加了一個 or 關(guān)鍵字!因為for循環(huán)執(zhí)行到最后一條數(shù)據(jù)時不應(yīng)該再加上or,可是代碼很笨地給最后一條數(shù)據(jù)也加了or關(guān)鍵字,導(dǎo)致SQL語句語法出錯了。
這可怎么辦呢?
有了!你靈光一閃,想出了一個解決辦法。等SQL語句拼裝完成后,截取到最后一個or之前的位置不就好了么。于是你將代碼改成如下所示:

require 'mysql' 
 
class QueryUtil 
  def find_user_info usernames 
    @db = Mysql.real_connect("localhost","root","123456","test",3306); 
    sql = "select * from user_info where " 
    usernames.each do |user| 
      sql  "username = '" 
      sql  user 
      sql  "' or " 
    end 
    sql = sql[0 .. -" or ".length] 
    puts sql 
    result = @db.query(sql); 
    result.each_hash do |row| 
      #處理從數(shù)據(jù)庫讀出來的數(shù)據(jù) 
    end 
    #后面應(yīng)將讀到的數(shù)據(jù)組裝成對象返回,這里略去 
  ensure 
    @db.close 
  end 
end 

使用String的截取子字符串方法,只取到最后一個or之前的部分,這樣再運(yùn)行測試代碼,一切就正常了,打印的SQL語句如下所示:

select * from user_info where username = 'Tom' or username = 'Jim' or username = 'Anna' 

好了,完工!你自信滿滿。
你的leader開完會后,過來看了下你的成果??傮w來說,他還挺滿意,但對于你使用的SQL語句拼裝算法,他總是感覺有些不對勁,可是又說不上哪里不好。于是他告訴了你另一種拼裝SQL語句的算法,讓你加入到代碼中,但是之前的那種算法也不要刪除,先保留著再說,然后他又很忙似的跑開了。于是,你把他剛剛教你的算法加了進(jìn)去,代碼如下所示:

require 'mysql' 
 
class QueryUtil 
  def find_user_info(usernames, strategy) 
    @db = Mysql.real_connect("localhost","root","123456","test",3306); 
    sql = "select * from user_info where " 
    if strategy == 1 
      usernames.each do |user| 
        sql  "username = '" 
        sql  user 
        sql  "' or " 
      end 
      sql = sql[0 .. -" or ".length] 
    elsif strategy == 2 
      need_or = false 
      usernames.each do |user| 
        sql  " or " if need_or 
        sql  "username = '" 
        sql  user 
        sql  "'" 
        need_or = true 
      end 
    end 
    puts sql 
    result = @db.query(sql); 
    result.each_hash do |row| 
      #處理從數(shù)據(jù)庫讀出來的數(shù)據(jù) 
    end 
    #后面應(yīng)將讀到的數(shù)據(jù)組裝成對象返回,這里略去 
  ensure 
    @db.close 
  end 
end 

可以看到,你leader教你的拼裝算法,使用了一個布爾變量來控制是否需要加個or這個關(guān)鍵字,第一次執(zhí)行for循環(huán)的時候因為該布爾值為false,所以不會加上or,在循環(huán)的最后將布爾值賦值為true,這樣以后循環(huán)每次都會在頭部加上一個or關(guān)鍵字,由于使用了頭部添加or的方法,所以不用再擔(dān)心SQL語句的尾部會多出一個or來。然后你為了將兩個算法都保留,在find_user_info方法上加了一個參數(shù),strategy值為1表示使用第一種算法,strategy值為2表示使用第二種算法。
這樣測試代碼也需要改成如下方式:

qUtil = QueryUtil.new 
qUtil.find_user_info(["Tom", "Jim", "Anna"], 2) 

這里你通過參數(shù)指明了使用第二種算法來拼裝SQL語句,打印的結(jié)果和使用第一種算法是完全相同的。
你立刻把你的leader從百忙之中拖了過來,讓他檢驗一下你當(dāng)前的成果,可是他還是一如既往的挑剔。
“你這樣寫的話,find_user_info這個方法的邏輯就太復(fù)雜了,非常不利于閱讀,也不利于將來的擴(kuò)展,如果我還有第三第四種算法想加進(jìn)去,這個方法還能看嗎?”  你的leader指點你,遇到這種情況,就要使用策略模式來解決,策略模式的核心思想就是把算法提取出來放到一個獨立的對象中。
為了指點你,他不顧自己的百忙,開始教你如何使用策略模式進(jìn)行優(yōu)化。
首先定義一個父類,父類中包含了一個get_sql方法,這個方法就是簡單的拋出了一個異常:

class Strategy 
  def get_sql usernames 
    raise "You should override this method in subclass." 
  end 
end 

然后定義兩個子類都繼承上述父類,并將兩種拼裝SQL語句的算法分別加入兩個子類中:

class Strategy1 
  def get_sql usernames 
    sql = "select * from user_info where " 
    usernames.each do |user| 
      sql  "username = '" 
      sql  user 
      sql  "' or " 
    end 
    sql = sql[0 .. -" or ".length] 
  end 
end 

class Strategy2 
  def get_sql usernames 
    sql = "select * from user_info where " 
    need_or = false 
    usernames.each do |user| 
      sql  " or " if need_or 
      sql  "username = '" 
      sql  user 
      sql  "'" 
      need_or = true 
    end 
  end 
end 

然后在QueryUtil的find_user_info方法中調(diào)用Strategy的get_sql方法就可以獲得拼裝好的SQL語句,代碼如下所示:

require 'mysql' 
 
class QueryUtil 
  def find_user_info(usernames, strategy) 
    @db = Mysql.real_connect("localhost","root","123456","test",3306); 
    sql = strategy.get_sql(usernames) 
    puts sql 
    result = @db.query(sql); 
    result.each_hash do |row| 
      #處理從數(shù)據(jù)庫讀出來的數(shù)據(jù) 
    end 
    #后面應(yīng)將讀到的數(shù)據(jù)組裝成對象返回,這里略去 
  ensure 
    @db.close 
  end 
end 

最后,測試代碼在調(diào)用find_user_info方法時,只需要顯示地指明需要使用哪一個策略對象就可以了:

qUtil = QueryUtil.new 
qUtil.find_user_info(["Tom", "Jim", "Anna"], Strategy1.new) 
qUtil.find_user_info(["Jac", "Joe", "Rose"], Strategy2.new) 

打印出的SQL語句絲毫不出預(yù)料,如下所示:

select * from user_info where username = 'Tom' or username = 'Jim' or username = 'Anna' 
select * from user_info where username = 'Jac' or username = 'Joe' or username = 'Rose' 

使用策略模式修改之后,代碼的可讀性和擴(kuò)展性都有了很大的提高,即使以后還需要添加新的算法,你也是手到擒來了!

策略模式和簡單工廠模式結(jié)合的實例

需求:

商場收銀軟件,根據(jù)客戶購買物品的單價和數(shù)量,計算費用,會有促銷活動,打八折,滿三百減一百之類的。

1.使用工廠模式

# -*- encoding: utf-8 -*-

#現(xiàn)金收費抽象類
class CashSuper
  def accept_cash(money)
  end
end

#正常收費子類
class CashNormal  CashSuper
  def accept_cash(money)
    money
  end
end

#打折收費子類
class CashRebate  CashSuper
  attr_accessor :mony_rebate
  
  def initialize(mony_rebate)
    @mony_rebate = mony_rebate
  end

  def accept_cash(money)
    money * mony_rebate
  end
end

#返利收費子類
class CashReturn  CashSuper
  attr_accessor :mony_condition, :mony_return
  
  def initialize(mony_condition, mony_return)
    @mony_condition = mony_condition
    @mony_return = mony_return
  end

  def accept_cash(money)
    if money > mony_condition
      money - (money/mony_condition) * mony_return
    end
  end
end

#現(xiàn)金收費工廠類
class CashFactory
  def self.create_cash_accept(type)
    case type
    when '正常收費'
      CashNormal.new()
    when '打8折'
      CashRebate.new(0.8)
    when '滿三百減100'
      CashReturn.new(300,100)
    end
  end
end

cash0 = CashFactory.create_cash_accept('正常收費')
p cash0.accept_cash(700)

cash1 = CashFactory.create_cash_accept('打8折')
p cash1.accept_cash(700)

cash2 = CashFactory.create_cash_accept('滿三百減100')
p cash2.accept_cash(700)

做到了自定義折扣比例和滿減的數(shù)量。

存在的問題:

增加活動的種類時,打五折,滿五百減二百,需要在工廠類中添加分支結(jié)構(gòu)。

活動是多種多樣的,也有可能增加積分活動,滿100加10積分,積分一定可以領(lǐng)取活動獎品,這時就要增加一個子類。

但是每次增加活動的時候,都要去修改工廠類,是很糟糕的處理方式,面對算法有改動時,應(yīng)該有更好的辦法。

2.策略模式

CashSuper和子類都是不變的,增加以下內(nèi)容:

class CashContext
  
  attr_accessor :cs
  
  def initialize(c_super)
    @cs = c_super
  end
  
  def result(money)
    cs.accept_cash(money)
  end

end

type = '打8折'
cs=case type
  when '正常收費'
    CashContext.new(CashNormal.new())
  when '打8折'
    CashContext.new(CashRebate.new(0.8))
  when '滿三百減100'
    CashContext.new(CashReturn.new(300,100))
  end
p cs.result(700)

CashContext類對不同的CashSuper子類進(jìn)行了封裝,會返回對應(yīng)的result。也就是對不同的算法進(jìn)行了封裝,無論算法如何變化。都可以使用result得到結(jié)果。
不過,目前有一個問題,使用者需要去做判斷,來選擇使用哪個算法。可以和簡單工場類結(jié)合。

3.策略和簡單工場結(jié)合

class CashContext
  
  attr_accessor :cs
  
  def initialize(type)
    case type
    when '正常收費'
      @cs = CashNormal.new()
    when '打8折'
      @cs = CashRebate.new(0.8)
    when '滿三百減100'
      @cs = CashReturn.new(300,100)
    end
  end
  
  def result(money)
    cs.accept_cash(money)
  end

end

cs=CashContext.new('打8折')

p cs.result(700)

CashContext中實例化了不同的子類。(簡單工廠)
將子類選擇的過程轉(zhuǎn)移到了內(nèi)部,封裝了算法(策略模式)。

調(diào)用者使用更簡單,傳入?yún)?shù)(活動類型,原價),即可得到最終的結(jié)果。
這里使用者只需要知道一個類(CashContext)就可以了,而簡單工場需要知道兩個類(CashFactory的accept_cash方法和CashFactory),也就是說封裝的更徹底。

您可能感興趣的文章:
  • 設(shè)計模式中的觀察者模式在Ruby編程中的運(yùn)用實例解析
  • 實例解析Ruby設(shè)計模式開發(fā)中對觀察者模式的實現(xiàn)
  • 深入剖析Ruby設(shè)計模式編程中對命令模式的相關(guān)使用
  • Ruby設(shè)計模式編程中對外觀模式的應(yīng)用實例分析
  • 詳解組合模式的結(jié)構(gòu)及其在Ruby設(shè)計模式編程中的運(yùn)用
  • 設(shè)計模式中的模板方法模式在Ruby中的應(yīng)用實例兩則
  • 實例講解Ruby使用設(shè)計模式中的裝飾器模式的方法
  • Ruby設(shè)計模式編程中使用Builder建造者模式的實例
  • 詳解Ruby設(shè)計模式編程中對單例模式的運(yùn)用
  • Ruby設(shè)計模式編程之適配器模式實戰(zhàn)攻略
  • Ruby使用設(shè)計模式中的代理模式與裝飾模式的代碼實例
  • Ruby中使用設(shè)計模式中的簡單工廠模式和工廠方法模式
  • 解析proxy代理模式在Ruby設(shè)計模式開發(fā)中的運(yùn)用

標(biāo)簽:張家界 永州 三沙 普洱 荊門 梧州 遼寧 公主嶺

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《實例解析Ruby設(shè)計模式編程中Strategy策略模式的使用》,本文關(guān)鍵詞  實例,解析,Ruby,設(shè)計模式,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《實例解析Ruby設(shè)計模式編程中Strategy策略模式的使用》相關(guān)的同類信息!
  • 本頁收集關(guān)于實例解析Ruby設(shè)計模式編程中Strategy策略模式的使用的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    亚洲视频中文字幕| 精品国产免费一区二区三区香蕉| 欧美日韩亚洲综合在线| 亚洲欧美另类久久久精品2019| 成人av在线影院| 亚洲精品乱码久久久久久久久| 91福利区一区二区三区| 日韩一区欧美二区| 精品国产电影一区二区| 国产成人av电影在线观看| 国产精品护士白丝一区av| 在线观看视频欧美| 久久av中文字幕片| 亚洲精品成人精品456| 日韩精品一区二区在线观看| 成人激情小说乱人伦| 亚洲成人综合在线| 国产欧美一区二区精品秋霞影院| 日本道色综合久久| 国产精品白丝jk白祙喷水网站 | 久久午夜国产精品| 一本一道综合狠狠老| 极品瑜伽女神91| 亚洲大片精品永久免费| 国产女同性恋一区二区| 欧美一区二区大片| 色综合天天综合给合国产| 久久99久久99精品免视看婷婷| 一区二区三区丝袜| 国产精品少妇自拍| 精品国产亚洲一区二区三区在线观看| 色综合欧美在线视频区| 粉嫩aⅴ一区二区三区四区| 麻豆精品一区二区| 亚洲五码中文字幕| 亚洲男人的天堂在线aⅴ视频| 久久久五月婷婷| 日韩欧美二区三区| 欧美一区二区三区免费在线看| 日本韩国欧美国产| 91小视频免费观看| 成人的网站免费观看| 国产精品一区二区三区网站| 日韩精品欧美成人高清一区二区| 一区二区三区四区高清精品免费观看 | 91在线观看美女| 丰满少妇久久久久久久| 激情综合网最新| 美女爽到高潮91| 日本麻豆一区二区三区视频| 夜夜精品视频一区二区| 亚洲免费电影在线| 亚洲美腿欧美偷拍| 亚洲精品视频自拍| 洋洋av久久久久久久一区| 亚洲人成伊人成综合网小说| 亚洲视频一二区| 亚洲欧美在线另类| 一区2区3区在线看| 天堂午夜影视日韩欧美一区二区| 五月天一区二区| 免费美女久久99| 国产一区二区看久久| 国产一区91精品张津瑜| 国产成人精品三级麻豆| 99久久久国产精品免费蜜臀| 91色.com| 日韩一区二区在线看片| 久久久777精品电影网影网| 中文字幕第一区综合| 亚洲视频一二区| 视频一区在线视频| 国产一区二区三区精品欧美日韩一区二区三区| 免费成人结看片| 国产成人在线视频免费播放| 成人午夜电影网站| 欧美色精品在线视频| 日韩一区二区三区在线视频| 2017欧美狠狠色| 一区二区三区在线视频免费| 麻豆极品一区二区三区| 成人国产免费视频| 欧美日韩高清一区二区不卡| 欧美不卡一区二区三区四区| 中文文精品字幕一区二区| 自拍偷拍国产亚洲| 日韩国产欧美在线播放| 粉嫩久久99精品久久久久久夜| 在线观看日韩毛片| 国产亚洲视频系列| 日韩经典中文字幕一区| 国产成人自拍网| 欧美一区二区三区系列电影| 亚洲国产精品黑人久久久| 午夜av一区二区| 97久久精品人人做人人爽| 日韩精品一区二区在线观看| 亚洲小说春色综合另类电影| 国产福利一区二区三区| 欧美一级高清大全免费观看| 一区二区欧美视频| 国产成人av一区二区三区在线| 91精品视频网| 亚洲一区二区三区四区不卡| 国产白丝精品91爽爽久久| 日韩一级视频免费观看在线| 一区二区三区四区在线| av成人动漫在线观看| 久久影院午夜片一区| 日本aⅴ精品一区二区三区| 日本乱人伦aⅴ精品| 国产精品每日更新在线播放网址| 久久99精品网久久| 欧美va亚洲va国产综合| 人人爽香蕉精品| 777久久久精品| 日韩高清电影一区| 91精品国产高清一区二区三区蜜臀| 亚洲欧美国产毛片在线| 97精品电影院| 最新热久久免费视频| caoporm超碰国产精品| 国产精品欧美一级免费| gogogo免费视频观看亚洲一| 国产女主播一区| 不卡电影一区二区三区| 亚洲丝袜制服诱惑| 欧美亚洲一区三区| 香蕉影视欧美成人| 欧美精品色综合| 毛片av一区二区| 精品国产一区二区三区不卡| 久久电影网电视剧免费观看| 日韩一级高清毛片| 国产在线视视频有精品| 国产视频视频一区| 成人免费视频网站在线观看| 国产精品女主播在线观看| 91麻豆免费在线观看| 香蕉乱码成人久久天堂爱免费| 欧美调教femdomvk| 秋霞成人午夜伦在线观看| 精品女同一区二区| 国产91富婆露脸刺激对白| 中文字幕欧美一区| 欧美三级午夜理伦三级中视频| 午夜国产精品影院在线观看| 欧美草草影院在线视频| 成人激情小说乱人伦| 亚洲丰满少妇videoshd| 欧美v亚洲v综合ⅴ国产v| 成人91在线观看| 亚洲国产日韩精品| 日韩精品在线一区| 91色在线porny| 裸体歌舞表演一区二区| 国产精品久久久久影院亚瑟| 欧美狂野另类xxxxoooo| 国产精品1区二区.| 亚洲图片自拍偷拍| 国产欧美日韩另类视频免费观看| 不卡一区二区在线| 首页亚洲欧美制服丝腿| 欧美激情一区二区三区在线| 欧美老肥妇做.爰bbww| 成人av资源站| 久久99久国产精品黄毛片色诱| 亚洲三级在线看| 欧美mv日韩mv亚洲| 91行情网站电视在线观看高清版| 精品一区二区免费| 亚洲小说欧美激情另类| 国产精品国产三级国产a| 日韩一区二区免费电影| 欧美在线999| 成人av网址在线| 精品一区二区在线观看| 丝袜亚洲精品中文字幕一区| 国产精品麻豆久久久| 日韩视频一区二区| 色偷偷88欧美精品久久久| 国产精品一级黄| 蜜桃视频在线观看一区| 一区二区三国产精华液| 国产精品久久久久久久第一福利| 欧美夫妻性生活| 欧洲视频一区二区| 99麻豆久久久国产精品免费优播| 久久99精品久久久久久国产越南| 亚洲国产欧美在线| 亚洲黄色片在线观看| 综合久久久久久久| 中文字幕不卡在线观看| 久久久.com| 久久亚洲春色中文字幕久久久| 日韩一卡二卡三卡国产欧美| 91精品国产综合久久福利软件| 91久久精品日日躁夜夜躁欧美| 91在线porny国产在线看| av电影在线观看一区|