我有一个非常简单的项目可以编译,但无法在模拟器上启动。问题在于这种方法:
private void bar(@Some String a) {} // java.lang.VerifyError
如果删除了注释,则可以避免此问题
private void bar(String a) {} // OK
或方法可见性已更改:
void bar(@Some String a) {} // OK
public void bar(@Some String a) {} // OK
protected void bar(@Some String a) {} // OK
知道原始方法有什么问题吗?这是一个dalvik错误,还是?
如果有人想试验代码,那么它就是:
Test.java:
public class Test {
private void bar(@Some String a) {}
public void foo() {
bar(null);
}
}
Some.java:
public @interface Some {}
MainActivity.java:
public class MainActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
new Test().foo();
}
}
堆栈追踪:
ERROR/dalvikvm(1358): Could not find method com.my.Test.bar, referenced from method com.my.Test.foo
WARN/dalvikvm(1358): VFY: unable to resolve direct method 11: Lcom/my/Test;.bar (Ljava/lang/String;)V
WARN/dalvikvm(1358): VFY: rejecting opcode 0x70 at 0x0001
WARN/dalvikvm(1358): VFY: rejected Lcom/my/Test;.foo ()V
WARN/dalvikvm(1358): Verifier rejected class Lcom/my/Test;
DEBUG/AndroidRuntime(1358): Shutting down VM
WARN/dalvikvm(1358): threadid=3: thread exiting with uncaught exception (group=0x4000fe70)
ERROR/AndroidRuntime(1358): Uncaught handler: thread main exiting due to uncaught exception
ERROR/AndroidRuntime(1358): java.lang.VerifyError: com.my.Test
ERROR/AndroidRuntime(1358): at com.my.MainActivity.onCreate(MainActivity.java:13)
ERROR/AndroidRuntime(1358): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1123)
ERROR/AndroidRuntime(1358): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2231)
ERROR/AndroidRuntime(1358): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2284)
ERROR/AndroidRuntime(1358): at android.app.ActivityThread.access$1800(ActivityThread.java:112)
ERROR/AndroidRuntime(1358): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1692)
ERROR/AndroidRuntime(1358): at android.os.Handler.dispatchMessage(Handler.java:99)
ERROR/AndroidRuntime(1358): at android.os.Looper.loop(Looper.java:123)
ERROR/AndroidRuntime(1358): at android.app.ActivityThread.main(ActivityThread.java:3948)
ERROR/AndroidRuntime(1358): at java.lang.reflect.Method.invokeNative(Native Method)
ERROR/AndroidRuntime(1358): at java.lang.reflect.Method.invoke(Method.java:521)
ERROR/AndroidRuntime(1358): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:782)
ERROR/AndroidRuntime(1358): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:540)
ERROR/AndroidRuntime(1358): at dalvik.system.NativeStart.main(Native Method)
答案 0 :(得分:3)
这实际上是Eclipse 3.5编译器(Bug 289576)的一个错误,它使用带注释的参数更改方法的private
修饰符,以便该方法成为“包私有”方法。所以你的:
private void bar(@Some String a) {…}
.class 文件中的变为:
void bar(@Some String a) {…}
然而,更改的方法仍由invokespecial JVM指令调用,该指令仅用于私有方法调用(也适用于其他一些非方法的东西),但令人惊讶的是也适用于“包私有”方法在Sun / Oracle JVM上
在Android .class =>期间.dex 转换 invokespecial JVM指令转换为 invoke-direct Dalvik指令,该指令只能调用私有方法和构造函数。由于bar()
方法已成为包可见方法, invoke-direct 无法找到它并抛出NoSuchMethodError
。
解决方案是使用Eclipse 3.6+或javac编译器(通过build.xml
ant脚本)。
答案 1 :(得分:0)
我的猜测是“private void bar(String){}”标记为完全可由编译器内联,并且从未实际创建过。正是为什么foo()中的引用随后发生(vs inlining)很难说,但注释可能会破坏编译器的簿记。
(这里的线索是“私人的” - 私人方法几乎总是很好的内联候选者,特别是那些有虚体的人。)