Ruby on Rails:在视图中使用帮助程序初始化实例变量

我遇到了某种类型的范围问题,它阻止实例变量被视图调用的助手正确初始化。

#sample_controller.rb class SampleController < ApplicationController def test end end #application_controller.rb helper_method :display def display if not defined? @display return @display = "a" else return @display += "a" end end #test.html.erb     

当显示样品/测试时,它会在“评估nil.reverse”时出现错误。 这是令人惊讶的,因为显示的前两个调用应该初始化@display我会想到。 如果删除,则输出为“a aa”,表示辅助方法正在设置@display实例变量,但视图中无法访问它。

如果控制器被修改成为(使用原始视图代码):

 class SampleController < ApplicationController def test display end end 

输出变成“aa aaa aa”。 如果我在控制器中进行2次调用,我会得到“aaa aaaa aa aa”。 因此,似乎只有控制器中的调用才会修改SampleController实例变量,而视图中的调用将修改视图无法访问的ApplicationController的实例变量。

这是Rails中的错误还是我的误解,这是出于某种原因预期的function?

我遇到这个bug的上下文是试图创建一个logged_in? ApplicationController方法,在第一次调用时设置@user变量,如果用户登录则返回true或false。除非在尝试在视图中使用它之前在控制器中添加了不必要的调用,否则这将无效。

渲染时有两个@display ivars,一个在控制器中,一个在视图中。 视图无法访问控制器中的视图,反之亦然。 在渲染开始时,Rails将所有控制器ivars复制到视图中。 但那时@display不存在,不会被复制。 在任何ruby程序中,对@my_undefined_ivar的调用将返回nil – 这正是发生在你身上的事情。

可能令人困惑,这里有另一种说法。 Rails将ivars从控制器复制到渲染开始时的视图中。 因此,视图的ivar中的更改不会反映在控制器ivar中,反之亦然。 您定义的帮助程序允许视图调用控制器上的方法,因此控制器的ivar可以作为返回值传递回视图,但对控制器ivar本身的更改不会反映在另一个视图中,ivar。

在这种情况下只需使用辅助方法,忘记视图中的@display。

我想这可能是本周我见过的第二个保留字问题……

提示:您几乎不需要在Ruby中使用return关键字。 另外,你defined?的使用defined? 我吓坏了…可能值得这样做:

  if @display.nil? @display = "a" else @display += "a" end