class 類: javascript
public static String getBasePath() { String savaPath = ""; savaPath = 類.class.getRecource("").getPath(); int i = savaPath.lastIndexOf("WEB-INF"); savaPath = savaPath.substring(1, i); return savaPath; }
這個是轉換: java
String savePath = java.net.URLDecoder.decode(savaPath,"utf-8");
java 路徑解決方案Java的路徑問題,相對來講就比較繁雜。最近的工做涉及到建立和讀取文件的工做,現將實際使用中遇到的問題總結以下:
一 相對路徑的解釋
1.相對路徑(即相對於當前用戶目錄的相對路徑)都可經過如下方式得到(不管是通常的java項目仍是web項目) web
String relativelyPath=System.getProperty("user.dir");
對於通常的java項 目中的文件是相對於項目的根目錄,而對於web項目中的文件路徑,多是服務器的某個路徑,同時不一樣的web服務器也不一樣(tomcat是相對於 tomcat安裝目錄\bin)。爲此,我的認爲,在web項目中,最好不要使用「相對於當前用戶目錄的相對路徑」。然而默認狀況下,java.io 包中的類老是根據當前用戶目錄來分析相對路徑名。此目錄由系統屬性 user.dir 指定,一般是 Java 虛擬機的調用目錄。這就是說,在使用java.io包中的類時,最好不要使用相對路徑。不然,雖然在SE程序中可能還算正常,可是到了EE程序中,弄很差,就會帶來問題一片哦。 tomcat
2.相對於classpath的相對路徑
如:相對於file:/D:/mywork/javaprj/MyTest/bin這個路徑的相對路徑。其中,bin是本項目的classpath。全部的Java源文件編譯後的.class文件複製到這個目錄中。 服務器
二 類加載目錄(即當運行某一類時得到其裝載目錄)
1.不管是通常的java項目仍是web項目,先定位到能看到包路徑的第一級目錄 app
InputStream is=ReadWrite.class.getClassLoader().getResourceAsStream("DeviceNO");
其中,DeviceNO文件的路徑爲 項目名\src\DeviceNO;類ReadWrite所在包的第一級目錄位於src目錄下。 webapp
2.與1類似,不一樣的是此方法必須以'/'開頭 函數
InputStream is=ReadWrite.class.getResourceAsStream("DeviceNO");
其中,DeviceNO文件的路徑爲 項目名\src\DeviceNO;類ReadWrite所在包的第一級目錄位於src目錄下。 this
三. web項目根目錄獲取
1. 可創建一個servlet,在其init方法中寫入以下語句 編碼
ServletContext sc=this.getServletContext(); String temp=sc.getRealPath("/");
獲得的輸出路徑結果相似:"D:\Apache\Tomcat6.0\webapps\windpower\ " (windpower爲項目名字) ,若是是調用了s1.getRealPath("")則輸出"D:\Apache\Tomcat6.0\webapps\windpower"(注意,在 最後少了一個"\")
2. 在httpServletRequest中,能夠經過下面語句
String cp=request.getSession().getServletContext().getRealPath("/"); 獲得的輸出路徑結果相似:"D:\Apache\Tomcat6.0\webapps\windpower\ "
四 .類路徑( classpath)的獲取(在Eclipse/MyEclipse中,則爲得到src或者classes目錄的路徑)
方法1. Thread.currentThread().getContextClassLoader().getResource("").getPath()
例如:
String path=Thread.currentThread().getContextClassLoader().getResource("").getPath(); System.out.println(path);
打印:「/D:/windpower/WebRoot/WEB-INF/classes/」
方法2. ParsingXML.class.getClassLoader().getResource("").getPath()(ParsingXML爲src某一個包中的類,下同)
例如:
String path=ParsingXML.class.getClassLoader().getResource("").getPath(); System.out.println("ParsingXML.class.getClassLoader().getResource--"+path);
打印: 「ParsingXML.class.getClassLoader().getResource--/D:/windpower/WebRoot/WEB-INF/classes/」
另外,若是想把文件放在某一包中,則能夠經過如下方式得到到文件所在目錄,即先定位到該包的最後一級目錄。
ParsingXML.class.getResource("").getPath();
例如:
String path=ParsingXML.class.getResource("").getPath(); System.out.println("ParsingXML.class.getResource---"+p2);
打印: 「ParsingXML.class.getResource---/D:/windpower/WebRoot/WEB-INF/classes/parsing/ 」(ParsingXML爲src目錄下parsing包中的類)
五. 屬性文件的讀取:
方法1.
static { ps = new Properties(); try { InputStream in = ReadWrite.class.getResourceAsStream("DeviceNO"); ps.load(in); in.close(); } catch (Exception e) { e.printStackTrace(); } ps.getProperty("key") ; }
方法2.
Locale locale = Locale.getDefault(); ResourceBundle localResource = ResourceBundle.getBundle("windpower/DeviceNOProperties", locale); String value = localResource.getString("1"); System.out.println("DeviceNO: " + value);
工程src目錄下文件DeviceNOProperties.properties(名字後綴必須爲properties)文件內容以下:1=3輸出結果爲:「DeviceNO:3」
六.編碼轉換問題:
ClassLoader的getResource方法使用了utf-8對路徑信息進行了編碼,當路徑中存在中文和空格時,他會對這些字符進行轉換,這樣, 獲得的每每不是咱們想要的真實路徑,在此,調用了URLDecoder的decode方法進行解碼,以便獲得原始的中文及空格路徑
例如:結果是file:/C:/Documents%20and%20Settings/%e5%ba%84%e6%99%93%e6%af%85
/Local%20Settings/Temp/temp0.jar!/db/dmozdata.mdb
而咱們指望是 C:/Documents 路徑p source 等等。這裏咱們只要在獲取到路徑以前把返回值decode下就能夠了. 用utf-8編碼. Java代碼 :
String configPath = this.getClass().getClassLoader().getResource("allowPath.xml").getFile(); configPath = java.net.URLDecoder.decode(configPath,"utf-8");
另外java中URL 的編碼和解碼函數java.net.URLEncoder.encode(String s)和java.net.URLDecoder.decode(String s);在javascript 中URL 的編碼和解碼函數escape(String s)和unescape(String s) ;
七.總結:
咱們在使用相對路徑時,應當使用相對於當前classpath的相對路徑。
ClassLoader類的getResource(String name),getResourceAsStream(String name)等方法,使用相對於當前項目的classpath的相對路徑來查找資源。
讀取屬性文件經常使用到的ResourceBundle類的getBundle(String path)也是如此。
經過查看ClassLoader類及其相關類的源代碼,發現它實際上仍是使用了URI形式的絕對路徑。經過獲得當前classpath的URI形式的絕對路徑,再去構建相對路徑的URI形式的絕對路徑。
本身遇到的問題:
因爲安裝的tomcat在安裝後,生成的名稱爲Tomcat 6.0,就是中間有個空格,因此在查找web項目的路徑時,就報錯。
private String getBasePath() throws IOException { ClassPathResource resource = new ClassPathResource("../../"); return resource.getURL().getPath(); }後來發現,是在去路徑時,把空格讀成「%20」了,就把上面的方法裏返回一句添加了替換: return resource.getURL().getPath().replace("%20", " "); 就解決了問題。