在`initialize`方法之外声明一个实例变量

问题描述 投票:3回答:3

我被教导用def initialize声明我的实例变量。我的印象是我只能在我的initialize方法中声明实例变量。

尽管如此,我在@foo方法之外声明了一个实例变量initialize,并使其按预期工作:

class FooBar
    def initialize(bar)
        @bar = bar
    end

    def foo_as_instance_var
        @foo = @bar.split(' ')
        @foo
    end
end

x = "something wicked this way comes"
y = FooBar.new(x)
puts y.foo_as_instance_var

为什么我能在initialize方法之外声明一个实例变量?由于我可以在任何方法中声明实例变量,我是否应该遵循最佳实践规则,关于在何处声明实例变量(即在initialize中声明它们)或者它无关紧要?

ruby initialization instance-variables
3个回答
4
投票

我被教导用def initialize声明我的实例变量

由于initialize是对象生命周期中的第一个实例方法调用,因此通常在那里声明实例变量以确保正确初始化的变量。这也是我期望在阅读代码时定义实例变量的第一个地方。

我的印象是我只能在初始化方法中声明实例变量。

没有这样的限制。您可以在实例中的任何位置声明实例变量。

常见的用途是memoization

class FooBar
  def foo
    @foo ||= expensive_operation
  end
end

在第一次调用时,这将评估expensive_operation并将结果分配给@foo。在随后的调用中,返回@foo

另一个流行的例子是Rails,它使用实例变量将数据从控制器传递到其视图:

class FooController < ApplicationController
  def index
    @foos = Foo.all
  end
end

我应遵循的最佳实践规则是关于在何处声明实例变量

这取决于他们的目的(见上面的例子)。作为一般规则,以避免未定义变量(nil错误)的方式声明它们并构造代码,以便易于阅读/遵循。


3
投票

只是为了增加Stefan的优秀答案

我被教导用def initialize声明我的实例变量

ruby新手做的一个常见错误是这样的:

class Person
  @name = "John"

  def introduce
    puts "Hi, my name is #{@name}"
  end
end

然后他们想知道为什么他们的名字没有印刷。为了使这个工作,可以在初始化器中设置变量@name,就像指令所说的那样。


2
投票

让我们从最大的误称开始 - 在Ruby中没有单独的声明变量的步骤 - 在设置变量时声明变量。

有什么区别?以Java为例:

public class Bicycle {

    private int cadence;
    private int gear;
    private int speed;

    public Bicycle(int startCadence, int startSpeed, int startGear) {
        gear = startGear;
        cadence = startCadence;
        speed = startSpeed;
    }
}

在我们在初始化程序(Bicycle)中设置它们之前,我们必须声明所有实例变量。 Ruby中的相同代码读取:

class Bicycle
  def initialize(cadence, speed, gear)
    @cadence = cadence
    @speed = speed
    @gear = gear
  end
end

没有声明 - 仅限作业。 Ruby甚至可以让你访问没有设置而没有错误的实例变量。

irb(main):003:0> @not_set
=> nil

你不能(通常)在必须定义变量*的语言中这样做。

我被教导用def initialize声明我的实例变量。我的印象是我只能在初始化方法中声明实例变量。

废话。您可以在任何地方分配实例变量它通常用于从setter和mutator(改变对象的方法)到工厂方法(返回实例的类方法)或你正在改变对象状态的任何地方。

class Book 

  def initialize(title, author)
    @title = title
    self.author = author # calls the setter.
  end

  # A factory method
  def create_from_csv(filename)
    # ...
  end

  # A very contrived setter
  def author=(author)
    @author = "#{author.forename.upcase}. #{author.surname}"
  end

  # a mutator
  def out_of_print!
    @out_of_print = true
    @last_printed = Date.today
  end
end

然而,initialize方法是您应该处理初始化对象(duuh)的地方,因此是设置初始值的明显位置。

© www.soinside.com 2019 - 2024. All rights reserved.