URI

URI 是 Uniform Resource Identifier 的缩写。

Uniform

  • 统一不同类型的资源。比如 txtmp3jpeg 等不同的类型的资源都可以使用 URI 来标识
  • 统一不同协议。比如 httpftpmailto 等不同的协议都可以使用 URI 来标识
  • 统一新旧资源。引入新资源不会影响已有资源

Resource

官方定义没有对资源的范围做任何限制,所以任何被 URI 标识的东西都可以叫做资源。

常见的例子有电子文档、图片、固定用途的信息(比如今天 Los Angeles 的天气预报)、一项服务(比如 HTTP-to-SMS 网关)或者是一组资源。一个资源不一定需要能通过互联网访问,比如一个组织和图书馆中的一些书也可以是资源。抽象的概念也可以是资源,比如某些关系(比如雇佣关系或者婚姻关系)。

Identifier

从其他资源中区分一个资源,无论用什么手段实现(比如名称、地址)。标识符可以是 ID,但也可以不是,所以不要将标识符与 ID 等价。

语法规则

由 协议、authority、path、query 和 fragment 组成

URI         = scheme ":" hier-part [ "?" query ] [ "#" fragment ]

hier-part   = "//" authority path-abempty
/ path-absolute
/ path-rootless
/ path-empty

协议必填且不能为空,路径必填但可以为空。Authority 不是必填,但如果填写了要以 // 开头。

abempty 的意思是 absolute or empty

两个示例:

foo://example.com:8042/over/there?name=ferret#nose
\_/ \______________/\_________/ \_________/ \__/
| | | | |
协议 authority path query fragment
| _____________________|__
/ \ / \
urn:example:animal:ferret:nose

协议

为了 robustness,协议是大小写不敏感的(比如 HTTPhttp 是一样的),但为了一致性,只应该使用小写来表示协议。常用的有:http、https、ftp、mailto、file、data 和 irc。

Authority

很多 URI 协议都包含一个用于命名 authority 的分级元素,所以 URI 的路径会指向 authority 对应的命名空间。

Authority 以 // 开头,以下一个 /?# 或者 URI 的结尾而结尾。

语法如下:

authority   = [ userinfo "@" ] host [ ":" port ]

这里不描述 userinfo,因为很少用到。

host 语法如下:

host        = IP-literal / IPv4address / reg-name

host 同样是大小写不敏感,但为了一致性也要使用小写表示。

  • IP-literal:使用 IPV6 以及之后版本的 IP 地址,要使用 [] 包裹

  • IPv4address:使用点分十进制表示

      IPv4address = dec-octet "." dec-octet "." dec-octet "." dec-octet
    
    dec-octet   = DIGIT                 ; 0-9
    / %x31-39 DIGIT ; 10-99
    / "1" 2DIGIT ; 100-199
    / "2" %x30-34 DIGIT ; 200-249
    / "25" %x30-35 ; 250-255
  • reg-name:点分字符串,字符串可以是字母或者 -

可以看出 IPv4address 和 reg-name 并没有很明显的区分。所以为了区分这两者,采用 "first-match-wins" 算法:如果 host 匹配 IPv4address的规则,那它是一个 IPv4address 而不是 reg-name。

路径采用与文件系统一样的路径表示方式,以 / 分割。设计者这样说:

The relative URI syntax is just unix pathname syntax reused without apology.[2]

PS:设计者后悔采用 // 的设计,因为这个太笨拙了[2],从使用情况就可以看出,一般我们都不会用到 // 而是直接输入 // 后面的部分(比如输入 www.google.com 而不是 //www.google.com)。

URI、URL 和 URN

"Uniform Resource Locator" (URL) 是 URIs 的一个子集,提供一种定位资源而不是标识资源的方法。"Uniform Resource Name" (URN) 被用来指 "urn" 协议的 URIs 和其他通过名称属性定位的 URIs[1, 3]。

PS:关于 URL 的疑惑:上面说 URL 是定位而不是标识,但是 URI 中的 Identifier 就是标识符的意思,这不矛盾吗?

人们实际使用中也没有很明显的区分 URI 和 URL。普通用户都是说 URL,程序员混用 URL 和 URI,甚至连 RFC 中也是混用这两者(比如 "URI Syntax" (RFC 2396)、"Registration Procedures for URL Schemes" (RFC 2717))。

所以,综合各种情况,可以这样使用:

  • 如果资源会同时用 URL 和 URN 表示,那么使用 URI
  • 如果资源只用 URN 表示,那么用 URN,因为 URN 特殊并且少见
  • 如果资源只用 URL 表示,那么可以用 URL 或 URI,但是确定后最好统一使用而不是混用这两者

参考

  1. rfc3986
  2. https://www.w3.org/People/Berners-Lee/FAQ.html#etc
  3. https://tools.ietf.org/html/rfc3305

