Java路徑中的空格問題
1, TestURL().class.getResource("").getPath()或TestURL().class.getResource("").getFile()獲得的路徑,不能被FileReader()和FileWriter()直接應用。
原因是URL對空格,特殊字符(%,#,[]等)和中文進行瞭編碼處理。
例如:空格變為%20。
有解決方法(1),使用repaceAll("%20",' ')替換後,隻能解決空格問題。但是路徑中包含%和中文就不行瞭。
有解決方法(2),使用URLDecoder.decode(str,"UTF-8")解碼,但是隻能解決一部分,若路徑中含有+,也是不能解決的,原因是URL並不是完全用URLEncoder.encode(str,"UTF-8")編碼的,+號被解碼後,卻變成瞭空格。
方法(3),可以解決所有的問題,用TestURL().class.getResource("").toURI().getPath(),但是需要處理URISyntaxException異常,比較麻煩點。
java 路徑中的空格問題
若路徑中帶空格,則
1. uri.getpath();返回的路徑中的空格仍以“空格”的形式出現,如/F:/MyEclipse Workspace/project/bin/…
除此之外,URL返回的一切路徑中的空格都以“%20”的形式出現,uri.toString()也以“%20”的形式出現。
2. new File(String filePath);接受正確URI格式的參數和帶“空格”(非20%)的正確相對/絕對字符串路徑,否則即使給的路徑是正確的也會出現找不到文件的異常。
3. URL/URI返回的路徑分隔符都是“/”,File返回的路徑分隔符都為“\”。對於存在的文件返回的路徑字符串,空格都以"空格"出現,而不存在的路徑new出的file,getPath()返回的路徑中的空格,仍是new File(String filePath)的參數中原有的形式,即若filePath中是空格的getPath()返回的仍是空格,是“%20”的仍是“%20”。
4.new URL();的參數可以為正確的URI,或者為URI格式的字符串;若字符串是非完整的URI格式,則創建失敗。
5.File.toURI()會將file的路徑名中的“空格”轉化為“%20”,然後在路徑前加protocol:"file:/",而 File.toURL()隻會在file路徑前簡單的加上protocol:"file:/",而不會將“空格”轉化為“%20”,原來的無論是“空格” 還是“%20”都隻會原樣保留!
6.Woden 中WSDLReader.readWSDL(String s)的實現要將參數s轉化為URL,所以字符串參數s中一定不能有空格,應以“20%”代替。參數s最好為標準的URI格式的字符串。
java 路徑解決方案
Java的路徑問題,相對來說就比較繁雜。最近的工作涉及到創建和讀取文件的工作,現將實際使用中遇到的問題總結如下:
一 相對路徑的解釋
1.相對路徑(即相對於當前用戶目錄的相對路徑)均可通過以下方式獲得(不論是一般的java項目還是web項目)
String relativelyPath=System.getProperty("user.dir");
對於一般的java項目中的文件是相對於項目的根目錄,而對於web項目中的文件路徑,可能是服務器的某個路徑,同時不同的web服務器也不同(tomcat是相對於 tomcat安裝目錄\bin)。為此,個人認為,在web項目中,最好不要使用“相對於當前用戶目錄的相對路徑”。然而默認情況下,java.io 包中的類總是根據當前用戶目錄來分析相對路徑名。此目錄由系統屬性 user.dir 指定,通常是 Java 虛擬機的調用目錄。這就是說,在使用java.io包中的類時,最好不要使用相對路徑。否則,雖然在SE程序中可能還算正常,但是到瞭EE程序中,弄不好,就會帶來問題一片哦。
2.相對於classpath的相對路徑
如:相對於file:/D:/mywork/javaprj/MyTest/bin這個路徑的相對路徑。其中,bin是本項目的classpath。所有的Java源文件編譯後的.class文件復制到這個目錄中。
二 類加載目錄(即當運行某一類時獲得其裝載目錄)
1.不論是一般的java項目還是web項目,先定位到能看到包路徑的第一級目錄
InputStream is=ReadWrite.class.getClassLoader().getResourceAsStream("DeviceNO");
其中,DeviceNO文件的路徑為 項目名\src\DeviceNO;類ReadWrite所在包的第一級目錄位於src目錄下。
2.與1相似,不同的是此方法必須以'/'開頭
InputStream is=ReadWrite.class.getResourceAsStream("DeviceNO");
其中,DeviceNO文件的路徑為 項目名\src\DeviceNO;類ReadWrite所在包的第一級目錄位於src目錄下。
三. 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形式的絕對路徑。
作者“ERDP技術架構”