Rails 路由全解

优质
小牛编辑
143浏览
2023-12-01

Rails 路由全解

本文介绍面向用户的 Rails 路由功能。

读完本文,你将学到:

  • 如何理解 routes.rb 文件中的代码;
  • 如何使用推荐的资源式,或使用 match 方法编写路由;
  • 动作能接收到什么参数;
  • 如何使用路由帮助方法自动创建路径和 URL;
  • 约束和 Rack 端点等高级技术;

1 Rails 路由的作用

Rails 路由能识别 URL,将其分发给控制器的动作进行处理,还能生成路径和 URL,无需直接在视图中硬编码字符串。

1.1 把 URL 和代码连接起来

Rails 程序收到如下请求时

GET /patients/17

会查询路由,找到匹配的控制器动作。如果首个匹配的路由是:

get '/patients/:id', to: 'patients#show'

那么这个请求就交给 patients 控制器的 show 动作处理,并把 { id: '17' } 传入 params

1.2 生成路径和 URL

通过路由还可生成路径和 URL。如果把前面的路由修改成:

get '/patients/:id', to: 'patients#show', as: 'patient'

在控制器中有如下代码:

@patient = Patient.find(17)

在相应的视图中有如下代码:

<%= link_to 'Patient Record', patient_path(@patient) %>

那么路由就会生成路径 /patients/17。这么做代码易于维护、理解。注意,在路由帮助方法中无需指定 ID。

2 资源路径:Rails 的默认值

使用资源路径可以快速声明资源式控制器所有的常规路由,无需分别为 indexshowneweditcreateupdatedestroy 动作分别声明路由,只需一行代码就能搞定。

2.1 网络中的资源

浏览器向 Rails 程序请求页面时会使用特定的 HTTP 方法,例如 GETPOSTPATCHPUTDELETE。每个方法对应对资源的一种操作。资源路由会把一系列相关请求映射到单个路由器的不同动作上。

如果 Rails 程序收到如下请求:

DELETE /photos/17

会查询路由将其映射到一个控制器的路由上。如果首个匹配的路由是:

resources :photos

那么这个请求就交给 photos 控制器的 destroy 方法处理,并把 { id: '17' } 传入 params

2.2 CRUD,HTTP 方法和动作

在 Rails 中,资源式路由把 HTTP 方法和 URL 映射到控制器的动作上。而且根据约定,还映射到数据库的 CRUD 操作上。路由文件中如下的单行声明:

resources :photos

会创建七个不同的路由,全部映射到 Photos 控制器上:

HTTP 方法路径控制器#动作作用
GET/photosphotos#index显示所有图片
GET/photos/newphotos#new显示新建图片的表单
POST/photosphotos#create新建图片
GET/photos/:idphotos#show显示指定的图片
GET/photos/:id/editphotos#edit显示编辑图片的表单
PATCH/PUT/photos/:idphotos#update更新指定的图片
DELETE/photos/:idphotos#destroy删除指定的图片

路由使用 HTTP 方法和 URL 匹配请求,把四个 URL 映射到七个不同的动作上。 I> NOTE: 路由按照声明的顺序匹配哦,如果在 get 'photos/poll' 之前声明了 resources :photos,那么 show 动作的路由由 resources 这行解析。如果想使用 get 这行,就要将其移到 resources 之前。

2.3 路径和 URL 帮助方法

声明资源式路由后,会自动创建一些帮助方法。以 resources :photos 为例:

  • photos_path 返回 /photos
  • new_photo_path 返回 /photos/new
  • edit_photo_path(:id) 返回 /photos/:id/edit,例如 edit_photo_path(10) 返回 /photos/10/edit
  • photo_path(:id) 返回 /photos/:id,例如 photo_path(10) 返回 /photos/10

这些帮助方法都有对应的 _url 形式,例如 photos_url,返回主机、端口加路径。

2.4 一次声明多个资源路由

如果需要为多个资源声明路由,可以节省一点时间,调用一次 resources 方法完成:

resources :photos, :books, :videos

这种方式等价于:

resources :photos
resources :books
resources :videos

2.5 单数资源

有时希望不用 ID 就能查看资源,例如,/profile 一直显示当前登入用户的个人信息。针对这种需求,可以使用单数资源,把 /profile(不是 /profile/:id)映射到 show 动作:

get 'profile', to: 'users#show'

如果 get 方法的 to 选项是字符串,要使用 controller#action 形式;如果是 Symbol,就可以直接指定动作:

get 'profile', to: :show

下面这个资源式路由:

resource :geocoder

会生成六个路由,全部映射到 Geocoders 控制器:

HTTP 方法路径控制器#动作作用
GET/geocoder/newgeocoders#new显示新建 geocoder 的表单
POST/geocodergeocoders#create新建 geocoder
GET/geocodergeocoders#show显示唯一的 geocoder 资源
GET/geocoder/editgeocoders#edit显示编辑 geocoder 的表单
PATCH/PUT/geocodergeocoders#update更新唯一的 geocoder 资源
DELETE/geocodergeocoders#destroy删除 geocoder 资源

有时需要使用同个控制器处理单数路由(例如 /account)和复数路由(例如 /accounts/45),把单数资源映射到复数控制器上。例如,resource :photoresources :photos 分别声明单数和复数路由,映射到同个控制器(PhotosController)上。

单数资源式路由生成以下帮助方法:

  • new_geocoder_path 返回 /geocoder/new
  • edit_geocoder_path 返回 /geocoder/edit
  • geocoder_path 返回 /geocoder

和复数资源一样,上面各帮助方法都有对应的 _url 形式,返回主机、端口加路径。

有个一直存在的问题导致 form_for 无法自动处理单数资源。为了解决这个问题,可以直接指定表单的 URL,例如:

form_for @geocoder, url: geocoder_path do |f|

2.6 控制器命名空间和路由

你可能想把一系列控制器放在一个命名空间内,最常见的是把管理相关的控制器放在 Admin:: 命名空间内。你需要把这些控制器存在 app/controllers/admin 文件夹中,然后在路由中做如下声明:

namespace :admin do
  resources :articles, :comments
end

上述代码会为 articlescomments 控制器生成很多路由。对 Admin::ArticlesController 来说,Rails 会生成:

HTTP 方法路径控制器#动作具名帮助方法
GET/admin/articlesadmin/articles#indexadmin_articles_path
GET/admin/articles/newadmin/articles#newnew_admin_article_path
POST/admin/articlesadmin/articles#createadmin_articles_path
GET/admin/articles/:idadmin/articles#showadmin_article_path(:id)
GET/admin/articles/:id/editadmin/articles#editedit_admin_article_path(:id)
PATCH/PUT/admin/articles/:idadmin/articles#updateadmin_article_path(:id)
DELETE/admin/articles/:idadmin/articles#destroyadmin_article_path(:id)

如果想把 /articles(前面没有 /admin)映射到 Admin::ArticlesController 控制器上,可以这么声明:

scope module: 'admin' do
  resources :articles, :comments
end

如果只有一个资源,还可以这么声明:

resources :articles, module: 'admin'

如果想把 /admin/articles 映射到 ArticlesController 控制器(不在 Admin:: 命名空间内),可以这么声明:

scope '/admin' do
  resources :articles, :comments
end

如果只有一个资源,还可以这么声明:

resources :articles, path: '/admin/articles'

在上述两种用法中,具名路由没有变化,跟不用 scope 时一样。在后一种用法中,映射到 ArticlesController 控制器上的路径如下:

HTTP 方法路径控制器#动作具名帮助方法
GET/admin/articlesarticles#indexarticles_path
GET/admin/articles/newarticles#newnew_article_path
POST/admin/articlesarticles#createarticles_path
GET/admin/articles/:idarticles#showarticle_path(:id)
GET/admin/articles/:id/editarticles#editedit_article_path(:id)
PATCH/PUT/admin/articles/:idarticles#updatearticle_path(:id)
DELETE/admin/articles/:idarticles#destroyarticle_path(:id)

如果在 namespace 代码块中想使用其他的控制器命名空间,可以指定控制器的绝对路径,例如 get '/foo' => '/foo#index'

2.7 嵌套资源

开发程序时经常会遇到一个资源是其他资源的子资源这种情况。假设程序中有如下的模型:

class Magazine < ActiveRecord::Base
  has_many :ads
end

class Ad < ActiveRecord::Base
  belongs_to :magazine
end

在路由中可以使用“嵌套路由”反应这种关系。针对这个例子,可以声明如下路由:

resources :magazines do
  resources :ads
end

除了创建 MagazinesController 的路由之外,上述声明还会创建 AdsController 的路由。广告的 URL 要用到杂志资源:

HTTP 方法路径控制器#动作作用
GET/magazines/:magazine_id/adsads#index显示指定杂志的所有广告
GET/magazines/:magazine_id/ads/newads#new显示新建广告的表单,该告属于指定的杂志
POST/magazines/:magazine_id/adsads#create创建属于指定杂志的广告
GET/magazines/:magazine_id/ads/:idads#show显示属于指定杂志的指定广告
GET/magazines/:magazine_id/ads/:id/editads#edit显示编辑广告的表单,该广告属于指定的杂志
PATCH/PUT/magazines/:magazine_id/ads/:idads#update更新属于指定杂志的指定广告
DELETE/magazines/:magazine_id/ads/:idads#destroy删除属于指定杂志的指定广告

上述路由还会生成 magazine_ads_urledit_magazine_ad_path 等路由帮助方法。这些帮助方法的第一个参数是 Magazine 实例,例如 magazine_ads_url(@magazine)

2.7.1 嵌套限制

嵌套路由可以放在其他嵌套路由中,例如:

resources :publishers do
  resources :magazines do
    resources :photos
  end
end

层级较多的嵌套路由很难处理。例如,程序可能要识别如下的路径:

/publishers/1/magazines/2/photos/3

对应的路由帮助方法是 publisher_magazine_photo_url,要指定三个层级的对象。这种用法很让人困扰,Jamis Buck 在一篇文章中指出了嵌套路由的用法总则,即:

嵌套资源不可超过一层。

2.7.2 浅层嵌套

避免深层嵌套的方法之一,是把控制器集合动作放在父级资源中,表明层级关系,但不嵌套成员动作。也就是说,用最少的信息表明资源的路由关系,如下所示:

resources :articles do
  resources :comments, only: [:index, :new, :create]
end
resources :comments, only: [:show, :edit, :update, :destroy]

这种做法在描述路由和深层嵌套之间做了适当的平衡。上述代码还有简写形式,即使用 :shallow 选项:

resources :articles do
  resources :comments, shallow: true
end

这种形式生成的路由和前面一样。:shallow 选项还可以在父级资源中使用,此时所有嵌套其中的资源都是浅层嵌套:

resources :articles, shallow: true do
  resources :comments
  resources :quotes
  resources :drafts
end

shallow 方法可以创建一个作用域,其中所有嵌套都是浅层嵌套。如下代码生成的路由和前面一样:

shallow do
  resources :articles do
    resources :comments
    resources :quotes
    resources :drafts
  end
end

scope 方法有两个选项可以定制浅层嵌套路由。:shallow_path 选项在成员路径前加上指定的前缀:

scope shallow_path: "sekret" do
  resources :articles do
    resources :comments, shallow: true
  end
end

上述代码为 comments 资源生成的路由如下:

HTTP 方法路径控制器#动作具名帮助方法
GET/articles/:article_id/comments(.:format)comments#indexarticle_comments_path
POST/articles/:article_id/comments(.:format)comments#createarticle_comments_path
GET/articles/:article_id/comments/new(.:format)comments#newnew_article_comment_path
GET/sekret/comments/:id/edit(.:format)comments#editedit_comment_path
GET/sekret/comments/:id(.:format)comments#showcomment_path
PATCH/PUT/sekret/comments/:id(.:format)comments#updatecomment_path
DELETE/sekret/comments/:id(.:format)comments#destroycomment_path

:shallow_prefix 选项在具名帮助方法前加上指定的前缀:

scope shallow_prefix: "sekret" do
  resources :articles do
    resources :comments, shallow: true
  end
end

上述代码为 comments 资源生成的路由如下:

HTTP 方法路径控制器#动作具名帮助方法
GET/articles/:article_id/comments(.:format)comments#indexarticle_comments_path
POST/articles/:article_id/comments(.:format)comments#createarticle_comments_path
GET/articles/:article_id/comments/new(.:format)comments#newnew_article_comment_path
GET/comments/:id/edit(.:format)comments#editedit_sekret_comment_path
GET/comments/:id(.:format)comments#showsekret_comment_path
PATCH/PUT/comments/:id(.:format)comments#updatesekret_comment_path
DELETE/comments/:id(.:format)comments#destroysekret_comment_path

2.8 Routing Concerns

Routing Concerns 用来声明通用路由,可在其他资源和路由中重复使用。定义 concern 的方式如下:

concern :commentable do
  resources :comments
end

concern :image_attachable do
  resources :images, only: :index
end

Concerns 可在资源中重复使用,避免代码重复:

resources :messages, concerns: :commentable

resources :articles, concerns: [:commentable, :image_attachable]

上述声明等价于:

resources :messages do
  resources :comments
end

resources :articles do
  resources :comments
  resources :images, only: :index
end

Concerns 在路由的任何地方都能使用,例如,在作用域或命名空间中:

namespace :articles do
  concerns :commentable
end

2.9 由对象创建路径和 URL

除了使用路由帮助方法之外,Rails 还能从参数数组中创建路径和 URL。例如,假设有如下路由:

resources :magazines do
  resources :ads
end

使用 magazine_ad_path 时,可以不传入数字 ID,传入 MagazineAd 实例即可:

<%= link_to 'Ad details', magazine_ad_path(@magazine, @ad) %>

而且还可使用 url_for 方法,指定一组对象,Rails 会自动决定使用哪个路由:

<%= link_to 'Ad details', url_for([@magazine, @ad]) %>

此时,Rails 知道 @magazineMagazine 的实例,@adAd 的实例,所以会调用 magazine_ad_path 帮助方法。使用 link_to 等方法时,无需使用完整的 url_for 方法,直接指定对象即可:

<%= link_to 'Ad details', [@magazine, @ad] %>

如果想链接到一本杂志,可以这么做:

<%= link_to 'Magazine details', @magazine %>

要想链接到其他动作,把数组的第一个元素设为所需动作名即可:

<%= link_to 'Edit Ad', [:edit, @magazine, @ad] %>

在这种用法中,会把模型实例转换成对应的 URL,这是资源式路由带来的主要好处之一。

2.10 添加更多的 REST 架构动作

可用的路由并不局限于 REST 路由默认创建的那七个,还可以添加额外的集合路由或成员路由。

2.10.1 添加成员路由

要添加成员路由,在 resource 代码块中使用 member 块即可:

resources :photos do
  member do
    get 'preview'
  end
end

这段路由能识别 /photos/1/preview 是个 GET 请求,映射到 PhotosControllerpreview 动作上,资源的 ID 传入 params[:id]。同时还生成了 preview_photo_urlpreview_photo_path 两个帮助方法。

member 代码块中,每个路由都要指定使用的 HTTP 方法。可以使用 getpatchputpostdelete。如果成员路由不多,可以不使用代码块形式,直接在路由上使用 :on 选项:

resources :photos do
  get 'preview', on: :member
end

也可以不使用 :on 选项,得到的成员路由是相同的,但资源 ID 存储在 params[:photo_id] 而不是 params[:id] 中。

2.10.2 添加集合路由

添加集合路由的方式如下:

resources :photos do
  collection do
    get 'search'
  end
end

这段路由能识别 /photos/search 是个 GET 请求,映射到 PhotosControllersearch 动作上。同时还会生成 search_photos_urlsearch_photos_path 两个帮助方法。

和成员路由一样,也可使用 :on 选项:

resources :photos do
  get 'search', on: :collection
end
2.10.3 添加额外新建动作的路由

要添加额外的新建动作,可以使用 :on 选项:

resources :comments do
  get 'preview', on: :new
end

这段代码能识别 /comments/new/preview 是个 GET 请求,映射到 CommentsControllerpreview 动作上。同时还会生成 preview_new_comment_urlpreview_new_comment_path 两个路由帮助方法。

如果在资源式路由中添加了过多额外动作,这时就要停下来问自己,是不是要新建一个资源。

3 非资源式路由

除了资源路由之外,Rails 还提供了强大功能,把任意 URL 映射到动作上。此时,不会得到资源式路由自动生成的一系列路由,而是分别声明各个路由。

虽然一般情况下要使用资源式路由,但也有一些情况使用简单的路由更合适。如果不合适,也不用非得使用资源实现程序的每种功能。

简单的路由特别适合把传统的 URL 映射到 Rails 动作上。

3.1 绑定参数

声明常规路由时,可以提供一系列 Symbol,做为 HTTP 请求的一部分,传入 Rails 程序。其中两个 Symbol 有特殊作用::controller 映射程序的控制器名,:action 映射控制器中的动作名。例如,有下面的路由:

get ':controller(/:action(/:id))'

