我已经创建了一个独立于平台的库,我想在J2SE和Android项目中使用它。在这个库中,我有一个Version
类,它从清单中加载它的版本细节。在PC上运行良好,在Android上我得到了NullPointerException
,我无法弄清楚原因。
这是我的班级:
public class Version {
private static int APPCODE = 12354;
private static int MAJOR;
private static int MINOR;
private static char RELEASE;
private static int BUILD;
private static int PROTOCOL;
static {
try {
Class clazz = Version.class;
String className = clazz.getSimpleName() + ".class";
String classPath = clazz.getResource(className).toString(); //NullPointerException
if (classPath.startsWith("jar")) {
String manifestPath = classPath.substring(0,
classPath.lastIndexOf("!") + 1) + "/META-INF/MANIFEST.MF";
Manifest manifest = new Manifest(new URL(manifestPath).openStream());
Attributes attr = manifest.getMainAttributes();
//APPCODE = Integer.parseInt(attr.getValue("APPCODE"));
MAJOR = Integer.parseInt(attr.getValue("MAJOR"));
MINOR = Integer.parseInt(attr.getValue("MINOR"));
RELEASE = attr.getValue("RELEASE").charAt(0);
BUILD = Integer.parseInt(attr.getValue("BUILD"));
PROTOCOL = Integer.parseInt(attr.getValue("PROTOCOL"));
} else {
System.err.println("Couldn't find manifest, reverting to test mode");
MAJOR = 0;
MINOR = 0;
RELEASE = 'n';
BUILD = 0;
//APPCODE = 12354;
}
} catch (IOException e) {
System.err.println("Failed to load manifest file. " + e);
MAJOR = 0;
MINOR = 0;
RELEASE = '0';
BUILD = 0;
//APPCODE = 12354;
PROTOCOL = 0;
} catch (NumberFormatException e) {
System.err.println("Failed to load manifest file. " + e);
MAJOR = 0;
MINOR = 0;
RELEASE = '0';
BUILD = 0;
//APPCODE = 12354;
PROTOCOL = 0;
}
}
public static int getProtocol() {
return PROTOCOL;
}
public static int getAppCode() {
return APPCODE;
}
public static int getBuildNumber() {
return BUILD;
}
public static int getMajor() {
return MAJOR;
}
public static int getMinor() {
return MINOR;
}
public static char getRelease() {
return RELEASE;
}
}
(请原谅System.err.println()
行,这些行用于在PC上进行调试。
为什么这在PC上运行良好但在Android上运行不正常?
完整堆栈跟踪:
03-12 22:13:11.687: E/AndroidRuntime(11780): FATAL EXCEPTION: main
03-12 22:13:11.687: E/AndroidRuntime(11780): java.lang.ExceptionInInitializerError
03-12 22:13:11.687: E/AndroidRuntime(11780): at com.logandam.wififileshare.android.MainActivity.onCreate(MainActivity.java:24)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.Activity.performCreate(Activity.java:4465)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1052)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1932)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1993)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.ActivityThread.access$600(ActivityThread.java:127)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1159)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.os.Handler.dispatchMessage(Handler.java:99)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.os.Looper.loop(Looper.java:137)
03-12 22:13:11.687: E/AndroidRuntime(11780): at android.app.ActivityThread.main(ActivityThread.java:4507)
03-12 22:13:11.687: E/AndroidRuntime(11780): at java.lang.reflect.Method.invokeNative(Native Method)
03-12 22:13:11.687: E/AndroidRuntime(11780): at java.lang.reflect.Method.invoke(Method.java:511)
03-12 22:13:11.687: E/AndroidRuntime(11780): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:790)
03-12 22:13:11.687: E/AndroidRuntime(11780): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:557)
03-12 22:13:11.687: E/AndroidRuntime(11780): at dalvik.system.NativeStart.main(Native Method)
03-12 22:13:11.687: E/AndroidRuntime(11780): Caused by: java.lang.NullPointerException
03-12 22:13:11.687: E/AndroidRuntime(11780): at com.logandam.wififileshare.net.Version.<clinit>(Version.java:30)
03-12 22:13:11.687: E/AndroidRuntime(11780): ... 15 more
在PC和Android上更新:className
= Version.class
。在PC上使用getName()
代替getSimpleName()
,但仍无法在Android上运行。
答案 0 :(得分:2)
您收到NullPointerException
因为方法调用clazz.getResource(className)
正在返回null
,这意味着无法找到className
指定的资源。
将资源文件放在一个文件夹中,并将其设置为资源文件夹。有关详细信息,请查看this answer on SO
您还可以查看this SO answer有关资源文件夹的详细信息。
答案 1 :(得分:1)
我不是100%肯定,但我不认为这种机制在Android下可以像在真正的JVM中一样工作(因为dalvik本质上不是JVM)。我认为(我可能在这里错了),在打包的APK中,JAR不再存在,因此您的资源路径可能是错误的。您的JAR的类将转换为dex文件格式和added to the classes.dex
file。因此,您无法解析foo.bar.class
之类的资源,因为它在类路径中不存在。
尝试将版本信息放入文本(或其他资源)文件中,并将其添加到JAR中并读取。应该正确添加资源文件,您应该能够使用该机制读取它。
答案 2 :(得分:0)
我认为您希望将版本信息保留在清单文件中并在运行时提取它。如果是这样,欢迎使用Android: - )
您有几个更好的选择,而不是您遇到问题的选项。
PackageInfo pi = context.getPackageManager().getPackageInfo(
context.getPackageName(), 0);
然后使用PackageInfo
,您有packageName
,versionCode
,versionName
...这些内容已在AndroidManifest.xml
中定义。
将您的版本信息存储在resources中,例如String
,Integer
...您还需要Context
来获取资源:
String appDescription = context.getString(R.string.app_description);
int majorBuild = context.getResources().getInteger(R.integer.major_build);
// ...
我认为那些更好。