URL包里的URL.getpath()對路徑中空格識別為%20的處理辦法


方法(1),使用repaceAll("%20",' ')替換后,只能解決空格問題。但是路徑中包含%和中文就不行了。 
方法(2),使用URLDecoder.decode(str,"UTF-8")解碼,但是只能解決一部分,若路徑中含有+,也是不能解決的,原因是URL並不是完全用  URLEncoder.encode(str,"UTF-8")編碼的,+號被解碼后,卻變成了空格。
方法(3),可以解決所有的問題,用TestURL().class.getResource("").toURI().getPath(),但是需要處理URISyntaxException異常,比較麻煩點。

若路徑中帶空格,則 
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的路徑問題,相對來說就比較繁雜。最近的工作涉及到創建和讀取文件的工作,現將實際使用中遇到的問題總結如下: 


一 相對路徑的解釋 

    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形式的絕對路徑。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM