这两天在写代码的时候,因为涉及到资源的位置,所以,须要在Java Bean中定义一些字段,用来表示资源的位置,好比:imgUrl,logoUri等等。可是,每次定义的时候,内心都很纠结,是该用imgUrl仍是imgUri呢?html
一样的,另一个问题:String HttpServletRequest.getRequestURI();和StringBuffer HttpServletRequest.getRequestURL();返回的内容有何不一样?为何会如此?java
带着这些问题到网上去搜了下,没发现让本身看了明白的解释,因而,想到了Java类库里有两个对应的类java.net.URI和java.net.URL,终于,在这两个类里的javadoc里找到了答案。api
首先,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。oracle
在Java的URI中,一个URI实例能够表明绝对的,也能够是相对的,只要它符合URI的语法规则。而URL类则不只符合语义,还包含了定位该资源的信息,所以它不能是相对的,schema必须被指定。ide
ok,如今回答文章开头提出的问题,究竟是imgUrl好呢,仍是imgUri好?显然,若是说imgUri是确定没问题的,由于即便它其实是 url,那它也是uri的一种。那么用imgUrl有没有问题呢?此时则要看它的可能取值,若是是绝对路径,可以定位的,那么用imgUrl是没问题的, 而若是是相对路径,那仍是不要用ImgUrl的好。总之,用imgUri是确定没问题的,而用imgUrl则要视实际状况而定。this
第二个,从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
总结一下:URL是一种具体的URI,它不只惟一标识资源,并且还提供了定位该资源的信息。URI是一种语义上的抽象概念,能够是绝对的,也能够是 相对的,而URL则必须提供足够的信息来定位,因此,是绝对的,而一般说的relative URL,则是针对另外一个absolute URL,本质上仍是绝对的。spa
注:这里的绝对(absolute)是指包含scheme,而相对(relative)则不包含scheme。.net
URI抽象结构 [scheme:]scheme-specific-part[#fragment]code
[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对象不适用。