未定义的方法...用于类(NoMethodError)

时间:2012-02-28 18:25:19

标签: ruby

我刚刚开始学习ruby而且我正在编写一个简单的程序,但是我遇到了错误undefined method 'send_for_beer' for Person:Class (NoMethodError) 这是一个代码:

class Person
    @iq = 0
    @speed = 0
    @power = 0
    @beauty = 0
    def initialize (iq, speed, power, beauty)
        @iq = iq
        @speed = speed
        @power = power
    end

    def send_for_beer
        result @iq * 2 + @speed * 10 + @power * 5 + @beauty
        return result
    end
end

number_of_people = 3
person_array = Array.new(number_of_people, Person)
n = 0
beer_person = 0
beer_cof = 0
number_of_people.times do 

    ............
    person_array.push(Person.new(iq, speed, power, beauty))

    if person_array[n].send_for_beer > beer_cof     <-----here is an error
        beer_cof = person_array[n].send_for_beer
        beer_person = n
    end
    n = n+1
end

3 个答案:

答案 0 :(得分:8)

这是你的问题:

person_array = Array.new(number_of_people, Person)

简而言之,不要像这样制作数组。使用[]文字语法。这回归的是:

[Person, Person, Person]

这是对Person类的3个引用,而不是实例。然后你会做:

person_array.push(Person.new(iq, speed, power, beauty))

你最终得到:

[Person, Person, Person, person_instance]

因此,当您迭代并在第一个项目上调用send_for_beer时,它确实具有该方法,因为send_for_beer是您在类对象上错误地调用的实例方法。

这里的解决方法是简单地将person_array分配给空数组文字,然后将内容推送给它。

person_array = []

还有一个小样式注释:<<通常比Array#push更受欢迎,这使得数组的填充看起来更像这样。

person_array << Person.new(iq, speed, power, beauty)

Ruby还支持隐式返回方法中的最后一个表达式。所以你不需要return result。相反,只需将返回值计算为方法中的唯一一行。

def send_for_beer
  @iq * 2 + @speed * 10 + @power * 5 + @beauty
end

实例变量也不像那样工作。如果直接在类主体中有@name,则不会为每个实例初始化实例变量。你实际上是在类对象上设置实例变量(这很奇怪,我知道)。您实际需要做的是从任何实例方法设置它们,通常是initialize,您在此处执行此操作。因此,您可以在此处完全删除类级别的实例变量设置。

答案 1 :(得分:1)

我认为方法send_for_beer中存在语法错误,变量=的效果中缺少result符号。

顺便说一句,该方法可以写成

def send_for_beer
    @iq * 2 + @speed * 10 + @power * 5 + @beauty
end

答案 2 :(得分:0)

如果您有一个固定长度的数组,则可以提供一个块来为每个元素创建一个新的Person对象。您可以按如下方式重写person_array行:

person_array = Array.new(number_of_people) { Person.new(0, 0, 0, 0) }

将以下行添加到班级的顶部。

attr_writer(:iq, :speed, :power, :beauty)

这段代码可以修改数组中的对象。

person_array.each do |p|
    p.iq, p.speed, p.power, p.beauty = rand(20) + 1, rand(5) + 1, 1
    p.beauty = 10 if (rand(2) == 0)
end