url和uri区别

在这里插入图片描述
这两天在写代码的时候,由于涉及到资源的位置,因此,需要在Java Bean中定义一些字段,用来表示资源的位置,比如:imgUrl,logoUri等等。但是,每次定义的时候,心里都很纠结,是该用imgUrl还是imgUri呢?

同样的,另外一个问题:String HttpServletRequest.getRequestURI();和StringBuffer HttpServletRequest.getRequestURL();返回的内容有何不同?为什么会如此?

带着这些问题到网上去搜了下,没发现让自己看了明白的解释,于是,想到了Java类库里有两个对应的类java.net.URI和java.net.URL,终于,在这两个类里的javadoc里找到了答案。

URIs, URLs, and URNs

首先,URI,是uniform resource identifier,统一资源标识符,用来唯一的标识一个资源。而URL是uniform resource locator,统一资源定位器,它是一种具体的URI,即URL可以用来标识一个资源,而且还指明了如何locate这个资源。而URN,uniform resource name,统一资源命名,是通过名字来标识资源,比如mailto:java-net@java.sun.com。也就是说,URI是以一种抽象的,高层次概念定义统一资源标识,而URL和URN则是具体的资源标识的方式。URL和URN都是一种URI。

在Java的URI中,一个URI实例可以代表绝对的,也可以是相对的,只要它符合URI的语法规则。而URL类则不仅符合语义,还包含了定位该资源的信息,因此它不能是相对的,schema必须被指定。

ok,现在回答文章开头提出的问题,到底是imgUrl好呢,还是imgUri好?显然,如果说imgUri是肯定没问题的,因为即使它实际上是url,那它也是uri的一种。那么用imgUrl有没有问题呢?此时则要看它的可能取值,如果是绝对路径,能够定位的,那么用imgUrl是没问题的,而如果是相对路径,那还是不要用ImgUrl的好。总之,用imgUri是肯定没问题的,而用imgUrl则要视实际情况而定。

第二个,从HttpServletRequest的javadoc中可以看出,getRequestURI返回一个String,“the part of this request’s URL from the protocol name up to the query string in the first line of the HTTP request”,比如“POST /some/path.html?a=b HTTP/1.1”,则返回的值为”/some/path.html”。现在可以明白为什么是getRequestURI而不是getRequestURL了,因为此处返回的是相对的路径。而getRequestURL返回一个StringBuffer,“The returned URL contains a protocol, server name, port number, and server path, but it does not include query string parameters.”,完整的请求资源路径,不包括querystring。

总结一下:URL是一种具体的URI,它不仅唯一标识资源,而且还提供了定位该资源的信息。URI是一种语义上的抽象概念,可以是绝对的,也可以是相对的,而URL则必须提供足够的信息来定位,所以,是绝对的,而通常说的relative URL,则是针对另一个absolute URL,本质上还是绝对的。

注:这里的绝对(absolute)是指包含scheme,而相对(relative)则不包含scheme。

