v-crn Code Log

主に備忘録

Rails Tutorial 第4版 第4章 演習 解答例

f:id:v-crn:20190711155257p:plain

この文書はRails Tutorial 第4版 第4章 Rails風味のRubyの演習に対する個人の解答例です。解答には誤りや不適切な表現が含まれていることがありますが、もし誤謬を見つけたらコメント頂けると嬉しいです。
それでは、やっていきましょう!

4.1 動機

4.1.1 組み込みヘルパー

演習なし。

4.1.2 カスタムヘルパー

演習なし。

4.2 文字列とメソッド

4.2.1 コメント

演習なし。

4.2.2 文字列

1. city変数に適当な市区町村を、prefecture変数に適当な都道府県を代入してください。

>> city = "新宿区"
=> "新宿区"
>> prefecture = "東京都"
=> "東京都"

2. 先ほど作った変数と式展開を使って、「東京都 新宿区」のような住所の文字列を作ってみましょう。出力にはputsを使ってください。

>> puts prefecture + " " + city
東京都 新宿区
=> nil

3. 上記の文字列の間にある半角スペースをタブに置き換えてみてください。(ヒント: 改行文字と同じで、タブも特殊文字です)

タブの特殊文字\tを使って置き換えます。

>> puts prefecture + "\t" + city
東京都  新宿区
=> nil

4. タブに置き換えた文字列を、ダブルクォートからシングルクォートに置き換えてみるとどうなるでしょうか?

タブに置き換えた文字列がそのまま表示されます。

>> puts prefecture + '\t' + city
東京都\t新宿区
=> nil

4.2.3 オブジェクトとメッセージ受け渡し

1. "racecar" の文字列の長さはいくつですか? lengthメソッドを使って調べてみてください。

>> "racecar".length
=> 7

2. reverseメソッドを使って、"racecar"の文字列を逆から読むとどうなるか調べてみてください。

>> "racecar".reverse
=> "racecar"

3. 変数sに "racecar" を代入してください。その後、比較演算子 (==) を使って変数sとs.reverseの値が同じであるかどうか、調べてみてください。

>> s == s.reverse
=> true

4. リスト 4.9を実行すると、どんな結果になるでしょうか? 変数sに "onomatopoeia" という文字列を代入するとどうなるでしょうか? ヒント: 上矢印 (またはCtrl-Pコマンド) を使って以前に使ったコマンドを再利用すると一からコマンドを全部打ち込む必要がなくて便利ですよ。

>> puts "It's a palindrome!" if s == s.reverse
It's a palindrome!
=> nil

4.2.4 メソッドの定義

1. リスト 4.10のFILL_INの部分を適切なコードに置き換え、回文かどうかをチェックするメソッドを定義してみてください。ヒント: リスト 4.9の比較方法を参考にしてください。

>> def palindrome_tester(s)
>>   if s == s.reverse
>>     puts "It's a palindrome!"
>>   else
>>     puts "It's not a palindrome."
>>   end
>> end

2. 上で定義したメソッドを使って “racecar” と “onomatopoeia” が回文かどうかを確かめてみてください。1つ目は回文である、2つ目は回文でない、という結果になれば成功です。

>> palindrome_tester("racecar")
It's a palindrome!
=> nil

3. palindrome_tester("racecar")に対してnil?メソッドを呼び出し、戻り値がnilであるかどうかを確認してみてください (つまりnil?を呼び出した結果がtrueであることを確認してください)。このメソッドチェーンは、nil?メソッドがリスト 4.10の戻り値を受け取り、その結果を返しているという意味になります。

>> palindrome_tester("racecar").nil?
It's a palindrome!
=> true

4.2.5 titleヘルパー、再び

演習なし。

4.3 他のデータ構造

4.3.1 配列と範囲演算子

1. 文字列「A man, a plan, a canal, Panama」を ", " で分割して配列にし、変数aに代入してみてください。

>> a = "A man, a plan, a canal, Panama".split(", ")
=> ["A man", "a plan", "a canal", "Panama"]

