我猜这两个函数编译成完全相同的字节码,但我想问一下这个问题。是否在不需要降低性能的情况下调用方法调用?
例如:
package com.my;
import android.app.Activity;
import android.os.Bundle;
public class Main extends Activity {
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
this.setContentView(R.layout.main); // Fully qualified
setContentView(R.layout.main); // Not fully qualified
}
}
标准是使用后者,但完全合格的电话this.setContentView()
是否有任何负面影响?
答案 0 :(得分:9)
不,没有任何惩罚。两个调用都将在字节码中使用invokespecial
来调用成员方法。
在标准Java中,以下代码:
public class TestQualified {
private void someMethod() {
}
public void otherMethod() {
this.someMethod();
someMethod();
}
}
产生以下字节码:
Compiled from "TestQualified.java"
public class TestQualified extends java.lang.Object{
public TestQualified();
Code:
0: aload_0
1: invokespecial #1; //Method java/lang/Object."<init>":()V
4: return
private void someMethod();
Code:
0: return
public void otherMethod();
Code:
0: aload_0
1: invokespecial #2; //Method someMethod:()V
4: aload_0
5: invokespecial #2; //Method someMethod:()V
8: return
}
如你所见,两个电话都是一样的。
现在Android使用.class
工具将dx
文件转换为dalvik字节码,该工具输出.dex
个文件。由于.class
文件没有显示两个调用之间的任何区别,因此相应的.dex
文件也不会显示任何差异:
Processing 'TestQualified.dex'...
Opened 'TestQualified.dex', DEX version '035'
Class #0 -
Class descriptor : 'LTestQualified;'
Access flags : 0x0001 (PUBLIC)
Superclass : 'Ljava/lang/Object;'
Interfaces -
Static fields -
Instance fields -
Direct methods -
#0 : (in LTestQualified;)
name : '<init>'
type : '()V'
access : 0x10001 (PUBLIC CONSTRUCTOR)
code -
registers : 1
ins : 1
outs : 1
insns size : 4 16-bit code units
0000e4: |[0000e4] TestQualified.<init>:()V
0000f4: 7010 0300 0000 |0000: invoke-direct {v0}, Ljava/lang/Object;.<init>:()V // method@0003
0000fa: 0e00 |0003: return-void
catches : (none)
positions :
0x0000 line=1
locals :
0x0000 - 0x0004 reg=0 this LTestQualified;
#1 : (in LTestQualified;)
name : 'someMethod'
type : '()V'
access : 0x0002 (PRIVATE)
code -
registers : 1
ins : 1
outs : 0
insns size : 1 16-bit code units
0000fc: |[0000fc] TestQualified.someMethod:()V
00010c: 0e00 |0000: return-void
catches : (none)
positions :
0x0000 line=4
locals :
0x0000 - 0x0001 reg=0 this LTestQualified;
Virtual methods -
#0 : (in LTestQualified;)
name : 'otherMethod'
type : '()V'
access : 0x0001 (PUBLIC)
code -
registers : 1
ins : 1
outs : 1
insns size : 7 16-bit code units
000110: |[000110] TestQualified.otherMethod:()V
000120: 7010 0200 0000 |0000: invoke-direct {v0}, LTestQualified;.someMethod:()V // method@0002
000126: 7010 0200 0000 |0003: invoke-direct {v0}, LTestQualified;.someMethod:()V // method@0002
00012c: 0e00 |0006: return-void
catches : (none)
positions :
0x0000 line=7
0x0003 line=8
0x0006 line=9
locals :
0x0000 - 0x0007 reg=0 this LTestQualified;
source_file_idx : 3 (TestQualified.java)
您可以看到这两个电话都是使用invoke-direct
进行的。因此,这里也没有性能损失。
答案 1 :(得分:1)
不,不。对编译器来说都是一样的。使用this
引用可能会被视为阅读帮助,例如java7 number literals