Java讀取resources中資源文件路徑以及jar中文件無法讀取的解決
Java讀取resources中資源文件路徑以及jar中文件無法讀取的解決
問題描述
現(xiàn)象
作為一個剛開始學習java的新人,很多東西都是摸著石頭過河,踩坑是常有的事。這不,今天我將maven管理的一個spring boot的WebAPP部署到服務(wù)器上,運行直接報錯!納尼!??!本地跑得好好的,一到服務(wù)器就出問題,關(guān)鍵是日志文件中的日志不全,無法馬上定位到問題。好吧,一步一步排除問題吧!
定位
是不是windows與linux的區(qū)別?不是,我在windows上跑了一下打包后的代碼,也出問題了,打包前沒問題,打包后出問題了,包有毒!然后我開放了日志,一步一步調(diào)試(蛋疼?。罱K發(fā)現(xiàn)配置文件沒有加載,路徑出了問題。。。
前言
工程文件結(jié)構(gòu)如下所示,目標是讀取resources/python/kafka_producer.py文件
1、本地運行讀取資源文件
采用getResource進行讀取:
URL urlPath = this.getClass().getResource("/python/kafka_producer.py"); String execStr = String.format("python %s", urlPath.getPath().substring(1));
它是在target文件中讀取,這時文件是我們熟悉的文件。正常讀取,運行。
2、讀取jar包中的文件信息
InputStream is=this.getClass().getResourceAsStream("/python/kafka_producer.py"); BufferedReader br1=new BufferedReader(new InputStreamReader(is)); String s1=""; while((s1=br1.readLine())!=null) System.out.println(s1);
如果你需要運行腳本文件,這時是不能直接通過路徑獲取的,具體可以看博客點擊。你需要重新將流寫入文件中,在運行,當然,也可以打war包,不用jar包。如果讀取配置文件有一下兩種方式:
InputStream in = this.getClass().getResourceAsStream("/properties/basecom.properties"); Properties properties = new Properties(); properties.load(in); properties.getProperty("property_name")
或者
InputStream xmlFile = this.getClass().getResourceAsStream("/jdbcType.xml"); Document document = xmlReader.read(xmlFile); Element xmlRoot = document.getRootElement(); Element childElement = xmlRoot.element(dbType); List<Element> childElements = childElement.elements(); for (Element child : childElements) { }
聊聊Java項目讀取resources資源文件路徑那點事
在Java程序中讀取resources資源下的文件,由于對Java結(jié)構(gòu)了解不透徹,遇到很多坑。
正常在Java工程中讀取某路徑下的文件時,可以采用絕對路徑和相對路徑,絕對路徑?jīng)]什么好說的,相對路徑,即相對于當前類的路徑。在本地工程和服務(wù)器中讀取文件的方式有所不同,以下圖配置文件為例:
1、本地讀取資源文件
Java類中需要讀取properties中的配置文件,可以采用文件(File)方式進行讀?。?/p>
File file = new File("src/main/resources/properties/test.properties"); InputStream in = new FileInputStream(file);
注意:當在IDEA中運行(不部署在服務(wù)器上),可以讀取到該文件;
原因:JavaWeb項目部署服務(wù)器中,會將項目打包成Jar包或者war包,此時就不會存在 src/main/resources 目錄,JVM會在編譯項目時,主動將 java文件編譯成 class文件 和 resources 下的靜態(tài)文件放在 target/classes目錄下;
理解:Java文件只有編譯成 class文件才會被JVM執(zhí)行,本地執(zhí)行時會,當前項目即為Java進程的工作空間,雖然class文件在target/classes目錄下,但是target/classes不是class文件運行的目錄,只是存放的目錄,運行目錄還是在IDEA的模塊下,所以運行時會找到 src/main/resources 資源文件!
2、服務(wù)器(Tomcat)讀取資源文件
當工程部署到Tomcat中時,按照上邊方式,則會拋出異常:FileNotFoundException。
原因:Java工程打包部署到Tomcat中時,properties的路徑變到頂層(classes下),這是由Maven工程結(jié)構(gòu)決定的。
由Maven構(gòu)建的web工程,主代碼放在src/main/java路徑下,資源放在src/main/resources路徑下,當構(gòu)建jar包 或 war包時,JVM虛擬機會自動編譯java文件為class文件存放在 target/classes目錄下,resource資源下的文件會原封不動的拷貝一份到 target/classes 目錄下:
方式一:此時讀取資源文件時
采用流(Stream)的方式讀取,并通過JDK中Properties類加載,可以方便的獲取到配置文件中的信息:
InputStream in = this.getClass().getResourceAsStream("/properties/test.properties"); Properties properties = new Properties(); properties.load(in); properties.getProperty("name");
重點理解:class.getResourceAStream() 與 class.getClassLoader().getResorceAsStream() 的區(qū)別
1) InputStream inStream = PropertiesTest.class.getResourceAsStream("test.properties"); 2) inStream = PropertiesTest.class.getResourceAsStream("/com/test/demo/test.properties") 3) inStream = PropertiesTest.class.getClassLoader().getResourceAsStream("com/test/demo/test.properties");
1)第一種和第二種方式采用 Class 對象去加載,第三種方式采用 ClassLoader 對象去加載資源文件,之所以 Class 可以加載資源文件,是因為 Class 類封裝的 ClassLoader 的 getResourceAsStream() 方法,從 Class 類中的源碼可以看出:
public InputStream getResourceAsStream(String name) { name = resolveName(name); ClassLoader cl = getClassLoader0(); if (cl==null) { // A system class. return ClassLoader.getSystemResourceAsStream(name); } return cl.getResourceAsStream(name); }
理由:??之所以這樣做無疑還是方便客戶端的調(diào)用,省的每次獲取ClassLoader才能加載資源文件的麻煩!
2).class 是獲取當前類的 class 對象,getClassLoader()是獲取當前的類加載器,什么是類加載器?簡單點說,就是用來加載java類的,類加載器就是負責把class文件加載進內(nèi)存中,并創(chuàng)建一個java.lang.Class類的一個實例,也就是class對象,并且每個類的類加載器都不相同,getResourceAsStream(path)是用來獲取資源的,因為這是ClassLoader(類加載器)獲取資源,而類加載器默認是從 classPath 下獲取資源的,因為這下面有class文件。
所以這段代碼總的意思是通過類加載器在 classPath 目錄下獲取資源,并且是以流的形式。我們知道在Java中所有的類都是通過加載器加載到虛擬機中的,而且類加載器之間存在父子關(guān)系,就是子知道父,父不知道子,這樣不同的子加載的類型之間是無法訪問的(雖然它們都被放在方法區(qū)中),所以在這里通過當前類的加載器來加載資源也就是保證是和類類型是同一個加載器加載的。
(3)class.getClassLoader().getResourceAsStream() 和 class.getResouceAsStream() 的區(qū)別
a)class.getClassLoader().getResourceAsStream(Stringname)默認從classpath中找文件(文件放在resources目錄下),name不能帶"/",否則會拋空指針。采用相對路徑, "/"就相當于當前進程的根目錄,即項目根目錄;
inStream = PropertiesTest.class.getClassLoader().getResourceAsStream("com/test/demo/test.properties");
b)class.getResourceAsStream(String name) 是采用絕對路徑,絕對路徑是相對于 classpath 根目錄的路徑,"/" 就代表著 classpath,所以 name 屬性需要前面加上 "/";
inStream = PropertiesTest.class.getResourceAsStream("/com/test/demo/test.properties")
方式二:采用Spring注解
如果工程中使用Spring,可以通過注解的方式獲取配置信息,但需要將配置文件放到Spring配置文件中掃描后,才能將配置信息放入上下文。
<context:component-scan base-package="com.xxxx.service"/> <context:property-placeholder location="classpath:properties/xxx.properties" ignore-unresolvable="true"/>
然后在程序中可以使用 @Value進行獲取properties文件中的屬性值,如下:
@Value("${xxxt.server}") private static String serverUrl;
方式三:采用Spring配置
也可以在Spring配置文件中讀取屬性值,賦予類成員變量
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd"> <bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location" value="classpath:properties/xxx.properties"/> </bean> <bean id="service" class="com.xxxx.service.ServiceImpl"> <property name="serverUrl" value="${xxxt.server}" /> </bean> </beans>
重點:SpringBoot項目啟動后,動態(tài)的讀取類路徑下文件數(shù)據(jù)
InputStream inputStream = EncryptUtil.class.getResourceAsStream("/HelloServiceEncryptFile.txt"); BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream)); String line = reader.readLine(); // 獲取類路徑下的文件路徑 File path = new File(ResourceUtils.getURL("classpath:").getPath()); if (!path.exists()) { path = new File(""); } log.info("path = {}", path.getAbsolutePath()); File upload = new File(path.getAbsolutePath(), "com/study/service"); if (!upload.exists()) { upload.mkdirs(); } FileOutputStream fos = new FileOutputStream(upload.getAbsolutePath() + File.separator + "hello.txt"); IoUtil.copy(inputStream, fos); fos.close(); inputStream.close();
注意:此時我想讀取 jar 包中根路徑下的 HelloServiceEncryptFile.txt 文件,然后重新寫入到根路徑下的 com.study/service 路徑下!
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java Comparable 和 Comparator 的詳解及區(qū)別
這篇文章主要介紹了Java Comparable 和 Comparator 的詳解及區(qū)別的相關(guān)資料,Comparable 自然排序和Comparator 定制排序的實例,需要的朋友可以參考下2016-12-12Java的分支結(jié)構(gòu)與循環(huán)你知道多少
這篇文章主要為大家詳細介紹了Java的分支結(jié)構(gòu)與循環(huán),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-02-02