2. 今度は、変数aの要素を連結した結果 (文字列) を、変数sに代入してみてください。

>> s = a.join
=> "A mana plana canalPanama"

3. 変数sを半角スペースで分割した後、もう一度連結して文字列にしてください (ヒント: メソッドチェーンを使うと1行でもできます)。リスト 4.10で使った回文をチェックするメソッドを使って、(現状ではまだ) 変数sが回文ではないことを確認してください。downcaseメソッドを使って、s.downcaseは回文であることを確認してください。

>> s = s.split(" ").join
=> "AmanaplanacanalPanama"
>> palindrome_tester(s)
It's not a palindrome.
=> nil
>> palindrome_tester(s.downcase)
It's a palindrome!
=> nil

4. aからzまでの範囲オブジェクトを作成し、7番目の要素を取り出してみてください。同様にして、後ろから7番目の要素を取り出してみてください。(ヒント: 範囲オブジェクトを配列に変換するのを忘れないでください)

alp = ('a'..'z').to_a
=> ["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z"]
>> alp[7-1]
=> "g"
>> alp[alp.length-7]
=> "t"

4.3.2 ブロック

1. 範囲オブジェクト0..16を使って、各要素の2乗を出力してください。

>> (0..16).each do |n|
?>   puts n * n
>> end
0
1
4
9
16
25
36
49
64
81
100
121
144
169
196
225
256
=> 0..16

2. yeller (大声で叫ぶ) というメソッドを定義してください。このメソッドは、文字列の要素で構成された配列を受け取り、各要素を連結した後、大文字にして結果を返します。例えばyeller([’o’, ’l’, ’d’])と実行したとき、"OLD"という結果が返ってくれば成功です。ヒント: mapとupcaseとjoinメソッドを使ってみましょう。

>> def yeller(s)
>>   s.map(&:upcase).join
>> end
=> :yeller
>> yeller(['o', 'l', 'd'])
=> "OLD"

※以下のようにmapを使わずに定義することもできます。

>> def yeller(s)
>>   s.join.upcase
>> end
=> :yeller
>> yeller(['o', 'l', 'd'])
=> "OLD"

※問題文の例yeller([’o’, ’l’, ’d’])の通り文字列の囲いとしてを使うと配列オブジェクトを定義できずエラーになります。文字列は'で囲いましょう。

>> yeller([’o’, ’l’, ’d’])
Traceback (most recent call last):
        1: from (irb):9

3. random_subdomainというメソッドを定義してください。このメソッドはランダムな8文字を生成し、文字列として返します。ヒント: サブドメインを作るときに使ったRubyコードをメソッド化したものです。

>> def random_subdomain
>>   rndm = ('a'..'z').to_a + ('A'..'Z').to_a + ('0'..'9').to_a
>>   rndm.shuffle[0..7].join
>> end
=> :random_subdomain

4. リスト 4.12の「?」の部分を、それぞれ適切なメソッドに置き換えてみてください。ヒント:split、shuffle、joinメソッドを組み合わせると、メソッドに渡された文字列 (引数) をシャッフルさせることができます。

>> def string_shuffle(s)
>>   s.split('').shuffle.join
>> end
>> string_shuffle("foobar")
=> "ofbroa"

4.3.3 ハッシュとシンボル

1. キーが’one’、’two’、’three’となっていて、それぞれの値が’uno’、’dos’、’tres’となっているハッシュを作ってみてください。その後、ハッシュの各要素をみて、それぞれのキーと値を"’#{key}’のスペイン語は’#{value}’"といった形で出力してみてください。

>> nums.each do |key, value|
?>   puts "'#{key}'のスペイン語は'#{value}'"
>> end
'one'のスペイン語は'uno'
'two'のスペイン語は'dos'
'three'のスペイン語は'tres'
=> {:one=>"uno", :two=>"dos", :three=>"tres"}

