一. Class类
1. class Class对象描述了运行中的classes和interfaces。通过Class对象可以取得运行中的classes和interfaces的相关信息。
2. 每个class都有一个相应的Class对象。每个class的Class对象存储于编译后的class所在的文件中。所以,当JVM装载一个.class文件时就会装载一个Class对象。所以,一个class只有一个Class对象。
二. Class类的基本应用
1. 如何获取一个class的Class对象
1.1 通过Class.forName(classname)方法,该方法接收一个String参数,用以指定要生成哪个class的Class对象.,如Class.forName(“Dog”)。
1.2 通过类字面常量(class literals)来获得。
1) 字面常量的形式为:classname.class。如Dog.class。
2) 对于基本类开,每种基本类型的外覆类都有一个名为TYPE的标准数据,能够产生一个指向相应的基本类型的Class对象的reference。如int.class等同于Integer.TYPE。
1.3 通过Object.getClass()方法来获得,如
Dog dog = new Dog();
dog.getClass();
1.4 一个产生Class对象的例子
class Cat{
Cat() { System.out.println("Init Cat()"); }
static{
System.out.println("Loading Cat");
}
}
class Dog{
Dog() { System.out.println("Init Dog()"); }
static{
System.out.println("Loading Dog");
}
}
class Duck{
Duck() { System.out.println("Init Duck()"); }
static{
System.out.println("Loading Duck");
}
}
public class Test{
public static void main(String[] args){
System.out.println("in main()");
new Cat();
System.out.println("After create Cat()");
try{
Class c1 = Class.forName("Dog"); //(1)
Class c2 = Dog.class; //(2)
}
catch(ClassNotFoundException cnfe){
cnfe.printStackTrace();
}
System.out.println("After Class.forName(/"Dog/")");
Duck d = new Duck();
System.out.println("After create Duck()");
Class c3 = d.getClass();
System.out.println("After Class.forName(/"Duck/")");
}
}
运行结果为:
in main()
Loading Cat
Init Cat()
After create Cat()
Loading Dog
After Class.forName("Dog")
Loading Duck
Init Duck()
After create Duck()
After Class.forName("Duck")
代码(1)(2)产生了class Dog的Class对象,但并没有产生class Dog对象。
**:产生一个class的Class对象不会导致产生一个class对象
2. 用Classc对象进行类型比较
2.1. 直接比较
两个Class对象不论是通过equals()函数还是直接用==运算符进行比较,比较的都是类型是否相同。
2.2. 通过Class.isInstance(object)函数进行比较,如
class Cat{}
class Dog{}
class Duck{}
public class Test{
public static void main(String[] args){
Class c2, c3;
Duck d = new Duck();
c2 = Dog.class;
c3 = d.getClass();
System.out.println("c2.isIntance(d):" + (c2.isInstance(d)));
System.out.println("c3.isIntance(d):" + (c3.isInstance(d)));
}
}
运行结果为:
c2.isIntance(d):false
c3.isIntance(d):true
2.3. 通过instanceof关键字进行比较。但是两个处于不同的继承体系中的类对象和Class对象进行比较,会产生编译错误。
class Cat{}
class Dog{}
class Duck{}
public class Test{
public static void main(String[] args){
Class c3;
//Object d = new Duck(); (1)
Duck d = new Duck(); //(2)
c3 = d.getClass();
System.out.println("d instanceof Dog:" + (d instanceof Dog));//(3)
}
}
由于Duck和Dog处于两个不同的继承体系中,所以代码(3)会发生编译错误。如果把代码(2)注释掉,并去掉代码(1)的注释,编译将通过。这是因为所有class都是继承自Object的,所以Object与Dog处于同一个继承体系中,可以进行比较。
2.4. 综合实例
class Base{}
class Derived{}
public class Test{
public static void test(Object x){
System.out.println("Testing x of type " +
x.getClass());
System.out.println("x instanceof Base " +
(x instanceof Base));
System.out.println("x instanceof Derived " +
(x instanceof Derived));
System.out.println("Base.isInstance(x) " +
Base.class.isInstance(x));
System.out.println("Derived.isInstance(x) " +
Derived.class.isInstance(x));
System.out.println("x.getClass() == Base.class " +
(x.getClass() == Base.class));
System.out.println("x.getClass() == Derived.class " +
(x.getClass() == Derived.class));
System.out.println("x.getClass().equals(Base.class) " +
(x.getClass().equals(Base.class)));
System.out.println("x.getClass().equals(Derived.class) " +
(x.getClass().equals(Derived.class)));
}
public static void main(String[] args){
test(new Base());
test(new Derived());
}
}
运行结果为:
Testing x of type class Base
x instanceof Base true
x instanceof Derived false
Base.isInstance(x) true
Derived.isInstance(x) false
x.getClass() == Base.class true
x.getClass() == Derived.class false
x.getClass().equals(Base.class) true
x.getClass().equals(Derived.class) false
Testing x of type class Derived
x instanceof Base false
x instanceof Derived true
Base.isInstance(x) false
Derived.isInstance(x) true
x.getClass() == Base.class false
x.getClass() == Derived.class true
x.getClass().equals(Base.class) false
x.getClass().equals(Derived.class) true