ruby-on-rails - Rails response_with : how does it work?

标签 ruby-on-rails ruby-on-rails-3

我一直在这里和那里阅读关于 respond_with 有多酷的文章。方法在 Rails 3 中。但我什至在 Rails API 中或通过搜索源都找不到对它的引用。任何人都可以向我解释它是如何工作的(您可以使用哪些选项等),或者将我指向它实际实现的位置,以便我可以自己仔细阅读代码?

最佳答案

Rails 4.2+ 更新
#respond_with::respond_to (n.b. 类方法)是 不再是 Rails 的一部分 .他们被迁移到第三方responders从 Rails 4.2 开始的 gem(release notes/commit 日期为 2014 年 8 月)。虽然响应者默认不包含在 Rails 中,但它是 Devise 的依赖项,因此在许多 Rails 应用程序中都可用。
#respond_to然而,实例方法是 仍然是 Rails 的一部分 (撰写本文时为 5.2rc1)。

ActionController::MimeResponds 的官方 Rails API 文档解释如何 #respond_to作品。 #respond_with 的原始 Rails Guides 文档注释和 ::respond_to仍然可以在 responders gem source code 中找到.

原始答案

响应者的代码基于一个类和一个模块。 MimeResponds包含在 ActionController::Base 中, 你的类(class)ApplicationController继承自。然后是ActionController::Responder它提供了使用 respond_with 时的默认行为。

默认情况下,rails 在响应中提供的唯一行为是隐式尝试呈现具有与操作匹配的名称的模板。除此之外的任何事情都需要操作中的更多指令,或者使用 block 来处理多种格式响应的自定义 respond_to 调用。

由于大多数 Controller 使用相当常见的定制模式,响应者通过引入更多默认行为来提供额外的抽象级别。读取特定格式的调用 to_xml/to_json 的操作,以及提供相同格式的变更操作以及成功的变更操作的重定向。

有一些机会可以自定义响应者的行为方式,从细微的调整到完全覆盖或扩展行为。

类(class)等级:respond_to
您可以在此处指定响应程序应处理的格式。这些格式可以自定义它们将应用于哪些操作。每种格式都可以通过单独的调用来指定,从而可以完全自定义每种格式的操作。

# Responds to html and json on all actions
respond_to :html, :json

# Responds to html and json on index and show actions only.
respond_to :html, :json, :only => [:index,:show]

# Responds to html for everything except show, and json only for index, create and update
respond_to :html, :except => [:show]
respond_to :json, :only => [:index, :create, :update]

类(class)等级:responder
这是一个保存响应者的类属性。这可以是响应调用的任何东西,这意味着您可以使用 proc/lambda 或响应调用的类。另一种选择是将一个或多个模块混合到现有响应程序中以重载现有方法,从而增强默认行为。
class SomeController < ApplicationController
  respond_to :json

  self.responder = proc do |controller, resources, options|
    resource = resources.last
    request = controller.request
    if request.get?
      controller.render json: resource
    elsif request.post? or request.put?
      if resource.errors.any?
        render json: {:status => 'failed', :errors => resource.errors}
      else
        render json: {:status => 'created', :object => resource}
      end
    end
  end
end

虽然可能有一些有趣的边缘用例,但将模块扩展或混合到默认响应者中更有可能是更常见的模式。在任何情况下,相关的选项都是资源和选项,因为它们是从 from respond_with 传递过来的。

实例级别:respond_with
这里的选项是那些将被传递给你的 Controller 中的 render 或 redirect_to 的选项,但它们只包含在成功的场景中。对于 GET 操作,这些将是渲染调用,对于其他操作,这将是重定向的选项。其中最有用的可能是 :location选项,可用于覆盖该重定向路径,以防 respond_with 的参数不足以构建正确的 URL。
# These two are essentially equal
respond_with(:admin, @user, @post)
respond_with(@post, :location => admin_user_post(@user, @post)

# Respond with a 201 instead of a 200 HTTP status code, and also
# redirect to the collection path instead of the resource path
respond_with(@post, :status => :created, :location => posts_path)

# Note that if you want to pass a URL with a query string
# then the location option would be needed.
# /users?scope=active
respond_with(@user, :location => users_path(:scope => 'active'))

作为替代方案,responders gem 不仅提供了一些模块来覆盖一些默认行为。它用一个扩展默认响应者的匿名类覆盖默认响应者,并提供一个类级别的方法来混合自定义模块到这个类。这里最有用的是 flash 响应器,它提供了一组默认的 flash,将自定义委托(delegate)给 I18n 系统,config/locales/en.yml默认情况下。

我在以前的项目中使用的一些自定义响应器示例包括一个响应器,它自动修饰我的资源,并提供一组默认的页面标题和一个用于轻松自定义或覆盖页面标题的界面。

关于ruby-on-rails - Rails response_with : how does it work?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6118639/

相关文章:

ruby-on-rails - 不能对现有成员使用 upsert

ruby-on-rails - 我应该如何通过关联建立 has_one?

ruby-on-rails - Bundler:如何删除已安装的 gem

ruby-on-rails - 在Rails 3.1中的scss文件之间共享mixin

javascript - 具有 Primary_key 支持的 Rails 3 自动完成解决方案

ruby-on-rails - Rails better_errors gem 替代 gem

ruby-on-rails - Rails 中 jqGrid 数据的自定义响应程序

ruby-on-rails - 与 Omniauth 一起设计。无法验证,因为 "Invalid credentials"。”

ruby - 尽管未定义操作和路由,但 RSpec 成功

ruby-on-rails - 延迟作业重启 + Capistrano 最佳实践