a.getClass()
在 Java 中,围绕使用or的选择存在哪些优点/缺点A.class
?两者都可以在任何Class<?>
预期的地方使用,但我想在不同的情况下使用两者都会有性能或其他微妙的好处(就像有Class.forName()
和ClassLoader.loadClass()
.
7 回答
我不会比较它们的优缺点,因为它们有不同的目的,而且很少在两者之间做出“选择”。
a.getClass()
返回 的运行时类型。a
即,如果你有A a = new B();
然后a.getClass()
将返回B
类。A.class
静态评估A
类,并用于通常与反射相关的其他目的。
在性能方面,可能存在可衡量的差异,但我不会说什么,因为最终它取决于 JVM 和/或编译器。
这篇文章在这里被改写为一篇文章。
它们在您可以使用它们的位置方面实际上是不同的。A.class
在编译时工作,而a.getClass()
需要类型的实例A
并在运行时工作。
也可能存在性能差异。虽然A.class
可以由编译器解析,因为它知道 的实际类型A
,a.getClass()
是在运行时发生的虚拟方法调用。
作为参考,针对字节码的编译器通常会发出以下指令Integer.getClass()
:
aload_1
invokevirtual #3; //Method java/lang/Object.getClass:()Ljava/lang/Class;
以及以下内容Integer.class
:
//const #3 = class #16; // java/lang/Integer
ldc_w #3; //class java/lang/Integer
前者通常涉及虚拟方法分派,因此可能需要更长的时间来执行。然而,这最终取决于 JVM。
看看下面的例子
a.getClass()!= A.class
,即 a 不是 A 的实例,而是 A 的匿名子类
a.getClass()
需要类型 A 的实例
a.getClass
当你有一个类/类型的实例并且你想得到它的确切类型时使用。当a.class
你type
有空并且你想创建它的实例时使用。
在编译时评估时还getClass()
返回实例的运行时类型。
考虑和的性能,比 具有更好的性能 。例子 :.class
getClass()
.class
.class
getClass()
public class PerfomanceClass {
public static void main(String[] args) {
// TODO Auto-generated method stub
long time=System.nanoTime();
Class class1="String".getClass();
class1="String".getClass();
class1="String".getClass();
class1="String".getClass();
System.out.println("time (getClass()) :"+(System.nanoTime()-time)+" ns");
long time2=System.nanoTime();
Class class2=String.class;
class2=String.class;
class2=String.class;
class2=String.class;
System.out.println("time (.class):"+(System.nanoTime()-time2)+" ns");
}
}
输出 :
time (getClass()) : 79410 ns
time (.class) : 8032 ns
我想补充一个区别。假设您有一个如下所示的构造函数类,其中包含一个接受 Class 对象的超类。您希望每当创建子类对象时,应将子类的类对象传递给超类。下面的代码将无法编译,因为您不能在构造函数中调用实例方法。在这种情况下,如果您替换myObject.getClass()
为MyClass.class
. 它会完美运行。
Class MyClass
{
private MyClass myObject = new MyClass();
public MyClass()
{
super(myObject.getClass()); //error line compile time error
}
}
有趣的是,上面示例中提到的性能差异似乎与其他原因有关。使用 3 个不同的类,平均而言,性能几乎相同:
import java.util.LinkedHashMap;
public class PerfomanceClass {
public static void main(String[] args) {
long time = System.nanoTime();
Class class1 = "String".getClass();
Class class11 = "Integer".getClass();
Class class111 = "LinkedHashMap".getClass();
System.out.println("time (getClass()) :" + (System.nanoTime() - time) + " ns");
long time2 = System.nanoTime();
Class class2 = String.class;
Class class22 = Integer.class;
Class class222 = LinkedHashMap.class;
System.out.println("time (.class):" + (System.nanoTime() - time2) + " ns");
} }
输出将类似于:
time (getClass()) :23506 ns
time (.class):23838 ns
并且切换调用的顺序甚至会导致getClass()
更快。
import java.util.LinkedHashMap;
public class PerfomanceClass {
public static void main(String[] args) {
long time2 = System.nanoTime();
Class class2 = LinkedHashMap.class;
System.out.println("time (.class):" + (System.nanoTime() - time2) + " ns");
long time = System.nanoTime();
Class class1 = "LinkedHashMap".getClass();
System.out.println("time (getClass()) :" + (System.nanoTime() - time) + " ns");
}}
输出:
time (.class):33108 ns
time (getClass()) :6622 ns
p.getClass()
,其中p
是对象的实例,返回此对象的运行时类p
。p
不能是会导致编译时错误的类型,它应该是对象的实例。
// B extends A
A a = new B();
System.out.println(a.getClass());
//output: class B
p.class
是一个表达式。.class
称为类语法。p
是一种类型。它可以是类、接口或数组的名称,甚至是原始类型。
a.getClass() == B.class
.
如果类型可用并且有实例,则可以使用getClass
方法来获取类型的名称。否则,使用.class
语法