类加载器的作用与类缓存:
类加载器的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口。
类缓存:标准的JavaSE类加载器可以按要求查找类,但一旦某个类被加载到类加载器中,它将维持加载(缓存)一段时间。不过JVM垃圾回收机制可以回收这些Class对象
JVM 规范定义了如下类型的类的加载器:
获取加载器的方法:
package Collections; public class text1 { public static void main(String[] args) throws ClassNotFoundException { //获取系统类的加载器 ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();//通过加载器调用方法 System.out.println(systemClassLoader); //获取系统类加载器的父类加裁器-->扩展类加裁器 ClassLoader parent = systemClassLoader.getParent();//通过系统类加载器调用方法 System.out.println(parent); //获取扩展类加载器的父类加裁器-->根加裁器(c/c++) ClassLoader parent1 = parent.getParent();//通过扩展类加载器调用方法 System.out.println(parent1); //测试当前类是哪个加载器加裁的 ClassLoader classLoader = Class.forName("Collections.text1").getClassLoader(); System.out.println(classLoader); //测试JDK内置的类是谁加载的 classLoader = Class.forName("java.lang.Object").getClassLoader(); System.out.println(classLoader); //获得系统类加载器可以加裁的路径 System.out.println(System.getProperty("java.class.path")); } }
输出:
jdk.internal.loader.ClassLoaders$AppClassLoader@78308db1 jdk.internal.loader.ClassLoaders$PlatformClassLoader@15aeb7ab null jdk.internal.loader.ClassLoaders$AppClassLoader@78308db1 null ------[省略]
获取运行时类的完整结构:
通过反射获取运行时类的完整结构:
Field、Method、Constructor、Superclass、Interface、Annotation
获得有关类自身的信息:
package Collections; import java.lang.reflect.Field; public class text2 { public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException { Class c1 = Class.forName("Collections.person"); //获得类名 System.out.println(c1.getName()); //获得包名 +类名 System.out.println(c1.getSimpleName()); //获得类的属性 System.out.println("==================================="); Field[] fields = c1.getFields(); //只能找到public属性 fields = c1.getDeclaredFields(); //获得全部的属性 for (Field field : fields) { System.out.println(field); } System.out.println("==================================="); //获得指定属性的值 Field name = c1.getDeclaredField("name"); System.out.println(name); } }
输出:
Collections.person person =================================== static java.lang.String Collections.person.name int Collections.person.age java.lang.String Collections.person.sex java.lang.String Collections.person.city static Collections.person Collections.person.person1 =================================== static java.lang.String Collections.person.name
注:在获得指定属性的值时,一定要使用getDeclaredField()方法,而不能使用getFields(),因为getFields只能获取到public属性
获取类的方法和构造器的信息:
package Collections; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; public class text1 { public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException { Class c1 = Class.forName("Collections.person"); Method[] methods = c1.getMethods(); //获得本类及其父类的全部public方法 for (Method method : methods) { System.out.println("正常的:"+method); } methods = c1.getDeclaredMethods();//获得本类的所有方法 for (Method method : methods) { System.out.println("getDeclaredMethods:"+method); } //获得指定方法 //重载 Method getName = c1.getMethod("getName", null) ; Method setName =c1.getMethod("setName", String.class) ; System.out.println(getName); System.out.println(setName); //获得指定的构造器 System.out.println("==================================="); Constructor[] constructors = c1.getConstructors();//获得public方法 for (Constructor constructor : constructors) { System.out.println(constructor); constructors = c1.getDeclaredConstructors();//获得所有方法 for (Constructor constructor1 : constructors) { System.out.println("#" + constructor1); } } //指定的某一个构造器 Constructor declaredConstructors=c1.getDeclaredConstructor(String.class,int.class, String.class, String.class); System.out.println("指定的某一个构造器:"+declaredConstructors); } }
获取Class对象的作用:
创建类的对象:调用Class对象的newlnstance()方法
1:类必须有一个无参数的构造器 2:类的构造器的访问权限需要足够
举例:
package Collections; public class person_text { public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException { //获得Class对象 Class c1 = Class.forName("Collections.person"); //构造一个对象 person user = (person)c1.newInstance(); //本质是调用了类的无参构造器 System.out.println(user); } }
报错:
Exception in thread "main" java.lang.InstantiationException: Collections.person at java.base/java.lang.Class.newInstance(Class.java:671) at Collections.person_text.main(person_text.java:8) Caused by: java.lang.NoSuchMethodException: Collections.person.<init>() at java.base/java.lang.Class.getConstructor0(Class.java:3617) at java.base/java.lang.Class.newInstance(Class.java:658) ... 1 more
由于person中没有无参构造器,因此,我们无法通过调用Class对象的newlnstance()方法去创建类的对象
难道没有无参的构造器就不能创建对象了吗?
答案当然不是如此,只要在操作的时候明确的调用类中的构造器并将参数传递进去之后,就可以实例化操作。
步骤如下:
1)通过Class类的getDeclaredConstructor(Class ... parameterTypes) 取得本类的指定形参类型的构造器 2)向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数 3)通过Constructor实例化对象
举例:
package Collections; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; public class person_text { public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException { //获得Class对象 Class c1 = Class.forName("Collections.person"); //通过构造器创建对象 Constructor constructor = c1.getDeclaredConstructor(String.class, int.class,String.class, String.class); person user2 = (person)constructor.newInstance("Lisa",19,"女","北京"); System.out.println(user2); } }
输出:
person{name='Lisa', age=19, sex='女', city='北京'}