Java反射——java.lang.Class和类的加载

反射的基础: java.lang.Class

Class类的实例对象,用于记录类描述信息。
源码说:represent classes and interfaces in a running Java application

Class类没有公共的构造方法,无法通过new运算符实例化;只能通过对象的getClass方法,或是通过Class.forName(…)来获得实例。

static ClassforName(String className)throws ClassNotFoundException 使用参数className来指定具体的类,来获得相关的类描述对象,该方法有可能抛出类加载异常(ClassNotFoundException),必须捕捉
Class getSuperclass() 获得当前类描述对象的父类的描述对象
String getName() 返回当前类描述对象的类名称

获取Class对象的三种方式:

public class _T11获取Class {
	// Class:类描述对象
	// (源码说:represent classes and interfaces in a running Java application)
	public static void main(String[] args) {
		Class<?> _class;
		// ***1*对象.getClass()
		String str = "";
		_class = str.getClass();
		System.out.println(_class + "-----对象名.getClass()");
		// ***2*类.class
		_class = String.class;
		System.out.println(_class + "-----类名.class");
		// ***3*Class.forName("")
		try {
			_class = Class.forName("java.lang.String");
			System.out.println(_class + "-----Class.forName(...)");
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
	}
}

class java.lang.String-----对象名.getClass()
class java.lang.String-----类名.class
class java.lang.String-----Class.forName(...)

.getSuperclass()

_class.getSuperclass():获得父类的描述对象

以下示例,打印StringBuffer的父类、父类的父类……

public class _T12getSuperclass {
	public static void main(String[] args) {
		System.out.println("-----.getSuperclass()获得父类的描述对象-----");
		try {
			Class _class = Class.forName("java.lang.StringBuffer");
			// 循环打印父类信息,直到没有父类
			while (_class != null) {
				System.out.println(_class);
				// getSuperclass():获得父类的描述对象
				_class = _class.getSuperclass();
			}
		} catch (ClassNotFoundException cnfe) {
			cnfe.printStackTrace();
		}
	}
}

-----.getSuperclass()获得父类的描述对象-----
class java.lang.StringBuffer
class java.lang.AbstractStringBuilder
class java.lang.Object

类的加载

Java程序运行在Java虚拟机进程中,同一个JVM的所有线程、所有变量都处于同一个进程里。
当系统出现以下几种情况时,JVM进程将被终止:
|--程序正常结束。
|--程序运行到System.exit()或Runtime.getRuntime().exit()代码。
|--程序执行过程中遇到未捕获的异常或错误。
|--强制结束JVM进程。

类加载,是指将类的.class文件读入内存,并为之创建一个java.lang.Class对象。

class Tester {
	static {
		System.out.println("静态初始化块...");
	}
	public Tester() {
		System.out.println("构造方法...");
	}
}
public class _T21TestClassLoader {
	public static void main(String[] args) throws ClassNotFoundException {
		ClassLoader _cLoader = ClassLoader.getSystemClassLoader();
		System.out.println("=====loadClass():加载类,但不做初始化=====");
		String _包类名 = "Tester";
		Class<?> loadClass = _cLoader.loadClass(_包类名);
		System.out.println("加载即得Class:" + loadClass);
		// -----------------------------------------------------
		System.out.println("=====Class.forName():初始化(静态代码块执行)=====");
		Class.forName(_包类名);
	}
}

=loadClass():加载类,但不做初始化=
加载即得Class:class Tester
=Class.forName():初始化=
静态初始化块...

static变量的初始化

使用static final的变量(常量),如果值可以在编译期确定,则类不需要初始化。

// static final修饰的变量,被用于“常量”,更类似于“宏定义”。
// 当其在编译器能确定时,不需要初始化类,使用“宏变量”替换的形式进行编译;
// 当其不能在编译期确定时,需要初始化类;
// 如果未加final,不是“宏定义”,需要初始化类。
class StaticField {
	static {
		System.out.println("---此代码不执行---");
	}
	static final String compileConstant = "static final变量,编译时能确定,类不做初始化,使用'宏替换'的形式编译";
}
class StaticField2 {
	static {
		System.out.println("【2:此代码执行,表示初始化过了】");
	}
	static final String compileConstant = "static final变量,编译时不能确定时,运行时才确定。"
			+ System.getProperty("os.name");
}
class StaticField3 {
	static {
		System.out.println("【3:此代码执行,因为就不是final】static{}变量不是final,编译时能确定,变量");
	}
	static String compileConstant = "非final的static变量";
}
public class _T22宏定义常量 {
	public static void main(String[] args) {
		System.out.println("\t" + StaticField.compileConstant);
		System.out.println("\t" + StaticField2.compileConstant);
		System.out.println("\t" + StaticField3.compileConstant);
	}
}

运行结果:

	static final变量,编译时能确定,类不做初始化,使用'宏替换'的形式编译
【2:此代码执行,表示初始化过了】
	static final变量,编译时不能确定时,运行时才确定。Windows 10
【3:此代码执行,因为就不是final】static{}变量不是final,编译时能确定,变量
	非final的static变量
posted @ 2019-07-14 00:44  虎老狮  阅读(1479)  评论(0编辑  收藏  举报