无法在所有计算机上访问JAR中的资源

时间:2011-12-23 23:48:57

标签: java file-io resources jar

我正在编写一个应用程序(特别是Bukkit Minecraft服务器的插件)。这样做需要我从应用程序的JAR访问.properties文件。这是我遇到一个奇怪问题的地方。当我在我的开发PC上测试程序时,它运行得很好。 .properties文件被加载,一切都很好。但是,在我测试的其他计算机上,我尝试启动应用程序,但无法加载属性,InputStreamnull。这是我加载文件的方法:

public class Points {
    private HashMap<String, MessageFormat> messages;

    public Points() {
         buildMessages();
    }

public static void buildMessages() {
        Properties messageProps = new Properties();
        InputStream in = Points.class.getResourceAsStream("resources/messages.properties");
        messages = new HashMap<String, MessageFormat>();
        Enumeration en;
        try {
            messageProps.load(in);
        } catch(IOException ex) {
            System.err.println("Couldn't read message properties file!");
            return;
        } catch(NullPointerException ex) {
            System.err.println("Couldn't read message properties file!");
            if(in == null)
                System.out.println("IOStream null");
            return;
        }
        en = messageProps.propertyNames();
        while(en.hasMoreElements()) {
            String key = (String)en.nextElement();
            String prop = messageProps.getProperty(key);
            MessageFormat form = new MessageFormat(prop.replaceAll("&", 
                "\u00a7").replaceAll("`", ""));
            messages.put(key, form);
        }
    }
}

我省略了一些不相关的代码,但这就是它的要点。 JAR的结构如下:

   com/
       pvminecraft/
           points/
               Points.java <-- The class where the file is loaded
               resources/
                   messages.properties <-- The file being loaded

在我的电脑上,文件是从resources/messages.properties加载的,但在另一个文件中,InputStream为空,并且运行catch的{​​{1}}块。可能导致问题的原因是什么,我该如何解决?感谢。

更新:即使使用完整路径(NullPointerException),同样的问题仍然存在。

更新2:以下是完整的堆栈跟踪:

/com/pvminecraft/points/resources/messages.properties

所有java.lang.NullPointerException at java.util.Properties$LineReader.readLine(Properties.java:435) at java.util.Properties.load0(Properties.java:354) at java.util.Properties.load(Properties.java:342) at com.pvminecraft.points.Points.buildMessages(Unknown Source) at com.pvminecraft.points.Points.onEnable(Unknown Source) at org.bukkit.plugin.java.JavaPlugin.setEnabled(JavaPlugin.java:188) at org.bukkit.plugin.java.JavaPluginLoader.enablePlugin(JavaPluginLoader.java:968) at org.bukkit.plugin.SimplePluginManager.enablePlugin(SimplePluginManager.java:280) at org.bukkit.craftbukkit.CraftServer.loadPlugin(CraftServer.java:186) at org.bukkit.craftbukkit.CraftServer.enablePlugins(CraftServer.java:169) at org.bukkit.craftbukkit.CraftServer.reload(CraftServer.java:436) at org.bukkit.Bukkit.reload(Bukkit.java:187) at org.bukkit.command.defaults.ReloadCommand.execute(ReloadCommand.java:22) at org.bukkit.command.SimpleCommandMap.dispatch(SimpleCommandMap.java:165) at org.bukkit.craftbukkit.CraftServer.dispatchCommand(CraftServer.java:378) at org.bukkit.craftbukkit.CraftServer.dispatchCommand(CraftServer.java:374) at net.minecraft.server.MinecraftServer.b(MinecraftServer.java:564) at net.minecraft.server.MinecraftServer.w(MinecraftServer.java:541) at net.minecraft.server.MinecraftServer.run(MinecraftServer.java:425) at net.minecraft.server.ThreadServerApplication.run(SourceFile:457) org.bukkit内容都是服务器。 .properties文件加载在org.craftbukkit方法中,由buildMessages的{​​{1}}方法调用。

更新3:在全新安装的Arch Linux上,正确加载了消息属性文件,一切正常。远程服务器是Ubuntu Linux,我的开发人员是Arch。

更新4:好吧,这是一种解决方案。这似乎是一个局部问题。我这样说是因为我设法访问了另外两台计算机,并且程序在两者上运行正常。虽然这很烦人,但我的代码或构建脚本似乎没有任何问题。我仍然想知道什么是错的,但它不再紧迫了。我会继续研究这个。谢谢大家。

