寫Java程序時會經常從classpath下讀取文件,是時候該整理一下了,並在不斷深入的過程中,陸續補充上。
現在Java project 都以maven項目居多, 比如像下面這樣的一個項目結構:
編譯后的class文件都到了target目錄,如下面的結構:
看代碼:
import java.io.File; import java.net.URL; public class Poem { public static void main(String[] args) { Poem poem = new Poem(); poem.getFile("extObj.txt"); } private void getFile(String fileName) { ClassLoader classLoader = getClass().getClassLoader(); /** getResource()方法會去classpath下找這個文件,獲取到url resource, 得到這個資源后,調用url.getFile獲取到 文件 的絕對路徑 */ URL url = classLoader.getResource(fileName); /** * url.getFile() 得到這個文件的絕對路徑 */ System.out.println(url.getFile()); File file = new File(url.getFile()); System.out.println(file.exists()); } }
通過上面這種方式就可以獲取到這個文件資源。
在一個static method 里可以直接通過類的ClassLoader對象獲取文件資源。
URL url = Poem.class.getClassLoader().getResource("extObj.txt"); File file = new File(url.getFile());
// 直接獲取到輸入流 // fileName 就是resources里的文件名 InputStream in = Poem.class.getClassLoader().getResourceAsStream(fileName);
綜上述,類里的getClassLoader去尋找fileName都是從classpath去找的,畢竟是ClassLoader嘛。
如果一個包里面有一個配置文件,那該怎么獲取呢? 如圖:
第一個dbconfig.properties在類package下,第二個dbconfig.properties在resources目錄下,
那怎么獲取到package下的dbconfig properties文件呢?
here goes code:
package com.getfilefromclasspath; import java.io.IOException; import java.io.InputStream; import java.util.Properties; public class ClassLoaderDemo { public static void main(String[] args) throws IOException { ClassLoaderDemo demo = new ClassLoaderDemo(); demo.loadProperties(); } public void loadProperties() throws IOException { InputStream input = null; try { /** /dbconfig.properties 絕對路徑, 取到的文件是classpath下的 resources/dbconfig.properties 相對路徑 獲取文件流 */ // 獲取到classpath下的文件 input = Class.forName(ClassLoaderDemo.class.getName()).getResourceAsStream("/dbconfig.properties"); // 獲取到package下的文件 // input = Class.forName(ClassLoaderDemo.class.getName()).getResourceAsStream("resources/dbconfig.properties"); } catch (ClassNotFoundException e) { e.printStackTrace(); } printProperties(input); } private void printProperties(InputStream input) throws IOException { Properties properties = new Properties(); properties.load(input); System.out.println(properties.getProperty("username")); } }
不使用Class.forName(), 通過具體對象獲取到Class對象:
//also can be this way: input = this.getClass().getResourceAsStream("resources/dbconfig.properties"); // 對應package下的文件 input = this.getClass().getResourceAsStream("/dbconfig.properties"); // 對應resources下的文件
Class對象還有getResource() 的方法去獲取文件資源,使用規則和上面的一樣。
maven項目還要注意一點,maven 的compiler插件在編譯時是不會將package下的文本文件給編譯到target下的,
下圖是我在用mybatis框架的時候將xml的mapper給放到package編譯后的效果:
這個得在pom.xml加對應的配置(這是在使用mybatis時遇到的坑):
<build> <finalName>java-io</finalName> <resources> <resource> <directory>src/main/java</directory> <includes> <!--properties的配置文件會和編譯后的class文件放在一起--> <include>**/*.properties</include> </includes> </resource> <resource> <!--加載配置的資源--> <directory>src/main/resources</directory> </resource> </resources> <plugins> <plugin> <artifactId>maven-compiler-plugin</artifactId> <configuration> <source>1.8</source> <target>1.8</target> </configuration> </plugin> </plugins> </build>