类加载的特点
*.java ---javac 编译为 ---> *.class ---> 类加载 --> 加载至jvm内存中 --> 可以使用这儿类
按需加载 - 用到时才加载,用不到不加载
-
main 方法的所在类,运行时一定会加载(运行java)
-
类名.static 用类静态变量或静态方法时,此类也会加载
-
new类() 加载此类
-
new 子类()加载此类,包括父类
-
反射使用类,也会导致类加载
类加载阶段
加载
- 把磁盘上*.class读入内存,形成了一个类对象
- 如果不是创建对象,而是就想用类本身,可以通过反射来访问类对象
链接
- 验证 : 检查类的格式是否正确,并保证不会被不安全的代码破坏虚拟机
- 准备 : 给类的静态变量分配空间
- 解析 : 将符号引用替换为直接引用,将其它的符号(例如类名) 替换为内存中真正的地址
初始化
- 给类中静态变量赋值
- 执行所有的静态代码块
class Test{
static int count = 20;
static {
count = 30;
}
}
类加载器
四类
JDK8以下
- Bootstrap ClassLoader(启动类加载器) 在java中用null来表示(不需要去使用)
- 在jdk \jre\lib\rt.jar(包含java核心的类)
- 扩展类加载器
- 在jdk \jre\lib\ext 此(扩展)目录下类
- System ClassLoader(系统类加载器也叫做 应用程序加载器Application ClassLoader)
- classpath 下的所有类
- 所谓的classpath自己编写的类的存储位置: idea在 out/production/*.class
- 自定义
JDK9开始
- Bootstrap ClassLoader
- 扩展类加载器 被替换为Platform ClassLoader(平台类加载器)
- System ClassLoader(也叫应用程序类加载器Application ClassLoader)
- classpath下的所有类
- 自定义
双亲委派
- 优先交给上一级来加载类
- 如果上级加载器能找到这个类,就有上级优先加载
- 如果上级找不到,再交给下级来加载
- 保证了上级加载器的加载的优先级更高
类路径
- 在src下创建的properties文件,实际上要拷贝到类路径下(out/production/模块名)
- 下面的代码找的实际是类路径下的文件
//使用类加载器获取到文件输入流
InputStream is = ClassLoader.getStreamClassLoader().getResourceAsStream("txt.properties");
//创建配置文件对象
Properties prop = new Properties();
//将文件输入流中的数据加载进配置文件对象中
prop.load(in);
System.out.println(prop);
//释放资源
is.close();
反射
获取类的"元信息" : 包括类的各个组成: 成员变量,成员方法,构造方法,注解
特点
- 可以通过反射,能突破访问修饰符的限制,可以访问到私有方法,变量
- 根据配置文件,动态创建对象和调用方法,可以减少java代码的修改,让代码更加通用
获取类对象
创建对象: new Student()
Class类 : Student.Class : 带class叫类对象
三种方法获取类对象
-
Class clazz = Class.forName(“全限定名”);
-
类名.class : 经常用于测试
-
对象.getClass() : 很少使用
类对象在内存中唯一
获取构造
Constructor 构造器,构造方法
Student.class.getConstructors(); //获取所有的公共的构造
Student.class.getConstructor(参数类型); //获取某个公共构造
Student.class.getDelcaredConstructors();//获取所有的构造方法包括private, protected, 默认的 public)
Student.class.getDelcaredConstructor(参数类型);//获取某个构造方法 **常用方法**
反射创建对象
Constructor //构造器,构造方法
Constructor.setAccessible(true);//允许访问(暴力反射)会破坏对象的封装性 获取并使用该类的私有构造方法
Method.setAccessible(true): //通过暴力反射获取该类的私有成员方法
Constructor.newInstance(实际参数);//创建新的对象
访问成员变量
//获取类对象
Class clazz = Student.class;
Field name = clazz.getDeclaredField("name");
Field age = clazz.getDeclaredField("age");
Student stu = new Student("张三",22);
//stu.setName("李四");
//使用反射方法给成员变量赋值
name.set(stu,"李四");
System.out.println(stu);//Student{name='李四', age=22}
System.out.println(age.get(stu));//22
反射调用方法
对象.方法(实际参数);//正常调用
方法.invoke(对象,实际参数);//反射调用成员方法
方法.invoke(null,实际参数);//反射调用静态方法,null表示静态方法不会关联对象
总结
Construcor
getConstructors //所有的公共构造
getDelcaredConstructors()//获取本类所有构造
Method
getMethods() //所有的公共方法,包括继承的公共方法
getDelcaredMethods() //本类所有方法,不包括继承的
Filed
getFileds() //所有的公共成员变量,包括继承的公共成员变量
getDeclcaredFields()//本类所有成员变量,不包括继承的