java 类加载机制和反射机制
一.类的加载机制
jvm把class文件加载到内存,并对数据进行校验、解析和初始化,最终形成jvm可以直接使用的java类型的过程。
(1)加载
将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区中的运行时数据结构,在堆中生成一个代表这个类的java.lang.Class对象,作为方法区类数据的访问入口。
(2)链接 将java类的二进制代码合并到jvm的运行状态之中的过程。
- 验证:确保加载的类信息符合jvm规范,没有安全方面的问题。
- 准备:正式为类变量(static变量)分配内存并设置类变量初始值的阶段,这些内存都将在方法区中进行分配。
- 解析:虚拟机常量池内的符号引用替换为直接引用的过程。(比如String s ="aaa",转化为 s的地址指向“aaa”的地址)
(3)初始化
初始化阶段是执行类构造器方法的过程。类构造器方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static块)中的语句合并产生的。
当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先初始化其父类的初始化
虚拟机会保证一个类的构造器方法在多线程环境中被正确加锁和同步
当访问一个java类的静态域时,只有真正声明这个静态变量的类才会被初始化。
二.java反射机制
通俗地说:反射就是将Student类中的方法、成员变量、构造方法,抽象出来可以让Class的对象单独进行访问。
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

1、获取Class对象的三种方式
(1)通过调用forName(String className),注意className的格式为:包名.类名获得Class对象。

(2)通过类获取Class对象,String.class、int.class这些字节码是Class类的对象。
(3)通过类的对象获取Class对象。
public static void getClass1() throws ClassNotFoundException {
//获取Class的方法
//正射
Student stu=new Student();
//第一种(优先选择):传的是包名.类名,通过包名和类名获得反射Class
Class cl1=Class.forName("day20.Student");
//第二种:通过类来获取Class对象
Class cl2=Student.class;
//第三种:通过对象获取Class对象
Student s=new Student();
Class cl3=s.getClass();
//判断三种方法是否是同一个Class对象
System.out.println(cl1==cl2);
System.out.println(cl2==cl3);
}
2、获取构造方法
(1)获取公开的构造方法

(2)获取所有的构造方法
主要为调取Student类中的私有方法。注意:不能获得Student类的父类Human中的构造方法。

(3)调用特定的构造方法
无参构造和有参构造方法Constructors1.newInstance();//实例化获取的类

