Закрытие не работает

Если блок является закрытием, почему делает этот код, не работает, и как заставить его работать?

def R(arg)
  Class.new do
    def foo
      puts arg
    end
  end
end

class A < R("Hello!")
end

A.new.foo #throws undefined local variable or method `arg' for #<A:0x2840538>
17
задан Andrew Marshall 13 February 2012 в 22:18
поделиться

2 ответа

Блоки являются закрытиями, и arg действительно доступен внутри блока Class.new. Он просто недоступен внутри метода foo, потому что def начинает новую область видимости. Если вы замените def на define_method, который принимает блок, вы увидите желаемый результат:

def R(arg)
    Class.new do
        define_method(:foo) do
           puts arg
        end
    end
end

class A < R("Hello!")
end

A.new.foo # Prints: Hello!
26
ответ дан 30 November 2019 в 12:27
поделиться

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

def R(arg)
  c = Class.new

  # Send the block through as a closure, not as an inline method
  # definition which interprets variables always as local to the block.
  c.send(:define_method, :foo) do
    arg
  end

  c
end

class A < R("Hello!")
end

puts A.new.foo.inspect
# => "Hello!"
5
ответ дан 30 November 2019 в 12:27
поделиться
Другие вопросы по тегам:

Похожие вопросы: