理解路由的目的

看懂routes.rb文件中的代码

使用经典的hash风格或者现在比较流行的Restful风格构造你自己的路径

断定一个路径会映射到哪一个controller和action

路由的双重作用

Rails的路由是一个双重的机制 - 你既能把树木变成纸张,也能把纸张变成树木.更准确的说,它既能将进入服务器的HTTP请求连接到你的controller,也能帮助你(在View内)生成URL而不需要使用硬编码的字符串.

从URL到代码

当你的Rails应用接收到HTTP请求后,比如:

GET /patients/17

Rails的路由引擎就是把请求分发到你的应用中合适点的那些代码.具体到这个例子,应用程序比较可能会运行patients controller中的show action,并展示id是17的patient的详细信息.

从代码到URL

路由也可以反过来作用.如果你的应用包含了以下代码

@patient = Patient.find(17)

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

那么,路由引擎就会把这段代码解释成这样的URL: http://example.com/patients/17.这样使用路由,比起硬编码URL,可以降低你的应用程序的脆弱程度(增加程序健壮性),并且可以增加代码的可读性,使你的程序更容易被理解.

Patient需要被声明成为resource才可以使用这种形式的路径转换

快速浏览Routes.rb

Rails的路由有两个组陈部分:作为Rails一部分的路由引擎,还有包含有你的应用程序真正路径的config/routes.rb文件.讲授可以把什么东西放到routes.rb里面去是本文的目的,不过在此之前我们先概览一下.

处理文件

从格式上讲,routes.rb只不过是被传递给ActionController::Routing:routes.draw的一个大大的 block. 这个文件里也可以有注释,不过大多数情况下,应该是移行移行的代码,每一行代表你应用程序的一条路径.路径有五种主要类型:

RESTful路径 命名路径 嵌套路径 常规路径 默认路径

每一个类型的路径都会在随后详细阐述. 当有请求进入时, routes.rb文件自顶向下出路.请求会被分发到第一个匹配的路径.如果没有匹配的路径,Rails会返回给调用者一个HTTP 404.

RESTful路径

RESTful路径沾了Rails内建REST机制的光,只用一句单独的声明就包含了很多路由信息.一个RESTful路径看起来是这样的:

map.resources :books

命名路径

命名路径在处理请求的同时,也可以给你的代码提供非常具有可读性的链接.这是一个典型的命名路径:

map.login ‘/login’, :controller => ’sessions’, :action => ‘new’

嵌套路径