URI 、URL 和 URN的更多相关文章

  1. URI, URL, and URN

    URI: uniform resource identifier,统一资源标识符,用来唯一的标识一个资源. URL: uniform resource locator,统一资源定位器,它是一种具体的U ...

  2. URL、URN、URI的区别?

    URL.URN.URI区别 既然Web应用程序的文件等资源是放在服务器上,而服务器是因特网(Internet)上的主机,当然必须要有个方法,告诉浏览器到哪里取得文件等资源.通常会听到有人这么说:“你要 ...

  3. 关于URI URL URN

    刚琢磨.整理了关于escape.encodeURIComponent.encodeURI的知识.突然又对URI有点模糊了,遂整理了以下资源 : 资源一: URL,URI 和URN 的举例理解 资源二: ...

  4. URI、URL、URN介绍

    注:1. 仅从http(Hypertext Transfer Portocol)角度阐述,不涉及语言层面的类库. 2. 以下内容均参考<Http权威指南>一书. 一.万维网构成       ...

  5. uri,url.urn

    uri:Web上可用的每种资源 - HTML文档.图像.视频片段.程序等 - 由一个通过通用资源标志符(Universal Resource Identifier, 简称"URI" ...

  6. URI、URL和URN

    URI.URL和URN URI :Uniform Resource Identifier,通用资源标识符: URL:Uniform Resource Locator,统一资源定位符: URN:Unif ...

  7. URI、URL以及URN的区别

    首先,URI,是uniform resource identifier,统一资源标识符,用来唯一的标识一个资源.而URL是uniform resource locator,统一资源定位器,它是一种具体 ...

  8. URI、URL、URN

    URI.URL.URN URI(Uniform Resource Identifie):统一资源标识符 URL(Uniform Resource Locator):统一资源定位符 URN(Unifor ...

  9. Http权威指南笔记(一) URI URL URN 关系

    定义 URI:统一资源标识符(Uniform Resource Indentifier)用来标识服务器上的资源. URL:统一资源定位符(Uniform Resouce Locator)是资源标识符最 ...

  10. URI ,URL 和 URN

    URI : 统一资源标识符,用来唯一标识互联网资源,包括URL和URN URL:统一资源定位器 包含: 协议,域名,端口,路由,参数,hash https://i.cnblogs.com/EditPo ...

随机推荐

  1. android BaseAdapter getView 理解

    ListView是安卓中很经常使用的一个控件. 安卓设计使用Adapter来对ListView进行管理. 可是系统提供的Adapter无法满足一些复杂的显示情况,这个时候我们就须要使用BaseAdap ...

  2. (3)FluidMoveBehavior 之模仿 Windows Phone 开始菜单的 Tile 长按后排序

    这个工程和上一篇 (2)中介绍的排序大同小异,只是比上一篇交换复杂一点,不是通过单击进行交换, 而是拖动一个 Tile 到另一个 Tile 上时,判断两个 Tile 的中心距离是否符合条件来判断是否进 ...

  3. 兼容浏览器的min-height和min-width

    http://www.cnblogs.com/pigtail/archive/2012/06/28/2568646.html CSS 子元素宽度变宽时,如何撑开父元素https://zhidao.ba ...

  4. Makefile 9——为依赖关系文件建立依赖关系

    现在我们再对complicated项目做一些更改,增加程序文件间依赖关系的复杂度. /× main.c ×/ #include"foo.h" int main(void) { fo ...

  5. jar 打包命令详解

    原文: https://blog.csdn.net/marryshi/article/details/50751764 本文详细讲述了JAR命令的用法,对于大家学习和总结jar命令的使用有一定的帮助作 ...

  6. 卖座网一处SQL注射(Http Referer sqlinjection)

    漏洞作者: 猪猪侠 漏洞详情 披露状态: 2015-01-13: 细节已通知厂商并且等待厂商处理中2015-01-14: 厂商已经确认,细节仅向厂商公开2015-01-24: 细节向核心白帽子及相关领 ...

  7. scp采用无密码在两台linux服务器之间传输数据

    一.root用户: 1. 在主机A上执行如下命令来生成配对密钥: ssh-keygen -t rsa 按照提示操作,注意,不要输入passphrase.提示信息如下 Generating public ...

  8. c#用picturebox显示多页TIF

    //引用 using System.Drawing; using System.Drawing.Imaging; //以下是方法 private Bitmap myImage = null; priv ...

  9. [转]SOA接口的两种常用实现比较:SOAP vs REST

    原文链接:http://blog.csdn.net/zhaohuabing/article/details/39643127 SOA架构用于异构系统的协作,因此需要一种跨操作系统.跨语言的通用的消息交 ...

  10. 使用导出导入(datapump)方式将普通表切换为分区表

    随着数据库数据量的不断增长,有些表须要由普通的堆表转换为分区表的模式. 有几种不同的方法来对此进行操作,诸如导出表数据,然后创建分区表再导入数据到分区表:使用EXCHANGE PARTITION方式来 ...