Я начал смотреть проект Euler site как способ узнать Haskell и улучшить свой Python и Ruby. Я думаю, что версии Haskell и Python в порядке, но я уверен, что для Ruby должен быть более чистый способ.
Это не о том, как сделать один язык похожим на другой.
Это Проблема 1:
Q: добавьте все натуральные числа ниже тысячи, кратные 3 или 5.
Haskell:
sum [ x | x <- [1..999], mod x 3 == 0 || mod x 5 == 0 ]
Python:
sum ( [ x for x in range(1,1000) if x % 3 == 0 or x % 5 == 0 ] )
Ruby:
(1..999) . map {|x| x if x % 3 == 0 || x % 5 == 0 } . compact . inject(:+)
Все они дают одинаковый ответ.
ОК, поэтому Python может стать:
sum ( x for x in range(1,1000) if x % 3 == 0 or x % 5 == 0 )
теперь это генератор (хорошо, поскольку мы не сохраняем список)
но еще больше удовольствия:
sum( set(range(0,1000,3)) | set(range(0,1000,5)) )
По какой-то причине я снова смотрел на это и пробовал подход суммирования, который должен быть постоянным. В Python 3:
def step_sum(mn,mx,step):
amax = mx - (mx - mn) % step
return (mn + amax) * ((1 + ((amax - mn) / step)) / 2)
step_sum(3,999,3) + step_sum(5,999,5) - step_sum(15,999,15)
Ruby может стать:
(1..999) . select {|x| x % 3 == 0 || x % 5 == 0} . inject(:+)
или
(1..999) . select {|x| x % 3 == 0 or x % 5 == 0} . reduce(:+)
Я предполагаю, что в отличие от map
, select
не производит 'nul', и поэтому нет необходимости вызывать compact
. хороший.
Haskell также может быть:
let ƒ n = sum [0,n..999] in ƒ 3 + ƒ 5 - ƒ 15
или чтобы быть понятнее:
let ƒ n = sum [ 0 , n .. 999 ] in ƒ 3 + ƒ 5 - ƒ (lcm 3 5)
как функция, которая позволяет нам предоставить два числа:
ƒ :: (Integral a) => a -> a -> a
ƒ x y = let ƒ n = sum [0,n..999] in ƒ x + ƒ y - ƒ (lcm x y)