JAVA文件中獲取路徑及WEB應用程序獲取路徑方法

1. 基本概念的理解

`絕對路徑`:你應用上的文件或目錄在硬盤上真正的路徑,如:URL、物理路徑javascript

例如:html

c:/xyz/test.txt表明了test.txt文件的絕對路徑;java

http://www.sun.com/index.htm也表明了一個URL絕對路徑;web

`相對路徑`:相對與某個基準目錄的路徑,包含Web的相對路徑(HTML中的相對目錄)。瀏覽器

例如:服務器

在Servlet中,"/"表明Web應用的根目錄,和物理路徑的相對錶示。網絡

例如:app

"./"表明當前目錄,"../"表明上級目錄。這種相似的表示,也是屬於相對路徑。eclipse

2. 關於JSP/Servlet中的相對路徑和絕對路徑。

2.1 服務器端的地址

`服務器端的相對地址`:指的是相對於你的web應用的地址,這個地址是在服務器端解析的(不一樣於html和javascript中的相對地址,他們是由客戶端瀏覽器解析的)也就是說這時候在jsp和servlet中的相對地址應該是相對於你的web應用,即相對於http://192.168.0.1/webapp/的。webapp

其用到的地方有:

`forward:servlet中的request.getRequestDispatcher(address);`這個address是在服務器端解析的,因此,你要forward到a.jsp應該這麼寫:

`request.getRequestDispatcher(「/user/a.jsp」)`這個/相對於當前的web應用webapp,其絕對地址就是:http://192.168.0.1/webapp/user/a.jsp

2.2 客戶端的地址

