Java 程序员必学:classpath 文件读取技巧

Java 程序员必学:classpath 文件读取技巧

  1. 后端开发与架构
  2. 2016.03.14
  3. 4 min read

写 Java 程序时会经常从 classpath 下读取文件, 是时候该整理一下了, 并在不断深入的过程中, 陆续补充上.

现在 Java project 都以 maven 项目居多, 比如像下面这样的一个项目结构:

20241229154732_YWDwQbhz.webp

编译后的 class 文件都到了 target 目录, 如下面的结构:

20241229154732_BvAjGpSn.webp

看代码:

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 嘛.

如果一个包里面有一个配置文件, 那该怎么获取呢? 如图:

20241229154732_unL2GHSm.webp

第一个 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 编译后的效果:

20241229154732_GlI663O1.webp

这个得在 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>
Java 学习笔记