Calling ruby method without instantiating class

2019-06-26 18:33发布

If I call a method on a rails active model method like so:

class Foo < ActiveRecord::Base

end

Foo.first

I'll get back the first active record. I don't have to instantiate the class.

But if I create my own class and call a method, I get an exception:

class Person < ActiveRecord::Base
  def greeting
    'hello'
  end
end

Person.greeting 

#EXCEPTION: undefined method `greeting' for Person:Class

How can I make that problem go away?

5条回答
SAY GOODBYE
2楼-- · 2019-06-26 18:51
class Person < ActiveRecord::Base
    def self.greeting
      'hello'
    end
end
查看更多
smile是对你的礼貌
3楼-- · 2019-06-26 18:55

There are several kinds of methods. The two most important ones are: instance methods and class instance methods.

Foo.first is a class instance method. It works on a class instance (Foo, in this case). If it stores some data inside the class, that data is shared globally across your program (because there's only one class with name Foo (or ::Foo, to be exact)).

But your greeting method is an instance method, it requires object instance. If your greeting method will use Person's name, for example, it has to be instance method, so that it will be able to use instance data (the name). If it doesn't use any instance-specific state and you really meant it to be a class instance method, then use the self "prefix".

class Person < ActiveRecord::Base
  def self.greeting
    'hello'
  end
end
查看更多
We Are One
4楼-- · 2019-06-26 18:57

Try class methods:

class Person < ActiveRecord::Base
  def self.greeting
    'hello'
  end
end

Or another syntax:

class Person < ActiveRecord::Base
  class << self
    def greeting
      'hello'
    end
  end
end
查看更多
仙女界的扛把子
5楼-- · 2019-06-26 18:57
class Person < ActiveRecord::Base
  def Person.greeting
    'hello'
  end
end

Will work too. I like it because it is very clear what it does; it will result in an error when you decide to rename the Person class however.

查看更多
贼婆χ
6楼-- · 2019-06-26 18:59

To do a static method try this:

class MyModel
    def self.do_something
        puts "this is a static method"
    end
end

MyModel.do_something  # => "this is a static method"
MyModel::do_something # => "this is a static method"

查看更多
登录 后发表回答