4 个答案:

答案 0 :(得分:2)

看起来像是不同的Java类加载器及其搜索路径之间的细微差别。 在进入这些细节之前;为什么不尝试这个jar文件中的完整路径? (即这样的事情:

Points.class.getResourceAsStream("com/pvminecraft/points/resources/messages.properties");

答案 1 :(得分:2)

Point.class.getClassLoader().getResourceAsStream("com/pvminecraft/points/resources/messages.properties");

尝试没有第一个'/',它应该可以在任何运行JVM的地方工作。

如果这不起作用,请尝试将文件放在JAR文件的ROOT中并再次尝试。

如果仍然无效,请尝试使用此方法:

public static byte[] getFile(File zip, String fileName) throws FileNotFoundException, ZipException, IOException {
        String filename = fileName;

        if (!zip.exists()) {
            throw new FileNotFoundException(zip.getName());
        }
        while (filename.charAt(0) == '/' || filename.charAt(0) == '\\') {
            filename = filename.substring(1);
        }

        if (filename.contains("\\")) {
            filename = filename.replace("\\", "/");
        }

        ZipFile zipFile = new ZipFile(zip);
        Enumeration entries = zipFile.entries();

        ByteArrayOutputStream output;
        byte[] result = null;

        while (entries.hasMoreElements()) {
            ZipEntry entry = (ZipEntry) entries.nextElement();

            if (entry.getName().equalsIgnoreCase(filename)) {
                FileUtils.copyInputStream(zipFile.getInputStream(entry), output = new ByteArrayOutputStream());
                result = output.toByteArray();
                zipFile.close();
                output.close();
                return result;
            }
        }

        zipFile.close();
        throw new FileNotFoundException(filename);
    }

你需要这个

public static void copyInputStream(InputStream in, OutputStream out) throws IOException {
    byte[] buffer = new byte[1024];
    int len;
    while (((len = in.read(buffer)) >= 0)) {
        out.write(buffer, 0, len);
    }
    out.flush();
}

获取正在运行的jar的路径

 String currentJar = "";
                                        // Get current jar path. Since user may rename this file, we need to do this way
              try {
                   currentJar = (Points.class.getProtectionDomain().getCodeSource().getLocation().toURI().getPath());
                   if (currentJar.startsWith("/")) currentJar = currentJar.substring(1);
                   } catch (URISyntaxException ex) {
                   }

第一个'/'我真的不记得它出现的原因,但确实如此,所以你必须删除它:

最后调用方法:getFile(currentJar, "PATH_TO_PROPERTIES_FILE");

您将拥有一个可以使用的字节数组。只需将其作为ByteArrayInputStream,您的问题就应该解决了。


该代码是我创建的util类的一部分,这就是为什么不必要的读取字节数组,但是,你可以改变它直接使用InputStream到Properties.load()方法。

ZIP util类的链接

http://all-inhonmodman.svn.sourceforge.net/viewvc/all-inhonmodman/ModManager/src/modmanager/utility/ZIP.java?revision=292&content-type=text%2Fplain

FileUtils util类的链接

http://all-inhonmodman.svn.sourceforge.net/viewvc/all-inhonmodman/ModManager/src/modmanager/utility/FileUtils.java?revision=294&content-type=text%2Fplain

答案 2 :(得分:0)

您可能还需要确保构建脚本(Ant,Maven)或IDE没有从生成的JAR文件中删除/重定位message.properties(因为它不是.class)。您可以使用7zip或WinZip等工具检查JAR内容。

答案 3 :(得分:0)

由于您正在加载属于同一个类(Point)的资源,因此您无需使用绝对路径来加载它。

服务器是否使用任何类型的缓存来加载插件?这可能是由于类路径中存在旧版本的插件jar。要验证服务器是否真的加载了正确版本的jar文件,您可以尝试部署一个版本的插件,该插件会将某些内容记录到控制台,并查看是否发生了某些事情(如果消息确实已记录)。

另外,我不知道如何在服务器中组织类加载器层次结构,但是您可以尝试从当前线程类加载器加载资源(通常是根父类加载器,它将在每个加载器中查找资源)其他子类加载器)。你必须使用绝对路径。

ClassLoader rootCL = Thread.currentThread().getContextClassLoader();
InputStream resource = rootCL.getResourceAsStream(
        "/com/pvminecraft/points/resources/messages.properties");

选中此question以详细了解不同的类加载器。