如果 /photos/show/1 由这个路由处理(没匹配路由文件中其他路由声明),会映射到 PhotosControllershow 动作上,最后一个参数 "1" 可通过 params[:id] 获取。上述路由还能处理 /photos 请求,映射到 PhotosController#index,因为 :action:id 放在括号中,是可选参数。

3.2 动态路径片段

在常规路由中可以使用任意数量的动态片段。:controller:action 之外的参数都会存入 params 传给动作。如果有下面的路由:

get ':controller/:action/:id/:user_id'

/photos/show/1/2 请求会映射到 PhotosControllershow 动作。params[:id] 的值是 "1"params[:user_id] 的值是 "2"

匹配控制器时不能使用 :namespace:module。如果需要这种功能,可以为控制器做个约束,匹配所需的命名空间。例如: I> I> I>ruby NOTE: get ':controller(/:action(/:id))', controller: /admin\/[^\/]+/ NOTE:

默认情况下,动态路径片段中不能使用点号,因为点号是格式化路由的分隔符。如果需要在动态路径片段中使用点号,可以添加一个约束条件。例如,id: /[^\/]+/ 可以接受除斜线之外的所有字符。

3.3 静态路径片段

声明路由时可以指定静态路径片段,片段前不加冒号即可:

get ':controller/:action/:id/with_user/:user_id'

这个路由能响应 /photos/show/1/with_user/2 这种路径。此时,params 的值为 { controller: 'photos', action: 'show', id: '1', user_id: '2' }

3.4 查询字符串

params 中还包含查询字符串中的所有参数。例如,有下面的路由:

get ':controller/:action/:id'

/photos/show/1?user_id=2 请求会映射到 Photos 控制器的 show 动作上。params 的值为 { controller: 'photos', action: 'show', id: '1', user_id: '2' }

3.5 定义默认值

在路由中无需特别使用 :controller:action,可以指定默认值:

get 'photos/:id', to: 'photos#show'

这样声明路由后,Rails 会把 /photos/12 映射到 PhotosControllershow 动作上。

路由中的其他部分也使用 :defaults 选项设置默认值。甚至可以为没有指定的动态路径片段设定默认值。例如:

get 'photos/:id', to: 'photos#show', defaults: { format: 'jpg' }

Rails 会把 photos/12 请求映射到 PhotosControllershow 动作上,把 params[:format] 的值设为 "jpg"

3.6 命名路由

使用 :as 选项可以为路由起个名字:

get 'exit', to: 'sessions#destroy', as: :logout

这段路由会生成 logout_pathlogout_url 这两个具名路由帮助方法。调用 logout_path 方法会返回 /exit

使用 :as 选项还能重设资源的路径方法,例如:

get ':username', to: 'users#show', as: :user

这段路由会定义一个名为 user_path 的方法,可在控制器、帮助方法和视图中使用。在 UsersControllershow 动作中,params[:username] 的值即用户的用户名。如果不想使用 :username 作为参数名,可在路由声明中修改。

3.7 HTTP 方法约束

一般情况下,应该使用 getpostputpatchdelete 方法限制路由可使用的 HTTP 方法。如果使用 match 方法,可以通过 :via 选项一次指定多个 HTTP 方法:

match 'photos', to: 'photos#show', via: [:get, :post]

如果某个路由想使用所有 HTTP 方法,可以使用 via: :all

match 'photos', to: 'photos#show', via: :all

同个路由即处理 GET 请求又处理 POST 请求有安全隐患。一般情况下,除非有特殊原因,切记不要允许在一个动作上使用所有 HTTP 方法。

3.8 路径片段约束

可使用 :constraints 选项限制动态路径片段的格式:

get 'photos/:id', to: 'photos#show', constraints: { id: /[A-Z]\d{5}/ }

这个路由能匹配 /photos/A12345,但不能匹配 /photos/893。上述路由还可简化成:

get 'photos/:id', to: 'photos#show', id: /[A-Z]\d{5}/

:constraints 选项中的正则表达式不能使用“锚记”。例如,下面的路由是错误的:

get '/:id', to: 'photos#show', constraints: {id: /^\d/}

之所以不能使用锚记,是因为所有正则表达式都从头开始匹配。

例如,有下面的路由。如果 to_param 方法得到的值以数字开头,例如 1-hello-world,就会把请求交给 articles 控制器处理;如果 to_param 方法得到的值不以数字开头,例如 david,就交给 users 控制器处理。

