1. 程式人生 > >java不同路徑讀取檔案的方式--Thread.currentThread().getContextClassLoader(), Test.class.getClassLoader()等

java不同路徑讀取檔案的方式--Thread.currentThread().getContextClassLoader(), Test.class.getClassLoader()等

Java code

?

1

2

3

4

5

6

7

8

9

10

11

public class Test {

 

    public static void main(String[] args) {

         

        // 此時三個ClassLoader是同一個物件

        System.out.println(Thread.currentThread().getContextClassLoader()); // 當前執行緒的類載入器

        System.out.println(Test.class.getClassLoader()); // 當前類的類載入器

        

System.out.println(ClassLoader.getSystemClassLoader()); // 系統初始的類載入器

         

    }

}


如果樓主瞭解過openfire應該對ClassLoader有比較深的理解。
打個簡單的比方,你一個WEB程式,釋出到Tomcat裡面執行。
首先是執行Tomcat org.apache.catalina.startup.Bootstrap類,這時候的類載入器是ClassLoader.getSystemClassLoader()。
而我們後面的WEB程式,裡面的jar、resources都是由Tomcat內部來載入的,所以你在程式碼中動態載入jar、資原始檔的時候,首先應該是使用Thread.currentThread().getContextClassLoader()。如果你使用Test.class.getClassLoader(),可能會導致和當前執行緒所執行的類載入器不一致(因為Java天生的多執行緒)。
Test.class.getClassLoader()一般用在getResource,因為你想要獲取某個資原始檔的時候,這個資原始檔的位置是相對固定的。

java的類載入機制(jvm規範)是委託模型,簡單的說,如果一個類載入器想要載入一個類,首先它會委託給它的parent去載入,如果它的所有parent都沒有成功的載入那麼它才會自己親自來,有點兒像兒子使喚老子的感覺。。jvm也拼爹啊,,,,,
在jvm中預設有三類loaer,bootstrap,ext,app,其中boot最大是爺爺,app最小是孫子,ext中間是爹。
它們有許可權訪問的classpath也不一樣,boot是jdk或jre下面的lib目錄,ext是jdk或jre的ext目錄,而app是由使用者指定的路徑,比如用-cp引數指定的目錄或jar。他們沒有權力訪問其他人的classpath,這樣問題就來鳥,,,,可能有人會問狗司令大人閒得蛋疼啊,搞這麼複雜,據說是為了安全考慮,避免使用者的噁心意程式碼侵蝕jvm,,問題就是當bootstrap或ext想要載入使用者指定classpath中的類就會失敗,因為這倆貨沒有許可權訪問團app路徑中的類的,,所以就搞了這麼一個不倫不類的contextloader。。。。

 

 

 

讀取檔案的寫法,相對路徑
在當前的目錄結構中讀取test.txt的有四種寫法

簡單粗暴的 File file = new File(“src/test.txt”)
使用類的相對路徑
使用當前執行緒的類載入器
讀取web工程下的檔案 使用getRealPath()讀取


 File file = new File("src/test.txt");
 File file = new File(TestRelativePath.class.getResource("/test.txt").getFile());
  File file = new File(Thread.currentThread().getContextClassLoader().getResource("test.txt").getFile());
   File file = new File(getServletContext().getRealPath("/WEB-INF/classes/test.txt"));

下面我來一一介紹:

簡單粗暴的 File file = new File(“src/test.txt”);
  @Test
    /**
     * 這種方法 “” 空代表的是 這個Java專案 TestSomeTechnology  由於實際專案在打包後沒有src目錄 所以這種方法不常用
     */
    public  void  testMethod1() throws IOException{
        File file = new File("src/test.txt");
        BufferedReader br = new BufferedReader(new FileReader(file));
        String len = null;
        while ((len=br.readLine())!=null){
            System.out.println(len);
        }
    }
使用類的相對路徑
TestRelativePath.class.getResource(“/test.txt”).getFile()

 @Test
    /**
     * 使用類的相對路徑
     * 這種方法 “/” 代表的是bin  src資料夾和resources 資料夾下的的東西都會被載入到bin下面 因為這兩個檔案被配置為了source
     */
    public  void  testMethod2() throws IOException{

        File file = new File(TestRelativePath.class.getResource("/test.txt").getFile());
        BufferedReader br = new BufferedReader(new FileReader(file));
        String len = null;
        while ((len=br.readLine())!=null){
            System.out.println(len);
        }
    }
使用當前執行緒的類載入器
Thread.currentThread().getContextClassLoader().getResource(“test.txt”).getFile()

 @Test
    /**
     * 這種是通過當前執行緒的類載入器
     * 這種方法 “ ” 空代表的是bin  於是就直接填寫test 資料夾下的的東西都會被載入到bin下面 因為這兩個檔案被配置為了source
     */
    public  void  testMethod3() throws IOException{

        File file = new File(Thread.currentThread().getContextClassLoader().getResource("test.txt").getFile());
        BufferedReader br = new BufferedReader(new FileReader(file));
        String len = null;
        if ((len=br.readLine())!=null){
            System.out.println(len);
        }
    }

讀取web工程下的檔案 使用getRealPath()讀取
目錄如下 


讀取 index.jsp

@WebServlet(name = "TestServlet",urlPatterns = "/TestServlet")
public class TestServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request,response);
    }

    /**
     *  web工程的根目錄是 webRoot 使用 “/” 代表webroot webroot下面有index.jsp檔案
     */
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        File file = new File(getServletContext().getRealPath("/index.jsp"));
        BufferedReader br = new BufferedReader(new FileReader(file));
        String len = null;
        while ((len=br.readLine())!=null){
            System.out.println(len);
        }
    }
}
讀取 test.txt檔案 
不過如果想讀取test.txt 的話 我們可用用上面的方式

 File file = new File(Thread.currentThread().getContextClassLoader().getResource("test.txt").getFile());
1
也可以使用 getRealPath() 
不過由於是以 webroot為根目錄 我們需要從classes裡面讀

 File file = new File(getServletContext().getRealPath("/WEB-INF/classes/test.txt"));
1
getResourceAsStream()方法詳解
getResourceAsStream()用法與getResouce()方法一樣的,用getResource()取得File檔案後,再new FileInputStream(file) 與 getResourceAsStream() 的效果一樣。。

給出示例 
兩個程式碼效果一樣

        InputStream inputStream1 = new FileInputStream(new File(Thread.currentThread().getContextClassLoader().getResource("test.txt").getFile()));
//=============================
        InputStream inputStream2 = Thread.currentThread().getContextClassLoader().getResourceAsStream("test.txt");
--------------------- 
作者:guxiangfly 
來源:CSDN 
原文:https://blog.csdn.net/my__sun_/article/details/74450241 
版權宣告:本文為博主原創文章,轉載請附上博文連結!