嵌套路径可以让你在一个resource里面包含另一个resource.你待会儿就会知道它是怎么被转化成URL(绝对路径 http://niuwa.org/index.html)和path(相对路径 /index.html)的.举例来说,如果你的应用包含很多部分,每个部分属于一个包(assembly),你也许可以这么写:

map.resources :assemblies do |assemblies| assemblies.resources :parts end

常规路径

在很多应用中,你会看到非RESTful的路由,它们显式的链接URL和action.例如:

map.connect ‘parts/:number’, :controller => ‘inventory’, :action => ’show’

默认路径

默认路径十一个安全网,抓住那些没有匹配上其他路径的请求.很多Rails应用都会包含这两条默认路径:

map.connect ‘:controller/:action/:id’ map.connect ‘:controller/:action/:id.:format’

这两个默认路径是你创建Rails应用的时候自动生成的.你可以为你的应用里面的所有东西使用RESTful的路径,那你就有可能想去掉这两条默认路径.在你去掉它们之前请确保你没有用到它们.

`

RESTful路由:Rails的默认方式

RESTful路由是目前Rails的标准路由方式,并且它也是你在新建的应用中首先使用的方式.要理解RESTful路有可能要花上一点时间,不 过这样 的努力是值得的;你的代码会变得更容易阅读,并且当你使用这种路由方式的时候,你会跟Rails配合的很好,而不是跟Rails对着干.

什么是REST?

RESTful 路由的基础是由Roy Fielding的博士论文提出的,参见: Architectural Styles and the Design of Network-based Software Architectures. 幸运的是,你不需要看懂整篇博士论文就可以理解Rails中的REST是怎么回事.REST是Representational State Transfer(基于表现的状态转移)的缩写.对我们来说,主要有两点:

使用资源定位符(对我们来说就是URL)来表示资源 在系统的不同组件之间转移表现状态

举例来说,对于Rails应用中这样的一个请求:

DELETE /phote/17

会被理解成是操作ID是17的photo,并且指示出了期望的操作 - 删除该资源.REST是web应用架构的一种自然的方式,并且Rails进一步使用约定(convention)隐藏了RESTful的某些复杂性使之更加自然(大概是指用起来更简单).

CRUD,动词和action

在RAils中,一个RESTful路径提供了HTTP动词(也就是GET, PUT等等),controller actions和(隐含的)CRUD数据库操作之间的的映射.路由文件中一个单独的入口如下所示:

map.resources :photos

在你的应用中创建了七个不同的路径:

GET /photos Photos index display a list of all photos
GET /photos/new Photos new return anHTMLform for creating a new photo
POST /photos Photos create create a new photo
GET /photos/1 Photos show display a specific photo
GET /photos/1/edit Photos edit return anHTMLform for editing a photo
PUT /photos/1 Photos update update a specific photo
DELETE /photos/1 Photos destroy delete a specific photo

对这些路径(由之前的resource定义生成的),resource的id可以在相应的controller action中使用params[:id]得到.

如果你一直在你的应用程序里面使用RESTful路径,你应该删除routes.rb中的默认路径,这样就会迫使Rails使用HTTP动词和路径间的映射.

URL和Path

创建RESTful路径会在你的应用程序里面生成一堆helper:

photos_url 和photos_path映射到index和create两个action new_photo_url和new_photo_path映射到new action edit_photo_url和edit_photo_path映射到edit action photo_url和photo_path映射到show, update和destroy三个action

因为路由同时使用HTTP动词和path二者分发请求,因此这七个RESTful路由生成的路径只有4对helper.

在这里,以_url结尾的helper生成应用程序能理解的整个URL(包含主机名),而以_path结尾的helper仅生成从应用程序根目录开始的path(不包含主机名).例如:

photos_url# => “http://www.example.com/photos”

photos_path# => “/photos”

同时定义多个资源

如果你需要为多个资源创建路径,你可以使用一次map.resources调用就定义它们:

map.resources :photos, :books, :videos

跟这样定义的效果是一样的:

map.resources :photos map.resources :books map.resources :videos

单数形式资源

你也可以应用RESTful路由定义单数形式的资源.这是,你需要使用map.resource代替刚才的map.resources,这时生成的路径会略有不同:

map.resource :geocoder

创建六个不同的路径:

GET /geocoder/new Geocoders new return anHTMLform for creating the new geocoder
POST /geocoder Geocoders create create the new geocoder
GET /geocoder Geocoders show display the one and only geocoder resource
GET /geocoder/edit Geocoders edit return anHTMLform for editing the geocoder
PUT /geocoder Geocoders update update the one and only geocoder resource
DELETE /geocoder Geocoders destroy delete the geocoder resource

虽然routes.rb中的资源名称是单数,但是对应的controller名称依然是复数.

一个单数形式的RESTful路径生成如下的helper:

new_geocoder_url 和new_geocoder_path映射到new action edit_geocoder_url和edit_geocoder_path映射到edit action geocoder_url和geocode_path映射到create, update和destroy三个action 自定义资源

虽然使用RESTful路由约定对很多应用来说基本就足够了,不过还是有一些其他的方法自定义RESTful路径的工作方式.这些选项包括:

:controller :singular :requirements :conditions :as :path_names :path_prefix :name_prefix :only :except

你也可以使用:member和:collection选项添加路径,这个在本文中稍后讨论.

使用:controller

:controller选项使你可以使用和公开的资源名称不同的controller名称,例如:

map.resources :photos, :controller => “images”

生成helper时会依据资源的名称,而不是controller的名称.因此,在这个例子里,就会得到photos_path,new_photo_path等等.

controller命名空间和路由


默认路由:

Ruby代码

# Rails3:  match '/:controller(/:action(/:id))'  # Rails2:  map.connect ':controller/:action/:id'

# Rails3: match '/:controller(/:action(/:id))' # Rails2: map.connect ':controller/:action/:id'

正则路由:

Ruby代码

# Rails3:  match 'products/:id', :to => 'catalog#view'  # Rails2:  map.connect 'products/:id', :controller => 'catalog', :action => 'view'

# Rails3: match 'products/:id', :to => 'catalog#view' # Rails2: map.connect 'products/:id', :controller => 'catalog', :action => 'view'

命名路由:

Ruby代码

# Rails3:  match 'logout', :to => 'sessions#destroy', :as => 'logout'  # Rails2:  map.logout 'logout', :controller => 'sessions', :action => ''

# Rails3: match 'logout', :to => 'sessions#destroy', :as => 'logout' # Rails2: map.logout 'logout', :controller => 'sessions', :action => ''

根路由:

Ruby代码

# Rails3:  root => 'welcome#show'  # Rails2:  map.root :controller => 'welcome', :action => 'show'

# Rails3: root => 'welcome#show' # Rails2: map.root :controller => 'welcome', :action => 'show'

路由简写技巧:
:to 键的省略:

Ruby代码

match 'account' => 'account#index'  # 相当于:  match 'account', :to => 'account#index'    match 'info' => 'projects#info', :as => 'info'

match 'account' => 'account#index' # 相当于: match 'account', :to => 'account#index' match 'info' => 'projects#info', :as => 'info'
注意: 
:as 在rails3中是改变 helper, 在rails2中是改变 path

当路径和控制器(及action)一至时,可省略指派控制器部分

Ruby代码

match 'account/overview'  # 相当于:  match 'account/overview', :to => 'account#overview'

match 'account/overview' # 相当于: match 'account/overview', :to => 'account#overview'

Verb路由
当需要限制http请求方法的时候通过键 :via ,也可以直接把方法写在最前面:

Ruby代码

get 'account/overview'  # 相当于:  match 'account/overview', :via => 'get'    match 'account/setup', :via => [:get, :post]  # 支持get\post\put\delete四种HTTP方法

get 'account/overview' # 相当于: match 'account/overview', :via => 'get' match 'account/setup', :via => [:get, :post] # 支持get\post\put\delete四种HTTP方法

resources路由:

Ruby代码

resources :posts, :except => [:index]  resources :posts, :only => [:new, :create]    # edit_post GET    /posts/:id/modify(.:format) {:controller=>"posts", :action=>"edit"}  resources :posts, :path_names => { :edit => 'modify' }     resources :projects do    resources :tasks, :people  end    resources :products do    collection do      get  :sold      post :on_offer, :search    end    get :buy, :on => :member    post :batch, :on => :collection  end    resource :session do    get :create  end

resources :posts, :except => [:index] resources :posts, :only => [:new, :create] # edit_post GET /posts/:id/modify(.:format) {:controller=>"posts", :action=>"edit"} resources :posts, :path_names => { :edit => 'modify' } resources :projects do resources :tasks, :people end resources :products do collection do get :sold post :on_offer, :search end get :buy, :on => :member post :batch, :on => :collection end resource :session do get :create end

:shallow用法:
Rails3中的shallow用法与Rails2中一致

Ruby代码

resources :blogs, :shallow => true do    resources :comments  end

resources :blogs, :shallow => true do resources :comments end 
使用:shallow前后相同部分:

blog_comments GET /blogs/:blog_id/comments(.:format) {:controller=>"comments", :action=>"index"}
blog_comments POST /blogs/:blog_id/comments(.:format) {:controller=>"comments", :action=>"create"}
new_blog_comment GET /blogs/:blog_id/comments/new(.:format) {:controller=>"comments", :action=>"new"}
blogs GET /blogs(.:format) {:controller=>"blogs", :action=>"index"}
blogs POST /blogs(.:format) {:controller=>"blogs", :action=>"create"}
new_blog GET /blogs/new(.:format) {:controller=>"blogs", :action=>"new"}
edit_blog GET /blogs/:id/edit(.:format) {:controller=>"blogs", :action=>"edit"}
blog GET /blogs/:id(.:format) {:controller=>"blogs", :action=>"show"}
blog PUT /blogs/:id(.:format) {:controller=>"blogs", :action=>"update"}
blog DELETE /blogs/:id(.:format) {:controller=>"blogs", :action=>"destroy"}

使用:shallow前后不同部分:
不使用shallow选项:

edit_blog_comment GET /blogs/:blog_id/comments/:id/edit(.:format) {:controller=>"comments", :action=>"edit"}
blog_comment GET /blogs/:blog_id/comments/:id(.:format) {:controller=>"comments", :action=>"show"}
blog_comment PUT /blogs/:blog_id/comments/:id(.:format) {:controller=>"comments", :action=>"update"}
blog_comment DELETE /blogs/:blog_id/comments/:id(.:format) {:controller=>"comments", :action=>"destroy"}

使用shallow选项后:

edit_comment GET /comments/:id/edit(.:format) {:controller=>"comments", :action=>"edit"}
comment GET /comments/:id(.:format) {:controller=>"comments", :action=>"show"}
comment PUT /comments/:id(.:format) {:controller=>"comments", :action=>"update"}
comment DELETE /comments/:id(.:format) {:controller=>"comments", :action=>"destroy"}

可以看出使用shallow选项后,对于已经存在的资源使用简化方式操作,具体行为涉及到 edit\show\update\destroy 四种 
另外,shallow选项的有效范围是对自身及嵌套的资源都有效,如下面这个例子:

Ruby代码

resources :publishers do    resources :magazines do      resources :albums, :shallow => true do        resources :photos do          resources :images        end      end    end  end

resources :publishers do resources :magazines do resources :albums, :shallow => true do resources :photos do resources :images end end end end 
这个例子中 albums、photos、images 都会使用简化方式,而 magazines 不会。特别注意:这种嵌套方式极不推荐,一般嵌套的层级最好不要超过一级

scope路由
:path 改变Path,:module 改变Controller, :name_prefix || :as 改变  helper

Ruby代码

scope 'admin' do    resources :posts  end  # 行当于:  scope :path => 'admin' do    resources :posts  end

scope 'admin' do resources :posts end # 行当于: scope :path => 'admin' do resources :posts end

Ruby代码

scope :module => 'admin' do    resources :posts  end  # 相当于:  resources :posts, :module => 'admin'  

scope :module => 'admin' do resources :posts end # 相当于: resources :posts, :module => 'admin'

Ruby代码

scope :name_prefix => 'admin' do    resources :posts  end  # 相当于:  resources :posts, :name_prefix => 'admin'

scope :name_prefix => 'admin' do resources :posts end # 相当于: resources :posts, :name_prefix => 'admin'

Ruby代码

scope 'admin', :module => 'admin', :name_prefix => 'admin' do    resources :posts  end  # 相当于:  namespace 'admin' do     resources :posts  end

scope 'admin', :module => 'admin', :name_prefix => 'admin' do resources :posts end # 相当于: namespace 'admin' do resources :posts end

在路由中定义跳转:

Ruby代码

match "/posts/github" => redirect("http://github.com/rails.atom")    # 地址 /foo/1 会自动跳转到 /bar/1s  match "/foo/:id", :to => redirect("/bar/%{id}s")      # /account/proc/inosin 会自动跳转到 /inosins  match 'account/proc/:name', :to => redirect {|params|   "/#{params[:name].pluralize}" }    match "/stories" => redirect {|p, req| "/posts/#{req.subdomain}" }

match "/posts/github" => redirect("http://github.com/rails.atom") # 地址 /foo/1 会自动跳转到 /bar/1s match "/foo/:id", :to => redirect("/bar/%{id}s") # /account/proc/inosin 会自动跳转到 /inosins match 'account/proc/:name', :to => redirect {|params| "/#{params[:name].pluralize}" } match "/stories" => redirect {|p, req| "/posts/#{req.subdomain}" }

路由中的限制:

Ruby代码

# 限制 id 只能为数字  match "/posts/show/:id", :to => "posts#index", :id => /\d+/  match "/posts/show/:id", :to => "posts#index", :constraints => {:id => /\d+/}    # 限制子域名  match "photos", :constraints => {:subdomain => "admin"}     # 限制访问者 IP  constraints(:ip => /127.0.0.1/) do    match  '/questions', :to => redirect("http://www.stackoverflow.com/")  end    # 当访问者 ip 是 192.168.1.* 的来访者访问 子域名为 "test"  match "/ttt" => proc{|env| [200, {}, ["hello test"]]}, \      :constraints => {:subdomain => "test", :ip => /192\.168\.1\.\d+/}

# 限制 id 只能为数字 match "/posts/show/:id", :to => "posts#index", :id => /\d+/ match "/posts/show/:id", :to => "posts#index", :constraints => {:id => /\d+/} # 限制子域名 match "photos", :constraints => {:subdomain => "admin"} # 限制访问者 IP constraints(:ip => /127.0.0.1/) do match '/questions', :to => redirect("http://www.stackoverflow.com/") end # 当访问者 ip 是 192.168.1.* 的来访者访问 子域名为 "test" match "/ttt" => proc{|env| [200, {}, ["hello test"]]}, \ :constraints => {:subdomain => "test", :ip => /192\.168\.1\.\d+/}

路由通配符:

Ruby代码

resources :photos, :id => /\d+/  match 'photos/*other' => 'photos#unknown'  #上面这两行路由则会把不符合7种path的其他url全部解析到PhotoController#unknown中去处理,params[:other]可得到path中/photos/之后的部分,注意这两行的顺序不能颠倒    match 'books/*section/:title' => 'books#show'   # 例如:books/some/section/last-words-a-memoir 中 params[:section] = "some/section", params[:title] = "last-words-a-memoir".    match '*a/foo/*b' => 'test#index'   # 例如:zoo/woo/foo/bar/baz 中 params[:a] = "zoo/woo", params[:b] = "bar/baz"

resources :photos, :id => /\d+/ match 'photos/*other' => 'photos#unknown' #上面这两行路由则会把不符合7种path的其他url全部解析到PhotoController#unknown中去处理,params[:other]可得到path中/photos/之后的部分,注意这两行的顺序不能颠倒 match 'books/*section/:title' => 'books#show' # 例如:books/some/section/last-words-a-memoir 中 params[:section] = "some/section", params[:title] = "last-words-a-memoir". match '*a/foo/*b' => 'test#index' # 例如:zoo/woo/foo/bar/baz 中 params[:a] = "zoo/woo", params[:b] = "bar/baz"

Rack:

Ruby代码

match "/foo", :to => proc {|env| [200, {}, ["Hello world"]] }     match 'rocketeer.js' => ::TestRoutingMapper::RocketeerApp     RocketeerApp = lambda { |env|    [200, {"Content-Type" => "text/html"}, ["javascripts"]]  }

match "/foo", :to => proc {|env| [200, {}, ["Hello world"]] } match 'rocketeer.js' => ::TestRoutingMapper::RocketeerApp RocketeerApp = lambda { |env| [200, {"Content-Type" => "text/html"}, ["javascripts"]] }

Rails ---> routes.rb 详解的更多相关文章

  1. redis cluster管理工具redis-trib.rb详解

    redis cluster管理工具redis-trib.rb详解 来源 http://weizijun.cn/2016/01/08/redis%20cluster%E7%AE%A1%E7%90%86% ...

  2. Flutter学习笔记(15)--MaterialApp应用组件及routes路由详解

    如需转载,请注明出处:Flutter学习笔记(15)--MaterialApp应用组件及routes路由详解 最近一段时间生病了,整天往医院跑,也没状态学东西了,现在是好了不少了,也该继续学习啦!!! ...

  3. redis-trib.rb命令详解

    redis-trib.rb是官方提供的Redis Cluster的管理工具,无需额外下载,默认位于源码包的src目录下,但因该工具是用ruby开发的,所以需要准备相关的依赖环境. 准备redis-tr ...

  4. redmine安装详解

    1.Linux:centos6.4(32位)2.Gcc的编译环境.使用make命令编辑.yum install gcc-c++ 3.PCRE PCRE(Perl Compatible Regular ...

  5. WordPress代码高亮插件SyntaxHighlighter终极使用详解

    子曰: 工欲善其事,必先利其器.作为码农一枚,再加上站长这个已经不再光鲜的称呼,岂能没有一款经济实用.操作简单.而且功能必须强大.样式也必须好看的Wordpress代码高亮插件?!作为一个视代码如生命 ...

  6. redis cluster 集群 安装 配置 详解

    redis cluster 集群 安装 配置 详解 张映 发表于 2015-05-01 分类目录: nosql 标签:cluster, redis, 安装, 配置, 集群 Redis 集群是一个提供在 ...

  7. [转]keil使用详解

    第一节 系统概述 Keil C51是美国Keil Software公司出品的51系列兼容单片机C语言软件开发系统,与汇编相比,C语言在功能上.结构性.可读性.可维护性上有明显的优势,因而易学易用.用过 ...

  8. Git 常用命令详解

    Git 是一个很强大的分布式版本管理工具,它不但适用于管理大型开源软件的源代码(如:linux kernel),管理私人的文档和源代码也有很多优势(如:wsi-lgame-pro) Git 的更多介绍 ...

  9. routes.rb和link_to的一些规则

    rails文档中描述了一个知识,link_to方法用于产生链接,但链接是根据routes.rb中的路由规则来产生的.这又分为面向资源和非面向资源两种产生链接的方法.比如 routes.rb文件中有两条 ...

随机推荐

  1. linux命令:chgrp

    1.命令介绍: chgrp用来改变文件或目录所属组的权限,要改变成的组必须在/etc/group文件存在才可以. 2.命令格式: chgrp [选项] 组 文件 3.命令参数: 必要参数: -c 当发 ...

  2. matlab中的数据结构

    一.cell 1. function: num2cell(A,n) n表示如何把A中的数据转换为cell. n=1表示把每列的所有行转换为cell:n=2表示把每行的所有列转换为cell. >& ...

  3. Less 语法特性

                                     ——(原创翻译:译者添加部分解释和代码运行结果) ♥在线Less编译器:LESS2CSS <一>综述 Less作为CSS的 ...

  4. cnblogs开篇留念

    之前看过很多大牛程序员们介绍的一些经验之类的文章,几乎每个人都提到了一点就是平时要写博客,记录一些自己平时学习和工作过程中学习到的一些技术点和心得.之前也用过一些其他的网站博客,上周有同事推荐了一篇文 ...

  5. 贝叶斯网引论 by 张连文

    贝叶斯网(Bayesian networks)是一种描述随机变量之间关系的语言,构造贝叶斯网是为了概率推理,理论上概率推理基于联合概率分布就行了,但是联合概率分布(基于表)的复杂度会呈指数增长,贝叶斯 ...

  6. javaEE-----org.springframework.dao.InvalidDataAccessApiUsageException: Write operation

    org.springframework.dao.InvalidDataAccessApiUsageException: Write operations are not allowed in read ...

  7. ionic中的service简单写法

    在service中写服务 服务名叫feedService .service('feedService',function($ionicLoading,$q,$http){ return{ //获取反馈 ...

  8. 前台给后台传JSON字符串,后台解析并保存

    前台 function BMSure() { var DanWeiName = $("[id$='BusinessName']").val(); var Address = $(& ...

  9. 特别实用而且功能强大的attributedText属性

    UILabel *sendNameLB = [[UILabel alloc]initWithFrame:CGRectMake(, , , )]; NSString * string = @" ...

  10. odoo-10.0 create database 失败

    在初始化数据库的界面点击[create database] create database 失败 报错如下 2017-01-05 20:15:18,529 4652 INFO ? werkzeug: ...