Java路徑中的空格問(wèn)題
1, TestURL().class.getResource("").getPath()或TestURL().class.getResource("").getFile()獲得的路徑,不能被FileReader()和FileWriter()直接應用。
原因是URL對空格,特殊字符(%,#,[]等)和中文進(jìn)行了編碼處理。
例如:空格變?yōu)?20。
有解決方法(1),使用repaceAll("%20",' ')替換后,只能解決空格問(wèn)題。但是路徑中包含%和中文就不行了。
有解決方法(2),使用URLDecoder.decode(str,"UTF-8")解碼,但是只能解決一部分,若路徑中含有+,也是不能解決的,原因是URL并不是完全用URLEncoder.encode(str,"UTF-8")編碼的,+號被解碼后,卻變成了空格。
方法(3),可以解決所有的問(wèn)題,用TestURL().class.getResource("").toURI().getPath(),但是需要處理URISyntaxException異常,比較麻煩點(diǎn)。
java 路徑中的空格問(wèn)題
若路徑中帶空格,則
1. uri.getpath();返回的路徑中的空格仍以“空格”的形式出現,如/F:/MyEclipse Workspace/project/bin/...
除此之外,URL返回的一切路徑中的空格都以“%20”的形式出現,uri.toString()也以“%20”的形式出現。
2. new File(String filePath);接受正確URI格式的參數和帶“空格”(非20%)的正確相對/絕對字符串路徑,否則即使給的路徑是正確的也會(huì )出現找不到文件的異常。
3. URL/URI返回的路徑分隔符都是“/”,File返回的路徑分隔符都為“\”。對于存在的文件返回的路徑字符串,空格都以"空格"出現,而不存在的路徑new出的file,getPath()返回的路徑中的空格,仍是new File(String filePath)的參數中原有的形式,即若filePath中是空格的getPath()返回的仍是空格,是“%20”的仍是“%20”。
4.new URL();的參數可以為正確的URI,或者為URI格式的字符串;若字符串是非完整的URI格式,則創(chuàng )建失敗。
5.File.toURI()會(huì )將file的路徑名中的“空格”轉化為“%20”,然后在路徑前加protocol:"file:/",而 File.toURL()只會(huì )在file路徑前簡(jiǎn)單的加上protocol:"file:/",而不會(huì )將“空格”轉化為“%20”,原來(lái)的無(wú)論是“空格” 還是“%20”都只會(huì )原樣保留!
6.Woden 中WSDLReader.readWSDL(String s)的實(shí)現要將參數s轉化為URL,所以字符串參數s中一定不能有空格,應以“20%”代替。參數s最好為標準的URI格式的字符串。
java 路徑解決方案
Java的路徑問(wèn)題,相對來(lái)說(shuō)就比較繁雜。最近的工作涉及到創(chuàng )建和讀取文件的工作,現將實(shí)際使用中遇到的問(wèn)題總結如下:
一 相對路徑的解釋
1.相對路徑(即相對于當前用戶(hù)目錄的相對路徑)均可通過(guò)以下方式獲得(不論是一般的java項目還是web項目)
String relativelyPath=System.getProperty("user.dir");
對于一般的java項目中的文件是相對于項目的根目錄,而對于web項目中的文件路徑,可能是服務(wù)器的某個(gè)路徑,同時(shí)不同的web服務(wù)器也不同(tomcat是相對于 tomcat安裝目錄\bin)。為此,個(gè)人認為,在web項目中,最好不要使用“相對于當前用戶(hù)目錄的相對路徑”。然而默認情況下,java.io 包中的類(lèi)總是根據當前用戶(hù)目錄來(lái)分析相對路徑名。此目錄由系統屬性 user.dir 指定,通常是 Java 虛擬機的調用目錄。這就是說(shuō),在使用java.io包中的類(lèi)時(shí),最好不要使用相對路徑。否則,雖然在SE程序中可能還算正常,但是到了EE程序中,弄不好,就會(huì )帶來(lái)問(wèn)題一片哦。
2.相對于classpath的相對路徑
如:相對于file:/D:/mywork/javaprj/MyTest/bin這個(gè)路徑的相對路徑。其中,bin是本項目的classpath。所有的Java源文件編譯后的.class文件復制到這個(gè)目錄中。
二 類(lèi)加載目錄(即當運行某一類(lèi)時(shí)獲得其裝載目錄)
1.不論是一般的java項目還是web項目,先定位到能看到包路徑的第一級目錄
InputStream is=ReadWrite.class.getClassLoader().getResourceAsStream("DeviceNO");
其中,DeviceNO文件的路徑為 項目名\src\DeviceNO;類(lèi)ReadWrite所在包的第一級目錄位于src目錄下。
2.與1相似,不同的是此方法必須以'/'開(kāi)頭
InputStream is=ReadWrite.class.getResourceAsStream("DeviceNO");
其中,DeviceNO文件的路徑為 項目名\src\DeviceNO;類(lèi)ReadWrite所在包的第一級目錄位于src目錄下。
三. web項目根目錄獲取
1. 可建立一個(gè)servlet,在其init方法中寫(xiě)入如下語(yǔ)句
ServletContext sc=this.getServletContext();
String temp=sc.getRealPath("/");
得到的輸出路徑結果類(lèi)似:"D:\Apache\Tomcat6.0\webapps\windpower\ " (windpower為項目名字) ,如果是調用了s1.getRealPath("")則輸出"D:\Apache\Tomcat6.0\webapps\windpower"(注意,在最后少了一個(gè)"\")
2. 在httpServletRequest中,可以通過(guò)下面語(yǔ)句
String cp=request.getSession().getServletContext().getRealPath("/"); 得到的輸出路徑結果類(lèi)似:"D:\Apache\Tomcat6.0\webapps\windpower\ "
四 .類(lèi)路徑( 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某一個(gè)包中的類(lèi),下同)
例如:
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/”
另外,如果想把文件放在某一包中,則可以通過(guò)以下方式獲得到文件所在目錄,即先定位到該包的最后一級目錄。
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包中的類(lèi))
五. 屬性文件的讀取:
方法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”
六.編碼轉換問(wèn)題:
ClassLoader的getResource方法使用了utf-8對路徑信息進(jìn)行了編碼,當路徑中存在中文和空格時(shí),他會(huì )對這些字符進(jìn)行轉換,這樣,得到的往往不是我們想要的真實(shí)路徑,在此,調用了URLDecoder的decode方法進(jìn)行解碼,以便得到原始的中文及空格路徑
例如:結果是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) ;
七.總結:
我們在使用相對路徑時(shí),應當使用相對于當前classpath的相對路徑。
ClassLoader類(lèi)的getResource(String name),getResourceAsStream(String name)等方法,使用相對于當前項目的classpath的相對路徑來(lái)查找資源。
讀取屬性文件常用到的ResourceBundle類(lèi)的getBundle(String path)也是如此。
通過(guò)查看ClassLoader類(lèi)及其相關(guān)類(lèi)的源代碼,發(fā)現它實(shí)際上還是使用了URI形式的絕對路徑。通過(guò)得到當前classpath的URI形式的絕對路徑,再去構建相對路徑的URI形式的絕對路徑。
聯(lián)系客服