Возвратите два и более значений из метода

Есть ли возможность вернуть несколько значений из метода? Что-то вроде этого:

def do()
  return ["a", 10, SomeObject.new]
end

[a, b, c] = do

Ответ 1

def sumdiff(x, y)
  return x+y, x-y
end
#=> nil

sumdiff(3, 4)
#=> [7, -1]

a = sumdiff(3,4)
#=> [7, -1]
a
#=> [7, -1]

a,b=sumdiff(3,4)
#=> [7, -1]
a
#=> 7
b
#=> -1

a,b,c=sumdiff(3,4)
#=> [7, -1]
a
#=> 7
b
#=> -1
c
#=> nil

Ответ 2

Ruby имеет ограниченную форму связывания destructuring:

ary = [1, 2, 3, 4]
a, b, c = ary
p a # => 1
p b # => 2
p c # => 3

a, b, *c = ary
p c # => [3, 4]

a, b, c, d, e = ary
p d # => 4
p e # => nil

Он также имеет ограниченную форму связывания структурирования:

 a = 1, 2, 3
 p a # => [1, 2, 3]

Вы можете объединить эти две формы следующим образом:

a, b = b, a # Nice way to swap two variables

a, b = 1, 2, 3
p b # => 2

def foo; return 1, 2 end
a, b = foo
p a # => 1
p b # => 2

Есть несколько других вещей, которые вы можете сделать с привязкой деструктурирования/структурирования. Я не показывал с помощью оператора splat (*) с правой стороны. Я не показывал гнездование (используя скобки). Я не показал, что вы можете использовать bind destructuring в списке параметров блока или метода.

Вот только закуска:

def foo(((a, b, c, d), e, *f), g, *h)
  local_variables.sort.each do |lvar| puts "#{lvar} => #{eval(lvar).inspect}" end
end

foo([[1, 2, 3], 4, 5, 6], 7, 8, 9)
# a => 1
# b => 2
# c => 3
# d => nil
# e => 4
# f => [5, 6]
# g => 7
# h => [8, 9]

Ответ 3

В то время как возвращение нескольких значений часто полезно, я обычно нахожу его указателем на новое требование к объекту.

То есть, я обычно обнаруживаю, что эти возвращаемые значения тесно связаны в смысле/контексте и передаются как таковые. Поэтому в этих случаях я бы создал новый объект, чтобы связать их вместе. Это особый запах кода, который я узнал.

Ответ 4

Вы можете добиться этого, возвращая массив, например

def sumdiff(x, y)
    [x+y, x-y]
end

который функционально эквивалентен

def sumdiff(x, y)
    return x+y, x-y
end