2. person1、person2、person3という3つのハッシュを作成し、それぞれのハッシュに:firstと:lastキーを追加し、適当な値 (名前など) を入力してください。その後、次のようなparamsというハッシュのハッシュを作ってみてください。1.) キーparams[:father]の値にperson1を代入、2). キーparams[:mother]の値にperson2を代入、3). キーparams[:child]の値にperson3を代入。最後に、ハッシュのハッシュを調べていき、正しい値になっているか確かめてみてください。(例えばparams[:father][:first]がperson1[:first]と一致しているか確かめてみてください)

>> person1 = {first: "tanaka", last: "taro"}
=> {:first=>"tanaka", :last=>"taro"}
>> person2 = {first: "sato", last: "hanako"}
=> {:first=>"sato", :last=>"hanako"}
>> person3 = {first: "suzuki", last: "saburo"}
=> {:first=>"suzuki", :last=>"saburo"}

>> params = {}
=> {}
>> params[:father] = person1
=> {:first=>"tanaka", :last=>"taro"}
>> params[:mother] = person2
=> {:first=>"sato", :last=>"hanako"}
>> params[:child] = person3
=> {:first=>"suzuki", :last=>"saburo"}

>> params[:father][:first] == person1[:first]
=> true
>> params[:mother][:last] == person2[:last]
=> true
>> params[:child] == person3
=> true

3. userというハッシュを定義してみてください。このハッシュは3つのキー:name、:email、:password_digestを持っていて、それぞれの値にあなたの名前、あなたのメールアドレス、そして16文字からなるランダムな文字列が代入されています。

>> user = {name: "v-crn", email: "vcrn@example.com", password_digest: (('a'..'z').to_a + ('A'..'Z').to_a + ('0'..'9').to_a).shuffle[0..15].join}
=> {:name=>"v-crn", :email=>"vcrn@example.com", :password_digest=>"bQmZrOwDWhTn3i96"}

4. Ruby API (訳注: もしくはるりまサーチ) を使って、Hashクラスのmergeメソッドについて調べてみてください。次のコードを実行せずに、どのような結果が返ってくるか推測できますか? 推測できたら、実際にコードを実行して推測があっていたか確認してみましょう。

  { "a" => 100, "b" => 200 }.merge({ "b" => 300 })

Ruby API
https://api.rubyonrails.org/classes/ActionController/Parameters.html#method-i-merge

るりまサーチ
https://docs.ruby-lang.org/ja/search/query:merge/#entry-0

推測
mergeの際に両者共通のキーがある場合、mergeの引数のハッシュ値で上書きされるため、次の結果が返ってくると予想します。

{"a"=>100, "b"=>300}

結果

=> {"a"=>100, "b"=>300}

推測と合致しました。

4.3.4 CSS、再び

演習なし。

4.4 Rubyにおけるクラス

4.4.1 コンストラク

1. 1から10の範囲オブジェクトを生成するリテラルコンストラクタは何でしたか? (復習です)

1..10

2. 今度はRangeクラスとnewメソッドを使って、1から10の範囲オブジェクトを作ってみてください。ヒント: newメソッドに2つの引数を渡す必要があります

>> Range.new(1,10)
=> 1..10

3. 比較演算子==を使って、上記2つの課題で作ったそれぞれのオブジェクトが同じであることを確認してみてください。

>> (1..10) == Range.new(1,10)
=> true

4.4.2 クラス継承

Rangeクラスの継承階層を調べてみてください。同様にして、HashとSymbolクラスの継承階層も調べてみてください。

>> Range.superclass
=> Object
>> Hash.superclass
=> Object
>> Symbol.superclass
=> Object

いずれの継承クラスもObjectクラスです。

リスト 4.15にあるself.reverseのselfを省略し、reverseと書いてもうまく動くことを確認してみてください。

>> class Word < String
>>   def palindrome?
>>     self == reverse
>>   end
>> end
=> :palindrome?
>> s = Word.new("level") 
=> "level"
>> s.palindrome?
=> true

4.4.3 組み込みクラスの変更

