二、Hello World即遭不测
显然,首先是来一个Hello World。
直接使用android-plugin的项目模板。生成的程序非常简单,当然,Hello World都非常简单。 Activity类的代码如下。
1 2 3 4 5 6 7 8 |
|
很简单,可以运行,就不截图了。 然后我把我的代码改成这样。
1 2 3 4 5 6 7 8 9 10 |
|
这里的doto方法是我自己的一个公共模块里的方法(作为一个jar引入到工程中),是我从clojure里面看着顺眼结果来的,作用是有机会调用一个object的一系列方法,然后还能返回这个对象本身。代码如下——
1 2 3 4 5 6 7 8 9 |
|
再运行,不测——NoSuchMethodException 看了log,发现了问题在于——虚拟机企图在Object上去找doto方法,肯定是找不到的。 于是我换了一种写法——
1 2 3 4 5 6 7 8 9 10 |
|
可以运行了,然而问题在哪里呢?为什么jvm中运行效果一样的两种写法在Dalvik虚拟机里面表现出不一样的结果呢? 看了一下两种写法编译出来的class代码 第一种(运行错误的一种——object Helper)
第二种(运行正确的一种——object Helper2)
可以看到,两种写法编译出来的方法type不同,主要是返回类型不同。这种差异确实是源代码差异的直接体现。从被调用方的情况看不出什么特别的情况。于是转换一下思路,从调用方来看看呢。 专门写了个简单的调用方,代码如下——
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
|
编译,再次看class文件。 UseHelper
UseHelper2
可以看到,编译出来的class代码是非常不同的。 虽然源代码几乎相同,但Helper的调用方使用的是reflection方式的调用,而Helper2的调用方没有使用Reflection,直接是调用了特定的类上面的方法。 那么为什么使用reflection的方式在Dalvik虚拟机上不能正常运行呢?有待进一步的研究。