反射的基础: 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变量