1. palindrome?メソッドを使って、“racecar”が回文であり、“onomatopoeia”が回文でないことを確認してみてください。南インドの言葉「Malayalam」は回文でしょうか? ヒント: downcaseメソッドで小文字にすることを忘れないで。

>> s = Word.new("racecar")
=> "racecar"
>> s.palindrome?
=> true
>> s = Word.new("onomatopoeia")
=> "onomatopoeia"
>> s.palindrome?
=> false
>> s = Word.new("Malayalam").downcase
=> "malayalam"
>> s.palindrome?
=> true

2. リスト 4.16を参考に、Stringクラスにshuffleメソッドを追加してみてください。ヒント: リスト 4.12も参考になります。

>> class String
>>   def shuffle
>>     self.split('').shuffle.join
>>   end
>> end
=> :shuffle
>> "foobar".shuffle
=> "rbofao"

3. リスト 4.16のコードにおいて、self.を削除してもうまく動くことを確認してください。

>> class String
>>   def shuffle
>>     split('').shuffle.join
>>   end
>> end
=> :shuffle
>> "foobar".shuffle
=> "oborfa"

4.4.4 コントローラクラス

1. 第2章で作ったToyアプリケーションのディレクトリでRailsコンソールを開き、User.newと実行することでuserオブジェクトが生成できることを確認してみましょう。

>> user = User.new
=> #<User id: nil, name: nil, email: nil, created_at: nil, updated_at: nil>

2. 生成したuserオブジェクトのクラスの継承階層を調べてみてください。

>> user.class
=> User(id: integer, name: string, email: string, created_at: datetime, updated_at: datetime)
>> user.class.superclass
=> ApplicationRecord(abstract)
>> user.class.superclass.superclass
=> ActiveRecord::Base
>> user.class.superclass.superclass.superclass
=> Object
>> user.class.superclass.superclass.superclass.superclass
=> BasicObject
>> user.class.superclass.superclass.superclass.superclass.superclass
=> nil

つまり継承階層は次のようになっています。
User
<< ApplicationRecord(abstract)
 << ActiveRecord::Base
  << Object
   << BasicObject

4.4.5 ユーザークラス

1. Userクラスで定義されているname属性を修正して、first_name属性とlast_name属性に分割してみましょう。また、それらの属性を使って "Michael Hartl" といった文字列を返すfull_nameメソッドを定義してみてください。最後に、formatted_emailメソッドのnameの部分を、full_nameに置き換えてみましょう (元々の結果と同じになっていれば成功です)

example_user.rb

class User
  attr_accessor :first_name, :last_name, :email

  def initialize(attributes = {})
    @first_name  = attributes[:first_name]
    @last_name = attributes[:last_name]
    @email = attributes[:email]
  end

  def formatted_email
    "#{full_name} <#{@email}>"
  end

  def full_name
    @first_name + " " + @last_name
  end
end

2. "Hartl, Michael" といったフォーマット (苗字と名前がカンマ+半角スペースで区切られている文字列) で返すalphabetical_nameメソッドを定義してみましょう。

example_user.rb

class User
  attr_accessor :first_name, :last_name, :email

  def initialize(attributes = {})
    @first_name  = attributes[:first_name]
    @last_name = attributes[:last_name]
    @email = attributes[:email]
  end

  def formatted_email
    "#{full_name} <#{@email}>"
  end

  def full_name
    @first_name + " " + @last_name
  end

  def alphabetical_name
    @last_name + ", " + @first_name
  end
end

Console

>> user = User.new(first_name: "Michael", last_name: "Hartl", email: "mhartl@example.com")
=> #<User:0x00007fe87103a500 @first_name="Michael", @last_name="Hartl", @email="mhartl@example.com">
>> user.alphabetical_name
=> "Hartl, Michael"

3. full_name.splitとalphabetical_name.split(’, ’).reverseの結果を比較し、同じ結果になるかどうか確認してみましょう。

>> user.full_name.split == user.alphabetical_name.split(', ').reverse
=> true

同じ結果になることを確認できました。

4.5 最後に

4.5.1 本章のまとめ

演習なし。