私有构造方法的调用,要注意要设置其访问权限Constructors1.setAccessible(true)。
public static void getConstruction() throws Exception {
//获取构造方法
Class cl=Class.forName("day20.Student");
System.out.println("==========公开的构造方法==========");
//获得公开的构造方法
Constructor[] cons=cl.getConstructors();
for(Constructor con:cons) {
System.out.println(con);
}
System.out.println("==========所有的构造方法==========");
//获得所有的构造方法
Constructor[] constructors=cl.getDeclaredConstructors();
for (Constructor con:constructors
) {
System.out.println(con);
}
System.out.println("==========调用构造方法==========");
//调用无参构造方法
Constructor constructors1=cl.getConstructor();
Student student=(Student)constructors1.newInstance();
//调用有参构造方法
Constructor constructors2=cl.getConstructor(String.class);
Student student1=(Student) constructors2.newInstance("英语");
//调用私有构造方法
Constructor constructors3=cl.getDeclaredConstructor(int.class);
constructors3.setAccessible(true);//因为是私有化的方法,所以要设置访问权限
Student student2=(Student) constructors3.newInstance(2);
}
执行结果:
|
==========公开的构造方法========== |
3、获取方法
私有属性的调用,要注意要设置其访问权限Constructors1.setAccessible(true)。
private static void getMethod1() throws Exception {
Class cl=Class.forName("day20.Student");
//获取所有公开的方法,包括Object中的类
Method[] method=cl.getMethods();
for (Method m:method
) {
System.out.println(m.getName());
}
//获取本类中所有的方法,不包括父类和eObject
System.out.println("==========私有方法==========");
Method[] methods=cl.getDeclaredMethods();
for (Method ms:methods
) {
System.out.println(ms.getName());
}
//调取方法
System.out.println("==========调取特有方法==========");
//获取非静态的无参方法
Method method1=cl.getMethod("eat");
//获取Student类的成员
Constructor constructor=cl.getConstructor();
Student student=(Student)constructor.newInstance();
method1.invoke(student);
//获取非静态含参方法
Method method2=cl.getMethod("study", String.class);
method2.invoke(student,"物理");
//获取静态无参方法
Method method3=cl.getMethod("sleep");
method3.invoke(null);
//获取静态含参方法
try {
Method method4=cl.getMethod("saveMoney", double.class);
method4.setAccessible(true);
method4.invoke(null,100.0);
}catch (NoSuchMethodException e){
e.getMessage();
}catch(Exception e){
e.getMessage();
}
}
执行结果:
|
getName |
4、获取属性
private static void getAttribute1() throws Exception{
Class cl=Class.forName("day20.Student");
//获取所有属性
Field[] fields=cl.getFields();
for (Field field:fields
) {
System.out.println("属性值有:"+field.getName());
}
//获取私有化属性
Field field=cl.getDeclaredField("money");
field.setAccessible(true);
//获取Student类的成员
Constructor constructor=cl.getConstructor();
Student student=(Student)constructor.newInstance();
student.setMoney(100);
System.out.println("通过对象.方法进行属性赋值:"+student.getMoney());
field.set(student,200);
System.out.println("通过Field对象.方法进行赋值:"+student.getMoney());
}
执行结果:
|
属性值有:name |
Student类和父类
package day20;
public class Student extends Human{
public String name;
int age;
private double money;
public Student() {
System.out.println("无参构造方法");
}
public Student(String subject) {
System.out.println("有参构造方法"+subject);
}
private Student(int i) {
System.out.println("私有构造方法"+i);
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public double getMoney() {
return money;
}
public void setMoney(double money) {
this.money = money;
}
public void eat(){
System.out.println("吃饭");
}
public void study(String subject){
System.out.println("学习"+subject);
}
public static void sleep(){
System.out.println("睡觉");
}
private static void saveMoney(double money){
System.out.println("存款"+money);
}
}
package day20;
public class Human {
private String sex;
public Human(){
}
private Human(int i){
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
}
java 类加载机制和反射机制的更多相关文章
- Java进阶之reflection(反射机制)——反射概念与基础
反射机制是Java动态性之一,而说到动态性首先得了解动态语言.那么何为动态语言? 一.动态语言 动态语言,是指程序在运行时可以改变其结构:新的函数可以引进,已有的函数可以被删除等结构上的变化.比如常见 ...
- java的泛型与反射机制
什么是泛型? 泛型,即“参数化类型”.顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参) ...
- Java 核心类库之反射机制
1:什么是反射机制? 2:反射机制它可以做什么呢? 3:反射机制对应的API又是什么? 1):通过反射机制来获取一个对象的全限定名称(完整包名),和类名: 2):实例化Class对象 3):获取对象的 ...
- Java类加载和类反射回顾
今天学习Spring,突然想重新复习一下Java类加载和类反射的.巩固一下底层原理.部分参考了李刚老师的<疯狂Java讲义>和陈雄华.林开雄的<Spring3.x企业应用开发实战&g ...
- JAVA(五)反射机制/Annotation
成鹏致远 | lcw.cnblog.com |2014-02-04 反射机制 1.认识Class类 在正常情况下,必须知道一个类的完整路径之后才可以实例化对象,但是在 java中也允许通过一个对象来找 ...
- java.lang.Class<T> -- 反射机制及动态代理
Interface : Person package java_.lang_.component.bean; public interface Person { String area = " ...
- java的RTTI和反射机制
RTTI,即Run-Time Type Identification,运行时类型识别.RTTI能在运行时就能够自动识别每个编译时已知的类型. 很多时候需要进行向上转型,比如Base类派生出Derive ...
- java.lang.Class<T> -- 反射机制
1反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为jav ...
- 在JAVA中,关于反射机制的讨论
一.什么是反射机制 简单的来说,反射机制指的是程序在运行时能够获取自身的信息.在java中,只要给定类的名字, 那么就可以通过反射机制来获得类的所有信息. 二.哪里用到反射机制 ...
随机推荐
- 【Hadoop 分布式部署 七: 使用ntp配置内网中集群机器时间同步 】
集群的时间要同步,如果时间不同步,会出现很多问题. 找一台机器做时间服务器 所有的机器与这台机器的时间进行定时的同步 比如,每日十分钟同步一次 我们这里使用 hadoop-senior.zuoyan ...
- Ubuntu16.04更新记
大概一周前因为不可抗因素,我再次安装了Ubuntu16.04LTS 对于之前发誓不想再用Ubuntu的我,我只想说一句:真香 写一点我现在Ubuntu的配置,方面自己以后查看,也方便如果有相同需求的人 ...
- EF Code First 整不明白 继续完善
1.Add-Migration RenameDesc 要修改列名先用这个,然后把要修改的列名手动修改一下. 多出这个文件 public partial class RenameDesc : DbM ...
- 【Python】【异步IO】
# [[异步IO]] # [协程] '''协程,又称微线程,纤程.英文名Coroutine. 协程的概念很早就提出来了,但直到最近几年才在某些语言(如Lua)中得到广泛应用. 子程序,或者称为函数,在 ...
- Android Studio 使用USB真机调试教程
允许安装未知来源的软件 允许USB调试 设置启动方式 选择USB device 然后运行 会自动安装软件启动! 参考: https://blog.csdn.net/fubo1990/article/d ...
- Python 循环与定义函数
break for i in range(10): if i == 2: break print i 0 1 continue for i in range(10): if i == 2: conti ...
- PHP中封装Redis购物车功能
<?php // 服务层 namespace Common\Service; use Vendor\Func\Red; class CartService extends CommonServi ...
- oracle 12c创建可插拔数据库(PDB)与用户详解
前言 由于oracle 12c使用了CDB-PDB架构,类似于docker,在container-db内可以加载多个pluggable-db,因此安装后需要额外配置才能使用. 一.修改listener ...
- ZZNU 2095 : 我只看看不写题
把所有时间加起来,最后从大到小排序,一定要把大的先减去.注意花费的时间都是1,这一秒用过就不能再用了,所有用到了并查集的部分知识 #include<iostream> #include&l ...
- VC.判断双字节字符集前导字节集(IsDBCSLeadByte)
ZC:这是 WIndows API 函数 1.“BOOL IsDBCSLeadByte( char );” 判断 某字节是否在 双字节字符集的前导字节集中 ZC:可以判断 如 汉字.日文.韩文等 Z ...