Бесконечные выходы от итератора - PullRequest
4 голосов
/ 05 марта 2009

Я пытаюсь выучить немного рубина. Представьте, что я зацикливаюсь и выполняю длительный процесс, и в этом процессе я хочу получить счетчик столько, сколько потребуется.

Чтобы я мог сделать:

a=['|','/','-','\\']
aNow=0
# ... skip setup a big loop
print a[aNow]
aNow += 1
aNow = 0 if aNow == a.length
# ... do next step of process
print "\b"

Но я думал, что будет чище:

def spinChar
  a=['|','/','-','\\']
  a.cycle{|x| yield x}
end
# ... skip setup a big loop
print spinChar
# ... do next step of process
print "\b"

Конечно, вызов spinChar хочет блок. Если я дам ему блок, он будет висеть бесконечно.

Как я могу получить только следующий результат этого блока?

Ответы [ 7 ]

6 голосов
/ 05 марта 2009

Ruby's yield не работает так, как хотелось бы вашему примеру. Но это может быть хорошим местом для закрытия :

def spinner()
  state = ['|','/','-','\\']
  return proc { state.push(state.shift)[0] }
end

spin = spinner

# start working
print spin.call
# more work
print spin.call
# etc...

На практике я думаю, что это решение может быть слишком «умным» для его же блага, но понимание идеи Proc s может быть полезно в любом случае.

5 голосов
/ 15 февраля 2013

Я думаю, вы были на правильном пути с cycle. Как насчет чего-то вроде этого:

1.8.7 :001 > spinner = ['|','/','-','\\'].cycle
 => #<Enumerable::Enumerator:0x7f111c165790> 
1.8.7 :002 > spinner.next
 => "|" 
1.8.7 :003 > spinner.next
 => "/" 
1.8.7 :004 > spinner.next
 => "-" 
1.8.7 :005 > spinner.next
 => "\\" 
1.8.7 :006 > spinner.next
 => "|" 
5 голосов
/ 05 марта 2009

Мне нравятся все эти предложения, но я нашел Генератор в стандартной библиотеке, и я думаю, что это больше похоже на то, что я хотел сделать:

spinChar=Generator.new{ |g|
  ['|','/','-','\\'].cycle{ |x|
    g.yield x
  }
}
#then
spinChar.next
#will indefinitly output the next character.

Обычные array index приращения с модулем на замороженном массиве кажутся самыми быстрыми.

Нить Влада изящная, но не совсем то, что я хотел. А в spinner class приращение в одну строку было бы лучше, если бы Ruby поддерживал i++, как GLYPHS[@i++%GLYPHS.length]

Макс spinner closure с push-shift мне кажется немного интенсивным, но результирующий синтаксис почти такой же, как этот Генератор. По крайней мере, я думаю, что это закрытие с proc там.

Чака with_spinner на самом деле довольно близко к тому, что я хотел, но зачем ломать голову, если вам не нужно использовать генератор, как указано выше.

Вадим, спасибо за указание, что generator будет медленным.

"Here's a test of 50,000 spins:"
                   user       system     total       real
"array index"      0.050000   0.000000   0.050000 (  0.055520)
"spinner class"    0.100000   0.010000   0.110000 (  0.105594)
"spinner closure"  0.080000   0.030000   0.110000 (  0.116068)
"with_spinner"     0.070000   0.030000   0.100000 (  0.095915)
"generator"        6.710000   0.320000   7.030000 (  7.304569)
4 голосов
/ 05 марта 2009

Не думаю, что вы понимаете, что yield делает в Ruby. Он не возвращает значение из блока - он передает значение в блок, который вы передали методу включения.

Я думаю, вы хотите что-то более похожее на это:

def with_spinner
  a=['|','/','-','\\']
  a.cycle do |x|
    print x
    $stdout.flush # needed because standard out is usually buffered
    yield # this will call the do-block you pass to with_spinner
  end
end

with_spinner do
  #process here
  #break when done
end
0 голосов
/ 13 октября 2012

хе-хе, ответ над моим - все грязное.

a=['|','/','-','\\']
a << a
a.each {|item| puts item}
0 голосов
/ 05 марта 2009

Когда-то я написал массив. Но это не просто массив, это массив с указателем, так что вы можете вызывать next foreverrr! http://gist.github.com/55955

Соедините этот класс с простым итератором или циклом, и вы - золотой.

 a = Collection.new(:a, :b, :c)
 1000.times do |i|
   puts a.current
   a.next
 end
0 голосов
/ 05 марта 2009

Ваш код немного наизнанку, если вы извините, я так сказал. :)

Почему бы и нет:

class Spinner
  GLYPHS=['|','/','-','\\']
  def budge
    print "#{GLYPHS[@idx = ((@idx || 0) + 1) % GLYPHS.length]}\b"
  end
end

spinner = Spinner.new
spinner.budge
# do something
spinner.budge
spinner.budge
# do something else
spinner.budge

Теперь , если вы хотите что-то вроде:

with_spinner do
  # do my big task here
end

... тогда вам придется использовать многопоточность :

def with_spinner
  t = Thread.new do
    ['|','/','-','\\'].cycle { |c| print "#{c}\b" ; sleep(1) }
  end
  yield
  Thread.kill(t) # nasty but effective
end
...