我正在编写一个应用程序(特别是Bukkit Minecraft服务器的插件)。这样做需要我从应用程序的JAR访问.properties文件。这是我遇到一个奇怪问题的地方。当我在我的开发PC上测试程序时,它运行得很好。 .properties文件被加载,一切都很好。但是,在我测试的其他计算机上,我尝试启动应用程序,但无法加载属性,InputStream
为null
。这是我加载文件的方法:
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:好吧,这是一种解决方案。这似乎是一个局部问题。我这样说是因为我设法访问了另外两台计算机,并且程序在两者上运行正常。虽然这很烦人,但我的代码或构建脚本似乎没有任何问题。我仍然想知道什么是错的,但它不再紧迫了。我会继续研究这个。谢谢大家。
答案 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类的链接
FileUtils util类的链接
答案 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以详细了解不同的类加载器。