Есть ли более функциональный способ, чем использование пользовательских операторов case, чтобы написать это в Ruby? - PullRequest
3 голосов
/ 23 мая 2011

Представьте себе следующий код:

class SimpleLetter
  def values
    ("a" .. "z").to_a
  end

  def ===(other)
    values.include?(other)
  end
end

class Vowel < SimpleLetter
  def values
    ["a","e","i","o","u"]
  end
end

class Consonant < SimpleLetter
  def values
    super - Vowel.new.values
  end
end

objects = ("a" .. "f").to_a + (1 .. 5).to_a

objects.each do |letter|
  case letter
    when Vowel.new
      puts "#{letter} it's a vowel"
    when Consonant.new
      puts "#{letter} it's a consonant"
    else
      puts "#{letter} it's something else"
  end
end

Я мог бы вместо этого выбрать любые другие классы, я просто использую их в качестве примера. Мне очень нравятся match и экстракторы Scala, и я подумал, что это может быть хорошим способом написать то же самое в Ruby. Есть ли лучший способ написания вышеупомянутого без необходимости создания новых объектов, чтобы я мог вызвать их === метод?

Просто чтобы избежать ненужных постов, да, я знаю, что смогу сделать это:

case letter
  when ["a","e","i","o","u"].include?(letter)
    # ...
end

Ответы [ 5 ]

5 голосов
/ 24 мая 2011

Вам не нужны классы для персонажей. Установите их как массивы и используйте оператор splat в операторе case.

SimpleLetter = ("a" .. "z").to_a
Vowel        = %w[a e i o u]
Consonant    = SimpleLetter - Vowel

(("a" .. "f").to_a + (1 .. 5).to_a).each do |letter|
  case letter
    when *Vowel
      puts "#{letter} it's a vowel"
    when *Consonant
      puts "#{letter} it's a consonant"
    else
      puts "#{letter} it's something else"
  end
end
4 голосов
/ 24 мая 2011

=== работает и на блоках:

Letters = ('a'..'z').to_a
Vowels = ['a','e','i','o','u']
Consonants = Letters - Vowels

Vowel = lambda { |x| Vowels.include? x }
Consonant = lambda { |x| Consonants.include? x }

objects = ("a" .. "f").to_a + (1 .. 5).to_a

objects.each do |object|
  case object
    when Vowel
      puts "#{object} is a vowel."
    when Consonant
      puts "#{object} is a consonant."
    else
      puts "#{object} is an object."
  end
end
3 голосов
/ 23 мая 2011

Вы можете использовать методы класса вместо методов экземпляра:

class SimpleLetter
    def self.values
        ("a" .. "z").to_a
    end 

    def self.===(other)
        values.include?(other)
    end 
end

class Vowel < SimpleLetter
    def self.values
        ["a","e","i","o","u"]
    end 
end

class Consonant < SimpleLetter
    def self.values
        super - Vowel.values
    end 
end

objects = ("a" .. "f").to_a + (1 .. 5).to_a

objects.each do |letter|

    case letter
        when Vowel
            puts "#{letter} it's a vowel"
        when Consonant
            puts "#{letter} it's a consonant"
        else
            puts "#{letter} it's something else"
    end 

end
2 голосов
/ 24 мая 2011

Код, который вы меня смутили, потому что SimpleLetter должен быть одной буквой, а не целым алфавитом.

Я бы наполовину испытал желание сделать следующее, хотя monkeypatching немного рискованно:

module Voweliness
  def vowel?
    self =~ /[aeiou]/i
  end

  def consonant?
    (self =~ /[a-z]/i and not vowel?)
  end
end

class String
  include Voweliness
end


objects.each do |letter|
  case
    when letter.vowel?
      puts "#{letter} is a vowel"
    when letter.consonant?
      puts "#{letter} is a consonant"
    else
      puts "#{letter} is something else"
  end
end
1 голос
/ 24 мая 2011

У вас уже есть несколько хороших ответов (например, sawa), так что я включаю просто для удовольствия без утверждения case:

SIMPLE_LETTER = [*"a" .. "z"]
VOWEL        = %w[a e i o u]
CONSONANT    = SIMPLE_LETTER - VOWEL

[*?a..?f,*1..5].each do |letter|
  letter_class = %w(vowel consonant).select { |c| Object.const_get(c.upcase).include? letter}.first
  puts "'#{letter}': #{ letter_class || "something else"}"
end

Выход:

'a': vowel
'b': consonant
'c': consonant
'd': consonant
'e': vowel
'f': consonant
'1': something else
'2': something else
'3': something else
'4': something else
'5': something else

Мультисплат и символьные литералы работают только в 1.9.

...