`全部的html頁面中的相對地址`都是相對於服務器根目錄(http://192.168.0.1/)的,而不是(根目錄下的該Web應用的目錄)http://192.168.0.1/webapp/的。

`Html中的form表單的action屬性的地址`應該是相對於服務器根目錄(http://192.168.0.1/)的,因此,若是提交到a.jsp爲:action="/webapp/user/a.jsp"或action="/user/a.jsp",提交到servlet爲action="/webapp/handleservlet"。

Javascript也是在客戶端解析的,因此其相對路徑和form表單同樣。

所以,通常狀況下,在JSP/HTML頁面等引用的CSS,Javascript,Action等屬性前面最好都加上webapp應用名稱,以確保所引用的文件都屬於Web應用中的目錄。

另外,應該儘可能避免使用相似".","./","../../"等相似的相對該文件位置的相對路徑,這樣當文件移動時,很容易出問題。

3. JSP/Servlet中得到當前應用的相對路徑和絕對路徑

3.1 JSP中得到當前應用的相對路徑和絕對路徑

根目錄所對應的絕對路徑:`request.getRequestURI();`

文件的絕對路徑:`application.getRealPath(request.getRequestURI());`

當前web應用的絕對路徑:`application.getRealPath("/");`

取得請求文件的上層目錄:`newFile(application.getRealPath(request.getRequestURI())).getParent();`

3.2 Servlet中得到當前應用的相對路徑和絕對路徑

根目錄所對應的絕對路徑:`request.getServletPath();`

文件的絕對路徑:`request.getSession().getServletContext().getRealPath();`

當前web應用的絕對路徑:`servletConfig.getServletContext().getRealPath("/");`

ServletContext對象得到幾種方式:

javax.servlet.http.HttpSession.getServletContext()
javax.servlet.jsp.PageContext.getServletContext()
javax.servlet.ServletConfig.getServletContext()

4. java的Class中得到相對路徑,絕對路徑的方法

4.1 單獨的Java類中得到絕對路徑

根據java.io.File的Doc文擋,可知:`默認狀況下newFile("/")表明的目錄爲:System.getProperty("user.dir");`。

如下程序得到執行類的當前路徑:

package org.cheng.file; 
import java.io.File; 
public class FileTest{ 
    public static void main(String[]args)throws Exception{ 
        System.out.println(Thread.currentThread().getContextClassLoader().getResource("")); 
        System.out.println(FileTest.class.getClassLoader().getResource("")); 
        System.out.println(ClassLoader.getSystemResource("")); 
        System.out.println(FileTest.class.getResource("")); 
        System.out.println(FileTest.class.getResource("/"));//Class文件所在路徑 
        System.out.println(newFile("/").getAbsolutePath()); 
        System.out.println(System.getProperty("user.dir")); 
    } 
}


4.2 服務器中的Java類得到當前路徑(來自網絡)

(1).Weblogic

WebApplication的系統文件根目錄是你的weblogic安裝所在根目錄。

例如:若是你的weblogic安裝在c:eaweblogic700.....

那麼,你的文件根路徑就是c:.

因此,有兩種方式可以讓你訪問你的服務器端的文件:

a.使用絕對路徑:

好比將你的參數文件放在c:yourconfig/yourconf.properties,

直接使用newFileInputStream("yourconfig/yourconf.properties");

b.使用相對路徑:

相對路徑的根目錄就是你的webapplication的根路徑,即WEB-INF的上一級目錄,將你的參數文件放在yourwebapp/yourconfig/yourconf.properties,

這樣使用:

newFileInputStream("./yourconfig/yourconf.properties");

這兩種方式都可,本身選擇。

(2).Tomcat

在類中輸出System.getProperty("user.dir");顯示的是%Tomcat_Home%/bin

(4).如何讀相對路徑哪?

在Java文件中getResource或getResourceAsStream都可

例:`getClass().getResourceAsStream(filePath);` // filePath能夠是"/filename",這裏的/表明web發佈根路徑下WEB-INF/classes

默認使用該方法的路徑是:WEB-INF/classes。已經在Tomcat中測試。

5.讀取文件時的相對路徑,避免硬編碼和絕對路徑的使用。

5.1採用Spring的DI機制得到文件,避免硬編碼。

Java中使用的路徑,分爲兩種:絕對路徑和相對路徑。具體而言,又分爲四種:

(1)URI形式的絕對資源路徑

如:file:/D:/java/eclipse32/workspace/jbpmtest3/bin/aaa.b

URL是URI的特例。URL的前綴/協議,必須是Java認識的。URL能夠打開資源,而URI則不行。

URL和URI對象能夠互相轉換,使用各自的toURI(),toURL()方法便可!

(2)本地系統的絕對路徑

D:/java/eclipse32/workspace/jbpmtest3/bin/aaa.b

Java.io包中的類,須要使用這種形式的參數。可是,它們通常也提供了URI類型的參數,而URI類型的參數,接受的是URI樣式的String。所以,經過URI轉換,仍是能夠把URI樣式的絕對路徑用在java.io包中的類中。

(3)相對於classpath的相對路徑

如:相對於 file:/D:/java/eclipse32/workspace/jbpmtest3/bin/這個路徑的相對路徑。其中,bin是本項目的classpath。全部的Java源文件編譯後的.class文件複製到這個目錄中。

(4)相對於當前用戶目錄的相對路徑

就是相對於System.getProperty("user.dir")返回的路徑。

對於通常項目,這是項目的根路徑。對於JavaEE服務器,這多是服務器的某個路徑。這個並無統一的規範!

因此,絕對不要使用「相對於當前用戶目錄的相對路徑」。

然而:默認狀況下,java.io 包中的類老是根據當前用戶目錄來分析相對路徑名。此目錄由系統屬性 user.dir 指定,一般是 Java 虛擬機的調用目錄。

這就是說,在使用java.io包中的類時,最好不要使用相對路徑。不然,雖然在J2SE應用程序中可能還算正常,可是到了J2EE程序中,必定會出問題!並且這個路徑,在不一樣的服務器中都是不一樣的!


`相對路徑最佳實踐`:推薦使用相對於當前classpath的相對路徑,所以,咱們在使用相對路徑時,應當使用相對於當前classpath的相對路徑。

`ClassLoader類的getResource(String name),getResourceAsStream(String name)等方法`,使用相對於當前項目的classpath的相對路徑來查找資源。


讀取屬性文件經常使用到的`ResourceBundle類的getBundle(String path)`也是如此。


經過查看ClassLoader類及其相關類的源代碼,它實際上仍是使用了URI形式的絕對路徑。經過獲得當前classpath的URI形式的絕對路徑,構建了相對路徑的URI形式的絕對路徑。(這個其實是猜測,由於JDK內部調用了SUN的源代碼,而這些代碼不屬於JDK,不是開源的。)


`相對路徑本質上仍是絕對路徑`,所以,歸根結底,Java本質上只能使用絕對路徑來尋找資源。全部的相對路徑尋找資源的方法,都不過是一些便利方法。不過是API在底層幫助咱們構建了絕對路徑,從而找到資源的!


下面是一些獲得classpath和當前類的絕對路徑的一些方法。你可能須要使用其中的一些方法來獲得你須要的資源的絕對路徑。

1. `FileTest.class.getResource("")`:獲得的是當前類FileTest.class文件的URI目錄。不包括本身!

如:file:/D:/java/eclipse32/workspace/jbpmtest3/bin/com/test/

2. `FileTest.class.getResource("/")`:獲得的是當前的classpath的絕對URI路徑。

如:file:/D:/java/eclipse32/workspace/jbpmtest3/bin/

3. `Thread.currentThread().getContextClassLoader().getResource("")`:獲得的也是當前ClassPath的絕對URI路徑。

如:file:/D:/java/eclipse32/workspace/jbpmtest3/bin/

4. `FileTest.class.getClassLoader().getResource("")`:獲得的也是當前ClassPath的絕對URI路徑。

如:file:/D:/java/eclipse32/workspace/jbpmtest3/bin/

5. `ClassLoader.getSystemResource("")`:獲得的也是當前ClassPath的絕對URI路徑。

如:file:/D:/java/eclipse32/workspace/jbpmtest3/bin/


`推薦使用`:Thread.currentThread().getContextClassLoader().getResource("")來獲得當前的classpath的絕對路徑的URI表示法。


Web應用程序中資源的尋址

上文中說過,當前用戶目錄,即相對於System.getProperty("user.dir")返回的路徑。對於JavaEE服務器,這多是服務器的某個路徑,這個並無統一的規範!而不是咱們發佈的Web應用程序的根目錄!這樣,在Web應用程序中,咱們絕對不能使用相對於當前用戶目錄的相對路徑。

在Web應用程序中,咱們通常經過`ServletContext.getRealPath("/")`方法獲得Web應用程序的根目錄的絕對路徑。這樣,咱們只須要提供相對於Web應用程序根目錄的路徑,就能夠構建出定位資源的絕對路徑。

這是咱們開發Web應用程序時通常所採起的策略。


通用的相對路徑解決辦法

Java中各類相對路徑很是多,不容易使用,很是容易出錯。所以,編寫了一個便利方法,幫助更容易的解決相對路徑問題。

Web應用程序中使用JavaSE運行的資源尋址問題

在JavaSE程序中,咱們通常使用classpath來做爲存放資源的目的地。可是,在Web應用程序中,咱們通常使用classpath外面的WEB-INF及其子目錄做爲資源文件的存放地。

在Web應用程序中,咱們通常經過ServletContext.getRealPath("/")方法獲得Web應用程序的根目錄的絕對路徑。這樣,咱們只須要提供相對於Web應用程序根目錄的路徑,就能夠構建出定位資源的絕對路徑。

Web應用程序,能夠做爲Web應用程序進行發佈和運行。可是,咱們也經常會以JavaSE的方式來運行Web應用程序的某個類的main方法。或者,使用JUnit測試。這都須要使用JavaSE的方式來運行。

這樣,咱們就沒法使用ServletContext.getRealPath("/")方法獲得Web應用程序的根目錄的絕對路徑。

而JDK提供的ClassLoader類,它的getResource(String name),getResourceAsStream(String name)等方法,使用相對於當前項目的classpath的相對路徑來查找資源。

讀取屬性文件經常使用到的ResourceBundle類的getBundle(String path)也是如此。

它們都只能使用相對路徑來讀取classpath下的資源,沒法定位到classpath外面的資源。


Classpath外配置文件讀取問題

如,咱們使用測試驅動開發的方法,開發Spring、Hibernate、iBatis等使用配置文件的Web應用程序,就會遇到問題。

儘管Spring本身提供了FileSystem(也就是相對於user.dir目錄)來讀取Web配置文件的方法,可是終究不是很方便。並且與Web程序中的代碼使用方式不一致!

至於Hibernate,iBatis就更麻煩了!只有把配置文件移到classpath下,不然根本不可能使用測試驅動開發!


這怎麼辦?

通用的相對路徑解決辦法

面對這個問題,編寫一個助手類`ClassLoaderUtil`,提供一個便利方法[public static URL getExtendResource(String relativePath)]。在Web應用程序等一切Java程序中,須要定位classpath外的資源時,都使用這個助手類的便利方法,而不使用Web應用程序特有的ServletContext.getRealPath("/")方法來定位資源。


利用classpath的絕對路徑,定位全部資源

這個便利方法的實現原理,就是「利用classpath的絕對路徑,定位全部資源」。

ClassLoader類的getResource("")方法可以獲得當前classpath的絕對路徑,這是全部Java程序都擁有的能力,具備最大的適應性!

而目前的JDK提供的ClassLoader類的getResource(String 相對路徑)方法,只能接受通常的相對路徑。這樣,使用ClassLoader類的getResource(String 相對路徑)方法就只能定位到classpath下的資源。

若是,它可以接受「../」這樣的參數,容許咱們用相對路徑來定位classpath外面的資源,那麼咱們就能夠定位位置的資源!

固然,我沒法修改ClassLoader類的這個方法,因而,我編寫了一個助手類ClassLoaderUtil類,提供了[public static URL getExtendResource(String relativePath)]這個方法。它可以接受帶有「../」符號的相對路徑,實現了自由尋找資源的功能。

相關文章
相關標籤/搜索