ClassLoader
獲取路徑的時候出錯了。
/** * @author mazhq * @Title: TestMain * @ProjectName: zeus * @Description: TODO * @date 2019/3/5 16:10 */ public class TestMain { public static void main(String[] args) { String path = TestMain.class.getClassLoader().getResource("1.xml").getPath(); System.out.println(path); } /** * 輸出: * */D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml */ }
可是在將SpringBoot打包放到Linux服務器啓動打印的目錄爲java
/data/zeus/service-hi-1.0.0-SNAPSHOT.war!/WEB-INF/classes!/1.xml
能夠看到在Linux中沒法直接訪問未經解壓的文件,因此就會找不到文件。數據庫
1. 經過ClassLoader
的getResourceAsStream()
方法獲取其流,就可以獲取到。服務器
讀取jar裏面的文件,咱們只能用流去讀取,不能用Fileapp
public class TestMain { public static void main(String[] args) { try { List<String> content = IOUtils.readLines(TestMain.class.getClassLoader().getResourceAsStream("1.xml"), "UTF-8"); } catch (IOException e) { e.printStackTrace(); } } }
2. 採用絕對路徑將文件放到服務器某個路徑,在application.properties中配置路徑讀取。this
3. 不推薦:將內容放到數據庫中。.net
一般在開發過程當中會碰到讀取配置文件的問題,通常有兩種方式進行讀取。一種是Class.getResource(String path)
,一種是ClassLoader.getResource(String path)
,這兩種雖然都能讀取文件,可是在path
的填寫上有一點點的不一樣。code
/
開頭:則是從ClassPath根下獲取/
開頭:默認是今後類所在的包下取資源下面有個例子xml
public class TestMain { public static void main(String[] args) { System.out.println(TestMain.class.getResource("/")); System.out.println(TestMain.class.getResource("")); } /** * 輸出: * * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/ * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/com/mazhq/servicehi/ */ }
那麼讀取在
resource下的1.xml,就以下的獲取方法
public class TestMain { public static void main(String[] args) { System.out.println(TestMain.class.getResource("/1.xml")); System.out.println(TestMain.class.getResource("../../../1.xml")); } /** * 輸出: * * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml */ }
ClassLoader.getResource
的path中不能以/
開頭,path是默認是從根目錄下進行讀取的blog
代碼以下:ip
public class TestMain { public static void main(String[] args) { System.out.println(TestMain.class.getClassLoader().getResource("")); System.out.println(TestMain.class.getClassLoader().getResource("/")); } /** * 輸出: * * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/ * null */ }
從上面例子咱們能夠看到
TestMain.class.getClassLoader().getResource("")=TestMain.class.getResource("/")
其實查看Class.getResource
中能夠看到
public java.net.URL getResource(String name) { name = resolveName(name); ClassLoader cl = getClassLoader0(); if (cl==null) { // A system class. return ClassLoader.getSystemResource(name); } return cl.getResource(name); }
他最後調用的仍是ClassLoader.getResource
這個方法,那麼爲何會有path
的差異呢,由於其resolveName
方法中對傳的/
進行了解析,解析爲了空字符串。
resolveName 方法實現以下:
private String resolveName(String name) { if (name == null) { return name; } if (!name.startsWith("/")) { Class<?> c = this; while (c.isArray()) { c = c.getComponentType(); } String baseName = c.getName(); int index = baseName.lastIndexOf('.'); if (index != -1) { name = baseName.substring(0, index).replace('.', '/') +"/"+name; } } else { name = name.substring(1); } return name; }
//傳入 "/" 返回 ""
最後:你們用的時候注意一下這些問題,避免在這個上面耽誤時間。