2

当使用带有部分布局的rails以及使用yield关键字编码为迭代器的辅助方法时,我看到了一些奇怪的行为。我希望有人可以:

  1. 解释发生了什么以及为什么我得到重复的渲染,也许
  2. 建议一种替代方法,希望不仅仅是将我的辅助方法重新编码为一个返回列表的简单函数(我已经将其作为临时解决方法)

因此,如果我在我的 rails 3 应用程序中创建以下 3 件事,我会得到意想不到的输出。

[更新] 我测试了以下组合:

Rails 3.0.0 + erb (has this issue)
Rails 3.0.0 + haml (OK)
Rails 3.0.3 + erb (has this issue)
Rails 3.0.3 + haml (OK)

所以也许这是一个 erb 与 haml 的事情,但是当我最初发现它时,它是在 haml 模板上。嗯……有谁知道怎么回事???

A) 一个看起来像这样的主模板 (app/views/main/index.html.erb)

  <h1>Main#index</h1>
  <p>This is content from main#index before the partial template rendering
  <%= render :partial => "partial" %>
  <p>This is content from main#index after the partial template rendering.</p>

B) 像这样的辅助方法 (app/helpers/main_helper.rb)

  module MainHelper

    def my_iterator
      yield 1
      yield 2
      yield 3
      yield 4
    end
  end

C)像这样的部分模板(app/views/main/_partial.html.erb)

  <% my_iterator do |x| %>
  <p>iterator running with <%= x %></p>
  <% end %>

当我在浏览器中查看结果时,我看到“iterator running with”总共阻塞了 8 次(1 2 3 4 1 2 3 4)。我已经确定它是 my_iterator 中的 yield 与 rails 部分模板机制有关。如果我对 my_iterator 进行如下编码,则输出与我预期的一样。(我还需要更改我的部分模板来执行 my_iterator.each)

def my_iterator
  logger.debug("my_iterator called")
  return [1, 2, 3, 4]
end

有没有办法对此进行编码,这样我就不会使用rails并获得重复的渲染,但仍然可以使用yield将辅助方法编码为迭代器?另外,有人可以准确解释重复渲染是如何发生的吗?

4

3 回答 3

3

我只是遇到了一个非常相似的问题,并且设法解决了它。我相信如果将上面的原始助手重写为这样,它会按预期工作......

module MainHelper
  def my_iterator(&block)
    block.call(1)
    block.call(2)
    block.call(3)
    block.call(4)
  end
end

如果您调用concat(capture(&block))concat(block.call). 在这两种情况下,只需放下concat(),重复的渲染就会消失。

于 2011-06-22T21:18:06.810 回答
1

我遇到了类似的问题,它只出现在生产中,而不是开发中。我把这个缩小到

    config.action_view.cache_template_loading = true

在我的 development.rb 中。当设置为 true 时,我在我的 haml 文件中看到了一个元素。通过添加和删除块,我将有问题的代码缩小到

    = if @user.bio
      = @user.bio

这导致重新打印之前的整个块。只需将 = 替换为 - 即可解决此问题。调试很尴尬,因为我必须在生产模式下进行调试。

我的建议是在逻辑运算符上寻找 errant = 输出。

    - if @user.bio
      = @user.bio
于 2011-09-13T19:39:11.103 回答
0

在帮助程序中使用 content_for 将在每次迭代中附加内容。您可以编写一个 ApplicationHelper 方法来在助手中产生内容,如下所示:

def yield_content(content_key)
  view_flow.content.delete(content_key)
end

然后在您的其他帮助文件中使用 yield_content 而不是 content_for。

于 2012-11-13T22:30:54.397 回答