类加载器的作用与类缓存:
类加载器的作用:将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 text1 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
Class c1 = Class.forName("Collections.person");
//获得类的名字
System.out.println(c1.getName());
//获得包名 +类名System.out.println(cl.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
===================================
java.lang.String Collections.person.name
int Collections.person.age
java.lang.String Collections.person.sex
java.lang.String Collections.person.city
===================================
java.lang.String Collections.person.name
注:在获得指定属性的值时,一定要使用getDeclaredField()方法,而不能使用getFields(),因为getFields只能获取到公共属性
获取类的方法和构造器的信息:
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='北京'}