URI抽象结构 [scheme:]scheme-specific-part[#fragment]

[scheme:][//authority][path][?query][#fragment]

authority为[user-info@]host[:port]

参考资料:

http://docs.oracle.com/javase/1.5.0/docs/api/java/net/URI.html

http://en.wikipedia.org/wiki/Uniform_Resource_Identifier

http://docs.oracle.com/javaee/5/api/javax/servlet/http/HttpServletRequest.html

ps:

java.net.URL类不提供对标准RFC2396规定的特殊字符的转义,因此需要调用者自己对URL各组成部分进行encode。而java.net.URI则会提供转义功能。因此The recommended way to manage the encoding and decoding of URLs is to use java.net.URI. 可以使用URI.toURL()和URL.toURI()方法来对两个类型的对象互相转换。对于HTML FORM的url encode/decode可以使用java.net.URLEncoder和java.net.URLDecoder来完成,但是对URL对象不适用。

U2FsdGVkX1877jSz5Fs+wlvYQNb4/e0/HzGMlX6u3BN/yjBscgBbSEiEqpQhe51A
yAf9jmWJBceGjhzy63mr5Mv7SiuF7HVQCVWCvwB4Ctfi//X7kpYpXvGI/NtI6jcC
KOt86hFdcakKHfsNgvXOcO9UVnnqkqKiy3O7t79HeSEeQLKhcovNrIMBKEaDyF/j
ldb+DyP2qxQXqWSW7eL5RPXfkiiBo5ZP9BHfgATlpdowoIRNEDSgkHfwrw4nHhbV
Ew21uAFSodHoV6ikBS4K+DblHXgwUcw6Pkq2BSH8gPKpe1P65SV7JbbLfSfRiSP9
xKDay4jJo/MsPu7MpO/SW6DQYnxjfMo3Iv/pHbDkoU7kyBXd7jIYIMJA9NM42F1g
KNEf9XtiKKHS7zRU6aL+diU4CFzE1tg8kvJP08LmaZLtTaEEzqkSPGWWJkzZ2KPJ
BOmPq3zAS1Mrfqi9SltMcK3yZJ7sGQb9SRRYIyjUQcwMtZ0PYJ5JXVCk2EyQE3g3
1c9iZ7Ez9xLc4lG42lB/EKtzbf9SFXKHqAJWdKc8aFZzutIzPFh2/YplZ9OgrFWJ
OXrAM2nNq5adV3hrGp5rGhkIABK5Niai1U+ypBySvdc2zuHHRmMG43dhLfPzDvSs
eTcg1t/Fbw/M2cIZvWCCJV3u4Unp1hM0EijAGpO7TVo5wiCwq+DUa2S/gCrd14X0
h4aGRdEkWV6nIIWqDcFXwDppuRTr6VRI2HYioXP5BzIc+vNMu9WVpLFpHg+l4QO8
L8l5nI46BLF7gNMJd9cj879uvClkQoAJMnRP4Vdd9oysqWs6gbsdNrzJ2+b4vd5h
lkESOniV5PT0iZHrWLrO4PCuXoRBMrSx2ckpC9ZPUz47YIJAL99K/EGqy5kW4S2y
iYiLVUMAIOR3XzrshyjYXqyxbP2hO266CrLENAUv9WgSLjaY0k3BBliIXJtXDsD8
LQyHEQow6c01bqKximCXuxeyRW1JmZLkhnX4iuUHm3yNc2aGAFoMOhn/AzUL0dw9
Y6o/razeB74ti5M+SvON10wSrslaFzBJDKVjagP1qvXHZf35wdrdA1vh7jU1BJGa
WohlEHjoz62Iq0Jsm5VPd0lpcGZodDIgVIC7rYzsvKZW0TpBqa++yuafaaYCcsdM
vHzjdZyf6nXXbChUsK+L0c7Z+3c+wjes76Rl7T9TBDMUbPzAVyDEGIzRw3YlO9Hj
UqQcvCkZm/2zQE+3eLOxHxvhrACdIo5dIz1SqZBomLjXq0JrYH3jWQOE7lWVXFNO
ROP/zF0oaFU/uwoQLRmKdKzBNyUM9VGElICE3FO9HA487+E/FWm5MdrUpe+RFbyh
C/H2wjFU3J4jMcHZoOACZO8V8uO8atgXty209I9qkTwLYuDA55SaPza0jYJmEtav
ceKuYaUnZ807uHUc/OcLeyntf5S5e+siGzjQbdLYrBuDEJiHhbpKORXRFXFPOn13
lihRQIc44VskwrxZ3BpZiP2R7n77NrgoDZbJHoLRNXXnakCNQfmdc+ZTNHUhPiG6
DfJ5N1U25HzCZ3WXvNxGBfjGdUOdxLgpmg6RL8DIZwXLHqlfKqb0Ei2sOZRgsLl0
NNDpjgYNb4WoC8hZEqe3CemaLiKKkjwpgEtvLxOmYFQJCg1bMrUHy3kJt+kagK1a
aAOb0HjKL8gKJLWxagNtiRn4kcbO+zWUnCm1enuxe8sI+bEXLCne5nDnMNhtN6pt
BHBlI25jKZxx0bl2KUYS+IlgL2+o64GeJmWoBuIV/zHcdrChK8j+i4dKYWA0Nymj
SbBaYEpjdp6Sh8qoN5v9kB6FE/UXSbHRZl0V8bfPtdFCrnbKxmohi7ciyAXjE84n
bDWqHk6lFeKX3rEMIA1WgYqk7FHzkk1SZ1uuEtLQ78B/TI3eNfA11f3JkUtWWq8X
Dfp40JO4jWlpE0lGXGnd6XRMVniVXr76ZZ3g40kLjixOMiTaKgQMSJmqkHgtih1J
4JbgeYE+wA2mgbgwqJ39Twnh2yyNYdO+h5232i6Qm5CRpB4ftmJ8deUSeEt+mWO8
KveuA4V23a3MjVAZYBTUfrttKq317dmaMSWF4yslnN556sma99YD3yR74pAqlNrU
wq2V+/xMGC3AnyPMqw7k1RfbUuDV7tpkPi9Lfln4GerGbT07vSl17r/4h8nejHmc
ylKrJCLgr2h5Ug5wbkxMLlBhrNyH9bNA6/9SGvbB74dOF97jMLf0VlHP6VpFEOPi
9oD+DvX8yFDrGw1fkZE+eL6Qz2y/tXUOpN2Lt1Wxt2AHveS03aIcnIqc56RrlFpp
fSQFKvhtB3tk7MtF823V0v5TfbECwh6BPBJLdWbAwY+H4+n/7ajycuEdRoRnCqcC
vlL+Yxl8Nitgj4tq0o1XJnGgLb0Sddv4wV2UCJjd2AAiT1BgaRtzQF5M37AucBKt
ix0RXJ9iGkAp4AXZ1gK355eTrvJoDmh5CMRtYsHxntFHf7791eLWXHGfWTM9rcPw
zH6G/8UtCf3F9VSZalo3l4wyRTNcK3DQerfx7RyPyZprQJkXAxQpu+YAbdQ16aM6
xNwZwMGVjnIgYAnRZjCx79lOrqihc2yGDklZX5Z5erBuFrkle7jCtToTyfXcJMC0
6Wj7g/MG43Iav8gaIjuzzc1HmHJXwWOOYlVwafw1CmpMo10MeHE5fFRcG8NT+4WL
TtRxlrKNTFkS8nY8fMfUZipqoIBsP4sWU+VFgTa8En44scv1edW2bdYKCYMpfLQR
zIQgTEKaRdHaBxSrtqx05abTaEwjphqclBe1QcVG7ZUY/dztsI7ZPPnFyaUcVY1/
Ho8bmCd2e0kyjLoW4vXp8CPXiPIndWqOcBsUqeK4IXOyFHyj9O1Lc1K+GQJQ/JgY
z6M+EL2dQWuJgwsaGcF2PTfad8ffBK9bg/mb2jiZQiGmg5pX1JH2nSbi6SsEmFx7
FYUnWovqot2NQZzPOev5TRXGJ4+yk09YdqjjJDfMKG01b0Nj+8z+R/cE6fV4zzqd
HO0LYFAMVDZbkjgoOVnXVzG3GMzw+P/h16X0U4ymx0fl3MZ33+s8GljKtUXv5p95
kuK5hwSuEGIs4tsuHoNmaQXPUVn5Cg3xvuj0aJy0TvyGmIdw/Npucegm3km5oSJN
xVqNvJB7KVWJpEPEywtB6x42n5cfUEvSNlbeDegktuXtU9b0ZLJavpzxoqIqGAR7
1Dj1nNCWp0K9F221/PdmSws445J2Cwc5WA54VJposL/V5cKQ9/QPqHync39HFOd1
I1lgyvbHhypESkFyvxRS1d3TbQmllR984k3fcdQyXQuiX/7GgRL3QxKesOjCapaP
PreihoLX15yifaM+KbYSUglDFapzPGWITVzPztX9NOErSyaeW2982j5dhQKcUd0Z
6hupDUP0+VqfIiKy5nNrYM93BkIsZUj68Mrgq7+BwxTtxk4seVTh2rP73VJnSexr
Kobnffdu+CCu7ZW2qct5jAc3jLShKAYdSuACIJmWkKwTqY8xwJffsUumqKRH1xGS
rqBs9fosN/BchfR7FaycYuAXafB8SSs44cq9Ln9V9PzE6IQn7EsVub8ahqfUtnHI
iPUQzM6W96QgrXyYV601xMv2rV3ZQxGEKRWPteLWQykNzZ8s6Ph+6PUxjPbap0jV
nkZRTtN157nyW2QkGEnxDw==

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值