`
haibin369
  • 浏览: 58872 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

第十四章-类型信息(1)

阅读更多

一、Class对象

      Class对象是一种特殊的对象,它包含了与类有关的信息。事实上,Class对象就是用来创建类的所有“常规”对象的。每个类都有一个Class对象,换言之,每当编写并编译了一个新类,就会产生一个Class对象(保存在一个同名的.class文件中)。为了生成这个类的对象,运行这个程序的Java虚拟机将使用“类加载器”。

      所有的类都是在对其第一次使用时,动态加载到JVM中的。当程序创建第一个对类的静态成员引用时,就会加载这个类。这证明了构造器也是类的静态方法,即使在构造器之前并没有使用static关键字。因此,使用new创建类的新对象也会被当作对类的静态成员的引用。

      类加载器首先检查这个类的Class对象是否已经加载。如果尚未加载,默认的类加载器就会根据类名查找.class文件。在这个类的字节码被加载时,它们就会接收验证,以确保其没有被破坏,并且不包含不良的Java代码。

      一旦某个类的Class对象被载入内存,它就被用来创建这个类的所有对象。

      Class对象中,forName()是取得Class对象引用的一种方法。它是用一个包含目标类的文本名(区分大小写)的String作为输入参数,返回的是该类的Class对象的引用,使用该方法会显式的要求JVM加载该类的.class文件并初始化该类。

class A
{
	static
	{
		System.out.println("From A!");
	}
}

class B extends A
{
	static
	{
		System.out.println("From B!");
	}
}

public class Test 
{
	public static void main(String[] args)
	{
		System.out.println("---------Using 'new'---------");
		//创建对象之前,根据继承关系,需要首先加载A.class,然后加载B.class
		B b = new B();
		System.out.println("-----------------------------");
		
		System.out.println("---Using 'Class.forname()'---");
		try
		{
			//显式要求加载B.class,但是B.class已经加载,所以不会重新加载
			Class.forName("B");
		}
		catch(Exception e)
		{
			e.printStackTrace();
		}
		System.out.println("-----------------------------");
	}
}

输出

---------Using 'new'---------
From A!
From B!
-----------------------------
---Using 'Class.forname()'---
-----------------------------

 

class A
{
	static
	{
		System.out.println("From A!");
	}
}

class B extends A
{
	static
	{
		System.out.println("From B!");
	}
}

public class Test 
{
	public static void main(String[] args)
	{

		System.out.println("---Using 'Class.forname()'---");
		try
		{
			//显式要求加载B.class,发现B.class和A.class都没加载
			//所以根据继承关系会先加载A.class然后加载B.class
			Class.forName("B");
		}
		catch(Exception e)
		{
			e.printStackTrace();
		}
		System.out.println("-----------------------------");
		
		System.out.println("---------Using 'new'---------");
		//B.class已经加载,可以创建对象
		B b = new B();
		System.out.println("-----------------------------");
	}
}

 

输出

---Using 'Class.forname()'---
From A!
From B!
-----------------------------
---------Using 'new'---------
-----------------------------

 

二、类字面常量

      Java提供了另一种方法来生成对Class对象的引用,即使用“类字面常量”:例如B.class。

      这样做不仅更简单,而且更安全,因为它在编译时就会受到检查(因此不需try&catch),并且它根除了对forName()方法的调用,所以更高效。

      需要注意的是,当使用该方式创建对Class对象的引用时,不会自动初始化该Class对象。为了使用类而做的准备工作实际上包含三个步骤:

      1.加载,这是由类加载器执行的。该步骤将查找字节码,并从这些字节码中创建一个Class对象。

      2.连接。在连接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必须的话,将解析这个类创建的对其他类的所有引用。

      3.初始化。如果该类有超类,则对其初始化,执行静态初始化器和静态初始化块。

       使用类字面常量时初始化被延迟到了对静态方法(构造器隐式地是静态的)或者非常数静态域进行首次引用时才执行。

class A
{
	static
	{
		System.out.println("From A!");
	}
}

class B extends A
{
	static
	{
		System.out.println("From B!");
	}
}

public class Test 
{
	public static void main(String[] args)
	{
		System.out.println("-----using '.class'------");
		//只加载.class而不初始化,所以无输出
		Class c = B.class;
		System.out.println("-------------------------");

		System.out.println("--using Class.forName()--");
		try
		{
			//虽然已经加载,但是还未初始化,因此会初始化
			Class.forName("B");
		}
		catch (Exception e) 
		{
			e.printStackTrace();
		}
		System.out.println("-------------------------");
	}
}

输出

-----using '.class'-----
-------------------------
--using Class.forName()--
From A!
From B!
-------------------------

 

三、instanceof

      instanceof的语法:[某个对象] instanceof [某个类或接口]

      它返回一个布尔值,用来告诉我们某个对象是否是某个类的实例,或者是某个类的子类的实例;当第二个参数是接口时,表示某个对象是否是该接口的某个实现的实例。它考虑了类的继承体系。

class A
{}

class B extends A
{}

interface C
{}

class D implements C
{}

public class Test 
{
	public static void main(String[] args)
	{
		B b = new B();
		System.out.println("b instance of B: " + (b instanceof B));
		System.out.println("b instance of A: " + (b instanceof A));
		
		D d = new D();
		System.out.println("d instance of D: " + (d instanceof D));
		System.out.println("d instance of C: " + (d instanceof C));
	}
}

输出

b instance of B: true
b instance of A: true
d instance of D: true
d instance of C: true

 

 

0
0
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics