我遇到某种类型的范围问题,阻止实例变量被视图调用的助手正确初始化。
#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
<%= display %>
<%= display %>
<%= @display %>
<%= @display.reverse %>
当显示样品/测试时,它会在评估nil.reverse时出错“。这是令人惊讶的,因为显示的前两个调用应该初始化@display我会想到。如果&lt;%= @ display.reverse%&gt;删除输出是“a aa”,表示@display实例变量是由辅助方法设置的,但在视图中无法访问它。
如果控制器被修改为(使用原始视图代码):
class SampleController < ApplicationController
def test
display
end
end
输出变成“aa aaa a a”。如果我在控制器中进行2次调用,我会得到“aaa aaaa aa aa”。因此,似乎只有控制器中的调用才会修改SampleController实例变量,而视图中的调用将修改视图无法访问的ApplicationController的实例变量。
这是Rails中的错误还是我误解了,这是出于某种原因预期的功能?
我遇到此错误的上下文是尝试创建一个logged_in? ApplicationController方法,在第一次调用时设置@user变量,如果用户登录则返回true或false。除非在尝试在视图中使用它之前在控制器中添加了不必要的调用,否则这将无效。
答案 0 :(得分:6)
渲染时有两个@display ivars,一个在控制器中,另一个在视图中。视图无法访问控制器中的视图,反之亦然。在渲染开始时,Rails将所有控制器ivars复制到视图中。但那时@display不存在,不会被复制。在任何ruby程序中,对@my_undefined_ivar的调用都将返回nil - 这正是发生在你身上的事情。
可能令人困惑,这里有另一种说法。 Rails将ivars从控制器复制到渲染开始时的视图中。因此,视图的ivar中的变化不会反映在控制器ivar中,反之亦然。您定义的帮助程序允许视图调用控制器上的方法,以便控制器的ivar可以作为返回值传递回视图,但是对控制器ivar本身的更改不会反映在另一个视图中,ivar。
在这种情况下只使用辅助方法,忘记视图中的@display。
答案 1 :(得分:0)
我认为这可能是本周我见过的第二个reserved words问题......
提示:您几乎不需要在Ruby中使用return
关键字。此外,您使用defined?
让我感到非常恐惧......可能值得尝试:
if @display.nil?
@display = "a"
else
@display += "a"
end