iT邦幫忙

第 12 屆 iThome 鐵人賽

DAY 29
0
自我挑戰組

Codewars Ruby Challenge in 30 Days系列 第 29

Codewars Ruby Challenge - Day 29/30

  • 分享至 

  • xImage
  •  

學習

  1. ruby 迴圈也可以使用 break :過去寫過 C 語言,在迴圈希望符合條件時提早結束會使用 break,今天發現原來在 ruby 也有這樣的用法。好處是可以提高效能
  2. 學習非線性思考:題目在 Codewars 的 Best Practice 會發現寫法非常短,看懂後發現如果想讓程式從能動到簡潔甚至效能可能更好,需要花時間從非線性著手。舉個例子,以線性思考,第二個參數有值時,我會想用第一個參數做 map 巢狀搭配 第二個參數做 each,拿第一個參數的值跟所有在第二個參數的每個值比較;但 Best Practice 的做法則是反過來,去看第二個參數是否有包含第一個參數內的每個值,這時就可以用 include? 不用寫落落長的 if 判斷式

題目:

A string is considered to be in title case if each word in the string is either (a) capitalised (that is, only the first letter of the word is in upper case) or (b) considered to be an exception and put entirely into lower case unless it is the first word, which is always capitalised.
Write a function that will convert a string into title case, given an optional list of exceptions (minor words). The list of minor words will be given as a string with each word separated by a space. Your function should ignore the case of the minor words string -- it should behave in the same way even if the case of the minor word string is changed.

翻譯:直接看下面需要過的測試,會給你兩個參數,第二個不一定會給。
1.第一個參數內的所有字元都要轉換(開頭大寫,其他字小寫)
2.如果第二個參數有給的話,用來跟第一個參數比對,字元有一樣的話(不分大小寫)整個字元都變成小寫。
3.最後,前面做完任何轉換,第一個參數的第一個字元的開頭字一定要是大寫

def title_case(title, minor_words = '')

end

答案需要過以下測試:

RSpec.describe "Title Case" do
  it "Example cases" do
    expect(title_case('')).to eq('')
    expect(title_case('a clash of KINGS', 'a an the of')).to eq('A Clash of Kings')
    expect(title_case('THE WIND IN THE WILLOWS', 'The In')).to eq('The Wind in the Willows')
    expect(title_case('the quick brown fox')).to eq('The Quick Brown Fox')
  end
end

我的答案

def title_case(title, minor_words = '')
  return '' if title == ''
  return title.split(" ").map(&:capitalize).join(" ") if title.length != 0 && minor_words == ''

  if minor_words.length != 0
    minor = minor_words.split(" ").map(&:capitalize)
  end

  result = title.split(" ").map(&:capitalize).map do |t|
             minor.each do |m|
               if t == m
                t = t.downcase
                break
               end
             end
             t
           end
  result[0] = result[0].capitalize
  result.join(" ")
end

思路:

  1. 上面邊在把題目翻譯成中文,也同時把題目拆成三個子項目
    (1) 把兩個參數每個字元先 capitalize(第二個參數如果是有,才會做)
    (2) 拿第一個參數每個字元去跟第二個參數的所有字元比,有一樣的話轉換成小寫
    (3) 最後把轉換後的第一個字元再次 capitalize (因為可能會是小寫)
  2. 針對上面規劃實作,發現透過 break 可以提升效能
  3. 發現最後一條測試不會過,發現有考慮兩個情況「第一個參數=空字串|第二個參數=空字串」、「第一個參數=有|第二個參數=有」,但少考慮「第一個參數=有|第二個參數=空字串」這個情況,又多新增一個 guard clause 在第二行

Best practice in Codewars

def title_case(title, minor_words = '')
  title.capitalize.split().map{|a| minor_words.downcase.split().include?(a) ? a : a.capitalize}.join(' ')
end

上一篇
Codewars Ruby Challenge - Day 28/30
下一篇
Codewars Ruby Challenge - Day 30/30
系列文
Codewars Ruby Challenge in 30 Days30
圖片
  直播研討會
圖片
{{ item.channelVendor }} {{ item.webinarstarted }} |
{{ formatDate(item.duration) }}
直播中

尚未有邦友留言

立即登入留言