get '/:id', to: 'articles#show', constraints: { id: /\d.+/ }
get '/:username', to: 'users#show'

3.9 基于请求的约束

约束还可以根据任何一个返回值为字符串的 Request 方法设定。

基于请求的约束和路径片段约束的设定方式一样:

get 'photos', constraints: {subdomain: 'admin'}

约束还可使用代码块形式:

namespace :admin do
  constraints subdomain: 'admin' do
    resources :photos
  end
end

3.10 高级约束

如果约束很复杂,可以指定一个能响应 matches? 方法的对象。假设要用 BlacklistConstraint 过滤所有用户,可以这么做:

class BlacklistConstraint
  def initialize
    @ips = Blacklist.retrieve_ips
  end

  def matches?(request)
    @ips.include?(request.remote_ip)
  end
end

TwitterClone::Application.routes.draw do
  get '*path', to: 'blacklist#index',
    constraints: BlacklistConstraint.new
end

约束还可以在 lambda 中指定:

TwitterClone::Application.routes.draw do
  get '*path', to: 'blacklist#index',
    constraints: lambda { |request| Blacklist.retrieve_ips.include?(request.remote_ip) }
end

matches? 方法和 lambda 的参数都是 request 对象。

3.11 通配片段

路由中的通配符可以匹配其后的所有路径片段。例如:

get 'photos/*other', to: 'photos#unknown'

这个路由可以匹配 photos/12/photos/long/path/to/12params[:other] 的值为 "12""long/path/to/12"。以星号开头的路径片段叫做“通配片段”。

通配片段可以出现在路由的任何位置。例如:

get 'books/*section/:title', to: 'books#show'

这个路由可以匹配 books/some/section/last-words-a-memoirparams[:section] 的值为 'some/section'params[:title] 的值为 'last-words-a-memoir'

严格来说,路由中可以有多个通配片段。匹配器会根据直觉赋值各片段。例如:

get '*a/foo/*b', to: 'test#index'

这个路由可以匹配 zoo/woo/foo/bar/bazparams[:a] 的值为 'zoo/woo'params[:b] 的值为 'bar/baz'

如果请求 '/foo/bar.json',那么 params[:pages] 的值为 'foo/bar',请求类型为 JSON。如果想使用 Rails 3.0.x 中的表现,可以指定 format: false 选项,如下所示: I> I> I>ruby NOTE: get '*pages', to: 'pages#show', format: false NOTE: I> NOTE: 如果必须指定格式,可以指定 format: true 选项,如下所示: I> I> I>ruby NOTE: get '*pages', to: 'pages#show', format: true NOTE:

3.12 重定向

在路由中可以使用 redirect 帮助方法把一个路径重定向到另一个路径:

get '/stories', to: redirect('/articles')

重定向时还可使用匹配的动态路径片段:

get '/stories/:name', to: redirect('/articles/%{name}')

redirect 还可使用代码块形式,传入路径参数和 request 对象作为参数:

get '/stories/:name', to: redirect {|path_params, req| "/articles/#{path_params[:name].pluralize}" }
get '/stories', to: redirect {|path_params, req| "/articles/#{req.subdomain}" }

注意,redirect 实现的是 301 "Moved Permanently" 重定向,有些浏览器或代理服务器会缓存这种重定向,导致旧的页面不可用。

