Ruby String для имени класса - PullRequest
34 голосов
/ 14 марта 2011

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

"general_systems".camelize.singularize = Class.new < ActiveRecord::Base

Однако я получаю сообщение об ошибке:

undefined method `singularize=' for "GeneralSystems":String

Я также пытался constantize строка

>> foo = "general_systems".camelize.singularize
=> "GeneralSystem"
>> foo.constantize
NameError: uninitialized constant GeneralSystem
    from /var/lib/gems/1.9.1/gems/activesupport-3.0.5/lib/active_support/inflector/methods.rb:124:in `block in constantize'
    from /var/lib/gems/1.9.1/gems/activesupport-3.0.5/lib/active_support/inflector/methods.rb:123:in `each'
    from /var/lib/gems/1.9.1/gems/activesupport-3.0.5/lib/active_support/inflector/methods.rb:123:in `constantize'
    from /var/lib/gems/1.9.1/gems/activesupport-3.0.5/lib/active_support/core_ext/string/inflections.rb:43:in `constantize'
    from (irb):4
    from /usr/bin/irb:12:in `<main>'
>> foo.constantize = Class.new
NoMethodError: undefined method `constantize=' for "GeneralSystem":String
    from (irb):5
    from /usr/bin/irb:12:in `<main>'

Любая помощь будет принята с благодарностью.

Ответы [ 7 ]

33 голосов
/ 30 декабря 2012

Если вы используете Rails, он предоставляет метод с именем #constantize, который будет работать:

irb(main):017:0> Object.const_get 'House::Owns'
NameError: wrong constant name House::Owns

'House::Owns'.constantize
=> House::Owns
32 голосов
/ 14 марта 2011

Как то так?

>> Object.const_set("general_systems".classify, Class.new)
=> GeneralSystem
>> GeneralSystem.new
=> #<GeneralSystem:0x105b0f738>
13 голосов
/ 14 марта 2011
klazz = Class.new(ActiveRecord::Base) do
  def do_something_fun(param1)
    param1.have_fun!
  end
end

klazz_name = "general_systems".singularize.classify
Object.const_set(klazz_name, klazz)
8 голосов
/ 14 марта 2011

Посмотрите на этот пример из "Книги Ruby", включенной в установщик Ruby 1.9.

puts("What shall we call this class?> ")
className = gets.strip().capitalize()
Object.const_set(className,Class.new)
puts("I'll give it a method called > 'myname'" ) 
className = Object.const_get(className)
className::module_eval{
  define_method(:myname){ 
    puts("The name of my class is '#{self.class}'" ) 
 } }
 x = className.new x.myname
4 голосов
/ 26 сентября 2012

Если ваша строка содержит пространство имен, вы можете использовать:

require 'rubygems'
require 'active_support/inflector'
parent=String # using String to get a self contained example
# require 'active_record'   # uncomment for ActiveRecord::Base as parent class
# parent=ActiveRecord::Base # uncomment for ActiveRecord::Base as parent class
namespace = 'A::B::general_systems'.split('::') 
class_name = namespace.pop 
namespace = namespace.inject(Object) do |mod, name|
  if mod.constants.collect{|sym| sym.to_s}.include? name.classify
    mod.const_get name.classify
  else
    mod.const_set name.classify, Module.new
  end
end

klass = if namespace.constants.include? class_name.classify
  namespace.const_get class_name.classify
else
  namespace.const_set class_name.classify, Class.new(parent)
end

object = klass.allocate # allocate new object of klass
object.send :initialize # initialize object
object.class
object.class.superclass
1 голос
/ 15 декабря 2018

Дано:

class Banana
end

Вы можете получить класс в обычном Ruby с помощью:

Object.const_get 'Banana'
=> Banana

или если вы используете Rails:

'Banana'.constantize
=> Banana
0 голосов
/ 14 марта 2011

попробуй

>> "general_systems".classify
=> "GeneralSystem"
...