反射

概述

JAVA反射机制是在运行状态中,获取任意一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性;这种在运行状态动态获取信息以及动态调用对象方法的功能被称为java语言的反射机制。

类加载器

Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分,负责动态加载Java类到Java虚拟机的内存空间中。

java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、AppClassLoader。

BootstrapClassLoader(引导启动类加载器):
嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负载加载JAVA_HOME/lib下的类库,引导启动类加载器无法被应用程序直接使用。

ExtensionClassLoader(扩展类加载器):
ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类库。它的父加载器是BootstrapClassLoader

App ClassLoader(应用类加载器): App
ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文件。它的父加载器为ExtensionClassLoader

在这里插入图片描述

双亲委派模型:如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类)时,子类加载器才会尝试自己去加载。委派的好处就是避免有些类被重复加载。

得到class的三种方式

public class Demo2 {
    public static void main(String[] args) throws ClassNotFoundException {
        //第一种方式   通过包名.类名.class 加载类
        Class<Person> c1 = com.java.fanshe.Person.class;
        System.out.println(c1);

        //第二种方式 通过类的对象获取类得信息
        Person p =new Person();
        Class<Person> c2 = (Class<Person>) p.getClass();
        System.out.println(c2);

        System.out.println(c1 == c2);
        
        //第三种方式
        Class<Person> c3 = (Class<Person>) Class.forName("com.java.fanshe.Person");
        System.out.println(c3);
        System.out.println(c1 == c2 && c1 == c3);
    }
}

结果如下:
class com.java.fanshe.Person
class com.java.fanshe.Person
true
class com.java.fanshe.Person
true

上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存 ! 如果类已经在内存中存在, 不会重复加载, 而是重复利用

获取构造对象Constructor

1、通过指定的参数类型, 获取指定的单个构造方法

getConstructor(参数类型的class对象数组)

例如:
构造方法如下: Person(String name,int age)
得到这个构造方法的代码如下:
Constructor c = p.getClass().getConstructor(String.class,int.class);

2、获取构造方法数组

getConstructors();

3、 获取所有权限的单个构造方法

getDeclaredConstructor(参数类型的class对象数组)

4、 获取所有权限的构造方法数组

getDeclaredConstructors();

Constructor 创建对象

常用方法:

newInstance(Object... para)

调用这个构造方法, 把对应的对象创建出来

参数: 是一个Object类型可变参数, 传递的参数顺序 必须匹配构造方法中形式参数列表的顺序!

setAccessible(boolean flag)

如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

获取方法Method

  1. getMethod(String methodName , class… clss)
    根据参数列表的类型和方法名, 得到一个方法(public修饰的)
  2. getMethods();
    得到一个类的所有方法 (public修饰的)
  3. getDeclaredMethod(String methodName , class… clss)
    根据参数列表的类型和方法名, 得到一个方法(除继承以外所有的:包含私有, 共有, 保护, 默认)
  4. getDeclaredMethods();
    得到一个类的所有方法 (除继承以外所有的:包含私有, 共有, 保护, 默认)

执行方法

invoke(Object o,Object… para) :
调用方法 ,
参数1. 要调用方法的对象
参数2. 要传递的参数列表

getName()
获取方法的方法名称
setAccessible(boolean flag)
如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

获取属性Field

  1. getDeclaredField(String filedName)
    根据属性的名称, 获取一个属性对象 (所有属性)
  2. getDeclaredFields()
    获取所有属性
  3. getField(String filedName)
    根据属性的名称, 获取一个属性对象 (public属性)
  4. getFields()
    获取所有属性 (public)
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;

public class Demo5 {
    public static void main(String[] args) throws Exception {
        Class c = Class.forName("com.java.fanshe.Person");
        Constructor ct = c.getConstructor();
        Object o = ct.newInstance();

        Field phoneNumber = c.getField("phoneNumber");
        phoneNumber.set(o,"13843838438");


        //访问带权限的
        Field name = c.getDeclaredField("name");
        name.setAccessible(true);
        name.set(o,"张三");
        System.out.println(o);
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值