6

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

更新: className= Version.classPC と Android の両方で。getName()PCでブレークの代わりに使用getSimpleName()しても、Android では動作しません。

4

3 に答える 3

2

NullPointerExceptionメソッド呼び出しclazz.getResource(className)が返されているため、取得しています。nullこれは、で指定されたリソースclassNameが見つからないことを意味します。
リソースファイルをフォルダに入れ、リソースフォルダとして設定します。詳細については、SOでこの回答を確認してください。また、このSOの回答で、リソースフォルダに関する詳細を
確認することもできます。

于 2013-03-12T20:29:47.353 に答える
1

100% 確信はありませんが、このメカニズムが実際の JVM のように Android で機能するとは思いません (dalvik は本質的に JVM ではないため)。パッケージ化された APK では、JAR が存在しなくなったため、リソースへのパスがおそらく間違っていると思います(ここでは間違っている可能性があります)。JAR のクラスが dex ファイル形式に変換され、ファイルに追加されclasses.dexます。したがってfoo.bar.class、クラスパスに存在しないため、ようなリソースを解決できません。

バージョン情報をテキスト (または他のリソース) ファイルに入れて、それを JAR に追加して、それを読み取ってみてください。リソースファイルが適切に追加され、メカニズムで読み取ることができるはずです。

于 2013-03-12T20:44:14.880 に答える
0

バージョン情報をマニフェストファイルに保持し、実行時に抽出したいと思います。もしそうなら、Android へようこそ :-)

直面している問題よりも、いくつかのより良い選択肢があります。

  • PackageManagerとともに使用Context:

    PackageInfo pi = context.getPackageManager().getPackageInfo(
            context.getPackageName(), 0);
    

    次に、 、、 …PackageInfoがあります。これらは で定義されています。packageNameversionCodeversionNameAndroidManifest.xml

  • などのリソースにバージョン情報を保存します。リソースを取得するには、 も必要です。StringIntegerContext

    String appDescription = context.getString(R.string.app_description);
    int majorBuild = context.getResources().getInteger(R.integer.major_build);
    // ...
    

そっちの方がいいと思います。

于 2013-03-13T01:13:15.753 に答える