如果不指定主机(http://www.example.com),Rails 会从当前请求中获取。

3.13 映射到 Rack 程序

除了使用字符串,例如 'articles#index',把请求映射到 ArticlesControllerindex 动作上之外,还可使用 Rack 程序作为端点:

match '/application.js', to: Sprockets, via: :all

只要 Sprockets 能响应 call 方法,而且返回 [status, headers, body] 形式的结果,路由器就不知道这是个 Rack 程序还是动作。这里使用 via: :all 是正确的,因为我们想让 Rack 程序自行判断,处理所有 HTTP 方法。

其实 'articles#index' 的复杂形式是 ArticlesController.action(:index),得到的也是个合法的 Rack 程序。

3.14 使用 root

使用 root 方法可以指定怎么处理 '/' 请求:

root to: 'pages#main'
root 'pages#main' # shortcut for the above

root 路由应该放在文件的顶部,因为这是最常用的路由,应该先匹配。

root 路由只处理映射到动作上的 GET 请求。

在命名空间和作用域中也可使用 root。例如:

namespace :admin do
  root to: "admin#index"
end

root to: "home#index"

3.15 Unicode 字符路由

路由中可直接使用 Unicode 字符。例如:

get 'こんにちは', to: 'welcome#index'

4 定制资源式路由

虽然 resources :articles 默认生成的路由和帮助方法都满足大多数需求,但有时还是想做些定制。Rails 允许对资源式帮助方法做几乎任何形式的定制。

4.1 指定使用的控制器

:controller 选项用来指定资源使用的控制器。例如:

resources :photos, controller: 'images'

能识别以 /photos 开头的请求,但交给 Images 控制器处理:

HTTP 方法路径控制器#动作具名帮助方法
GET/photosimages#indexphotos_path
GET/photos/newimages#newnew_photo_path
POST/photosimages#createphotos_path
GET/photos/:idimages#showphoto_path(:id)
GET/photos/:id/editimages#editedit_photo_path(:id)
PATCH/PUT/photos/:idimages#updatephoto_path(:id)
DELETE/photos/:idimages#destroyphoto_path(:id)

要使用 photos_pathnew_photo_path 等生成该资源的路径。

命名空间中的控制器可通过目录形式指定。例如:

resources :user_permissions, controller: 'admin/user_permissions'

这个路由会交给 Admin::UserPermissions 控制器处理。

只支持目录形式。如果使用 Ruby 常量形式,例如 controller: 'Admin::UserPermissions',会导致路由报错。

4.2 指定约束

可以使用 :constraints选项指定 id 必须满足的格式。例如:

resources :photos, constraints: {id: /[A-Z][A-Z][0-9]+/}

这个路由声明限制参数 :id 必须匹配指定的正则表达式。因此,这个路由能匹配 /photos/RR27,不能匹配 /photos/1

使用代码块形式可以把约束应用到多个路由上:

constraints(id: /[A-Z][A-Z][0-9]+/) do
  resources :photos
  resources :accounts
end

当然了,在资源式路由中也能使用非资源式路由中的高级约束。

默认情况下,在 :id 参数中不能使用点号,因为点号是格式化路由的分隔符。如果需要在 :id 中使用点号,可以添加一个约束条件。例如,id: /[^\/]+/ 可以接受除斜线之外的所有字符。

4.3 改写具名帮助方法

:as 选项可以改写常规的具名路由帮助方法。例如:

resources :photos, as: 'images'

能识别以 /photos 开头的请求,交给 PhotosController 处理,但使用 :as 选项的值命名帮助方法:

HTTP 方法路径控制器#动作具名帮助方法
GET/photosphotos#indeximages_path
GET/photos/newphotos#newnew_image_path
POST/photosphotos#createimages_path
GET/photos/:idphotos#showimage_path(:id)
GET/photos/:id/editphotos#editedit_image_path(:id)
PATCH/PUT/photos/:idphotos#updateimage_path(:id)
DELETE/photos/:idphotos#destroyimage_path(:id)

4.4 改写 newedit 片段

:path_names 选项可以改写路径中自动生成的 "new""edit" 片段:

resources :photos, path_names: { new: 'make', edit: 'change' }

这样设置后,路由就能识别如下的路径:

/photos/make
/photos/1/change

这个选项并不能改变实际处理请求的动作名。上述两个路径还是交给 newedit 动作处理。

如果想按照这种方式修改所有路由,可以使用作用域。 T> T> T>ruby TIP: scope path_names: { new: 'make' } do TIP: # rest of your routes TIP: end TIP:

4.5 为具名路由帮助方法加上前缀

使用 :as 选项可在 Rails 为路由生成的路由帮助方法前加上前缀。这个选项可以避免作用域内外产生命名冲突。例如:

scope 'admin' do
  resources :photos, as: 'admin_photos'
end

resources :photos

这段路由会生成 admin_photos_pathnew_admin_photo_path 等帮助方法。

要想为多个路由添加前缀,可以在 scope 方法中设置 :as 选项:

scope 'admin', as: 'admin' do
  resources :photos, :accounts
end

resources :photos, :accounts

这段路由会生成 admin_photos_pathadmin_accounts_path 等帮助方法,分别映射到 /admin/photos/admin/accounts 上。

namespace 作用域会自动添加 :as 以及 :module:path 前缀。

路由帮助方法的前缀还可使用具名参数:

scope ':username' do
  resources :articles
end

这段路由能识别 /bob/articles/1 这种请求,在控制器、帮助方法和视图中可使用 params[:username] 获取 username 的值。

4.6 限制生成的路由

默认情况下,Rails 会为每个 REST 路由生成七个默认动作(indexshownewcreateeditupdatedestroy)对应的路由。你可以使用 :only:except 选项调整这种行为。:only 选项告知 Rails,只生成指定的路由:

resources :photos, only: [:index, :show]

此时,向 /photos 能发起 GET 请求,但不能发起 POST 请求(正常情况下由 create 动作处理)。

:except 选项指定不用生成的路由:

resources :photos, except: :destroy

此时,Rails 会生成除 destroy(向 /photos/:id 发起的 DELETE 请求)之外的所有常规路由。

如果程序中有很多 REST 路由,使用 :only:except 指定只生成所需的路由,可以节省内存,加速路由处理过程。

4.7 翻译路径

使用 scope 时,可以改写资源生成的路径名:

scope(path_names: { new: 'neu', edit: 'bearbeiten' }) do
  resources :categories, path: 'kategorien'
end

Rails 为 CategoriesController 生成的路由如下:

HTTP 方法路径控制器#动作具名帮助方法
GET/kategoriencategories#indexcategories_path
GET/kategorien/neucategories#newnew_category_path
POST/kategoriencategories#createcategories_path
GET/kategorien/:idcategories#showcategory_path(:id)
GET/kategorien/:id/bearbeitencategories#editedit_category_path(:id)
PATCH/PUT/kategorien/:idcategories#updatecategory_path(:id)
DELETE/kategorien/:idcategories#destroycategory_path(:id)

4.8 改写单数形式

如果想定义资源的单数形式,需要在 Inflector 中添加额外的规则:

ActiveSupport::Inflector.inflections do |inflect|
  inflect.irregular 'tooth', 'teeth'
end

4.9 在嵌套资源中使用 :as 选项

:as 选项可以改自动生成的嵌套路由帮助方法名。例如:

resources :magazines do
  resources :ads, as: 'periodical_ads'
end

这段路由会生成 magazine_periodical_ads_urledit_magazine_periodical_ad_path 等帮助方法。

5 路由审查和测试

Rails 提供有路由审查和测试功能。

5.1 列出现有路由

要想查看程序完整的路由列表,可以在开发环境中使用浏览器打开 http://localhost:3000/rails/info/routes。也可以在终端执行 rake routes 任务查看,结果是一样的。

这两种方法都能列出所有路由,和在 routes.rb 中的定义顺序一致。你会看到每个路由的以下信息:

  • 路由名(如果有的话)
  • 使用的 HTTP 方法(如果不响应所有方法)
  • 匹配的 URL 模式
  • 路由的参数

例如,下面是执行 rake routes 命令后看到的一个 REST 路由片段:

    users GET    /users(.:format)          users#index
          POST   /users(.:format)          users#create
 new_user GET    /users/new(.:format)      users#new
edit_user GET    /users/:id/edit(.:format) users#edit

可以使用环境变量 CONTROLLER 限制只显示映射到该控制器上的路由:

$ CONTROLLER=users rake routes

拉宽终端窗口直至没断行,这时看到的 rake routes 输出更完整。

5.2 测试路由

和程序的其他部分一样,路由也要测试。Rails 内建了三个断言,可以简化测试:

  • assert_generates
  • assert_recognizes
  • assert_routing
5.2.1 assert_generates 断言

assert_generates 检测提供的选项是否能生成默认路由或自定义路由。例如:

assert_generates '/photos/1', { controller: 'photos', action: 'show', id: '1' }
assert_generates '/about', controller: 'pages', action: 'about'
5.2.2 assert_recognizes 断言

assert_recognizesassert_generates 的反测试,检测提供的路径是否能陪识别并交由特定的控制器处理。例如:

assert_recognizes({ controller: 'photos', action: 'show', id: '1' }, '/photos/1')

可以使用 :method 参数指定使用的 HTTP 方法:

assert_recognizes({ controller: 'photos', action: 'create' }, { path: 'photos', method: :post })
5.2.3 assert_routing 断言

assert_routing 做双向测试:检测路径是否能生成选项,以及选项能否生成路径。因此,综合了 assert_generatesassert_recognizes 两个断言。

assert_routing({ path: 'photos', method: :post }, { controller: 'photos', action: 'create' })