public final class Class { private Class() {} }
因为JVM为每一个加载的class
建立了对应的Class
实例,并在实例中保存了该class
的全部信息,包括类名、包名、父类、实现的接口、全部方法、字段等,所以,若是获取了某个Class
实例,咱们就能够经过这个Class
实例获取到该实例对应的class
的全部信息。java
这种经过Class
实例获取class
信息的方法称为反射(Reflection)。apache
如何获取一个class
的Class
实例?有三个方法:数组
方法一:直接经过一个class
的静态变量class
获取:spa
Class cls = String.class;
方法二:若是咱们有一个实例变量,能够经过该实例变量提供的getClass()
方法获取:code
String s = "Hello";
Class cls = s.getClass();
方法三:若是知道一个class
的完整类名,能够经过静态方法Class.forName()
获取:对象
Class cls = Class.forName(java.lang.String);
由于Class
实例在JVM中是惟一的,因此,上述方法获取的Class
实例是同一个实例。能够用==
比较两个Class
实例:blog
Class cls1 = String.class; String s = "hello"; Class cls2 = s.getClass(); System.out.println(cls1 == cls2); // true
由于反射的目的是为了得到某个实例的信息。所以,当咱们拿到某个Object
实例时,咱们能够经过反射获取该Object
的class
信息:接口
1 public class Demo{ 2 public static void main(String[] args) { 3 printObjectInfo("wang"); 4 } 5 6 public static void printObjectInfo(Object object){ 7 Class cls = object.getClass(); 8 System.out.println(cls); //class java.lang.String 9 } 10 }
要从Class
实例获取某个class的基本信息内存
1 public class Demo12{ 2 public static void main(String[] args) { 3 //字符串实例 4 printClassInfo("".getClass()); 5 System.out.println("****************"); 6 //Runnable接口 7 printClassInfo(Runnable.class); 8 System.out.println("****************"); 9 printClassInfo(java.time.Month.class); 10 System.out.println("****************"); 11 printClassInfo(int[].class); 12 System.out.println("****************"); 13 printClassInfo(long.class); 14 } 15 16 static void printClassInfo(Class cls){ 17 //class全限定类名 18 System.out.println("Class name:" + cls.getName()); 19 //class类名 20 System.out.println("Simple name:" + cls.getSimpleName()); 21 //包名 22 if(cls.getPackage() != null){ 23 System.out.println("Package name:" + cls.getPackage().getName()); 24 } 25 //判断该类是不是接口 26 System.out.println("is interface:" + cls.isInterface()); 27 //判断该类是不是枚举 28 System.out.println("is enum:" + cls.isEnum()); 29 //判断该类是不是数组 30 System.out.println("is array:" + cls.isArray()); 31 //判断该类是不是基本类型 32 System.out.println("is primitive:" + cls.isPrimitive()); 33 } 34 }
若是获取到了一个Class
实例,咱们就能够经过该Class
实例来建立对应类型的实例字符串
//获取String的Class对象 Class cls = String.class; //建立一个String对象 String s = (String)cls.newInstance();
上述代码至关于new String()
。经过Class.newInstance()
能够建立类实例,
它的局限是:只能调用public
的无参数构造方法。带参数的构造方法,或者非public
的构造方法都没法经过Class.newInstance()
被调用
JVM在执行Java程序的时候,并非一次性把全部用到的class所有加载到内存,而是第一次须要用到class时才加载。
1 public class Demo{ 3 public static void main(String[] args){ 4 if(args.length > 0){ 5 create(args[0]); 6 } 7 } 8 9 static void create(String name){ 10 Person p = new Person(name); 11 } 12 13 }
当执行Demo.java时,因为用到了Demo,所以,JVM首先会把Demo.class加载到内存。然而,并不会加载Person.class,除非程序执行到create()方法,JVM发现须要加载Person类时,才会首次加载Person.class。若是没有执行create()方法,那么Person.class根本就不会被加载
这就是JVM动态加载class的特性
动态加载class的特性对于Java程序很是重要。利用JVM动态加载class的特性,咱们才能在运行期根据条件加载不一样的实现类。例如,Commons Logging老是优先使用Log4j,只有当Log4j不存在时,才使用JDK的logging.利用JVM动态加载特性,大体的实现代码以下:
1 //Commons logging优先使用Log4j 2 LogFactory factory = null; 3 if(isClassPresent("org.apache.logging.log4j.Logger")){ 4 factory = createLog4j(); 5 }else{ 6 factory = createJdkLog(); 7 } 8 9 boolean isClassPresent(String name){ 10 try{ 11 Class.forName(name); 12 return true; 13 }catch(Exception e){ 14 return false; 15 } 16 17 }
这就是为何咱们只须要吧Log4j的jar包放到classpath中,Commons Logging就会自动使用Log4j的缘由
1.JVM为每一个加载的class
及interface
建立了对应的Class
实例来保存class
及interface
的全部信息;
2.获取一个class
对应的Class
实例后,就能够获取该class
的全部信息;
3.经过Class实例获取class
信息的方法称为反射(Reflection);
4.JVM老是动态加载class
,能够在运行期根据条件来控制加载class。