反射

不管现实多么惨不忍睹,都要持之以恒地相信,这只是黎明前短暂的黑暗而已。不要惶恐眼前的难关迈不过去,不要担心此刻的付出没有回报,别再花时间等待天降好运。真诚做人,努力做事!你想要的,岁月都会给你。反射,希望对大家有帮助,欢迎收藏,转发!站点地址:www.bmabk.com,来源:原文

11.反射
11.1.概念
  • 框架:半成品软件。可以在框架的基础上进行软件开发,简化编码

  • 反射:框架设计的灵魂,将类的各个组成部分封装为其他对象,就是反射机制

  • 好处:
    1. 可以在程序运行过程中,操作这些对象。
    2. 可以解耦,提高程序的可扩展性。
  • 反射的使用方式,要先获取该类的Class对象,然后使用其方法获取构造器,方法,变量后,对这些再使用反射类里的方法进行操作。
11.2.获取Class对象的方式
  1. Class.forName(“全类名”):将字节码文件加载进内存,返回Class对象,多用于配置文件,将类名定义在配置文件中。读取文件,加载类

  2. 类名.class:通过类名的属性class获取,多用于参数的传递

  3. 对象.getClass():getClass()方法在Object类中定义。多用于对象的获取字节码的方式

    同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。

public class Reflect {

    /**
        获取Class对象的方式:
            1. Class.forName("全类名"):将字节码文件加载进内存,返回Class对象
            2. 类名.class:通过类名的属性class获取
            3. 对象.getClass():getClass()方法在Object类中定义着。

     */
    public static void main(String[] args) throws Exception {

        //1.Class.forName("全类名")
        Class cls1 = Class.forName("cn.itcast.domain.Person");
        System.out.println(cls1);
        //2.类名.class
        Class cls2 = Person.class;
        System.out.println(cls2);
        //3.对象.getClass()
        Person p = new Person();
        Class cls3 = p.getClass();
        System.out.println(cls3);

        //== 比较三个对象
        System.out.println(cls1 == cls2);//true
        System.out.println(cls1 == cls3);//true


        Class c = Student.class;
        System.out.println(c == cls1);
    }
}

public class Person {
    private String name;
    private int age;

    public Person() {}

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    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;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
11.3.java代码的三个阶段
  1. sourse源代码阶段:java源代码编译为.class字节码文件

  2. class类对象阶段:class文件通过类加载器加载进内存,成员变量,构造方法,成员方法均被封装为对象,通过数组存储,

  3. runtime运行阶段:通过new创建对象

11.4.Class对象功能
  1. 获取成员变量们

    • public Field[] getFields() :获取所有 public 修饰的成员变量
    • public Field getField(String name): 获取指定名称的 public 修饰的成员变量
    • public Field[] getDeclaredFields():获取所有的成员变量,不考虑修饰符
    • public Field getDeclaredField(String name):获取指定名称的成员变量,不考虑修饰符
  2. 获取构造方法们

    • public Constructor<?>[] getConstructors():获取所有 public 修饰的构造方法

    • public Constructor<T> getConstructor(class<?>... parameterTypes) :获取指定名称的 public 修饰的构造方法

    • public Constructor<?>[] getDeclaredConstructors():获取所有的构造方法,不考虑修饰符

    • public Constructor<T> getDeclaredConstructor(class<?>... parameterTypes):获取指定名称的构造方法,不考虑修饰符

  3. 获取成员方法们:

    • public Method[] getMethods():获取所有 public 修饰的成员方法

    • public Method getMethod(String name, class<?>... parameterTypes):获取指定名称的 public 修饰的成员方法

    • public Method[] getDeclaredMethods():获取所有的成员方法,不考虑修饰符

    • public Method getDeclaredMethod(String name, class<?>... parameterTypes):获取指定名称的成员方法,不考虑修饰符

  4. 获取全类名

    • public String getName():返回此 Class 对象所表示的实体(类、接口、数组类、基本类型或 void)名称。
11.5.反射方法介绍
  • java.lang.reflect.AccessibleObject类,是反射的根类,有三个已知子类,分别为Constructor(构造器),Field(变量),Method(方法)

    • AccessibleObject类下有个常用方法:
      • 暴力反射:public void setAccessible(true):忽略访问权限修饰符的安全检查,除public外都需要设置
  • Field:成员变量

    1. 设置值:public void set(Object obj, Object value):将指定对象变量上此 Field 对象表示的字段设置为指定的新值

    2. 获取值:public void get(Object obj):返回指定对象上此 Field 表示的字段的值

  • Constructor:构造器

    1. 创建对象:public T newInstance(Object... initargs):创建该构造方法的声明类的新实例,并用指定的初始化参数初始化该实例
    2. 如果使用空参数构造方法创建对象,操作可以简化:Class对象的newInstance方法
      • public T newInstance():创建此 Class 对象所表示的类的一个新实例。
  • Method:方法对象

    1. 执行方法:public Object invoke(Object obj, Object... args):对带有指定参数的指定对象调用由此 Method 对象表示的底层方法

    2. 获取方法名称:public String getName():返回此 Method 对象表示的方法名称。

  • Example:

public class Person {
    private String name;
    private int age;

    public String a;
    protected String b;
    String c;
    private String d;


    public Person() {
    }

    public Person(String name, int age) {

        this.name = name;
        this.age = age;
    }

    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;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", a='" + a + '\'' +
                ", b='" + b + '\'' +
                ", c='" + c + '\'' +
                ", d='" + d + '\'' +
                '}';
    }

    public void eat(){
        System.out.println("eat...");
    }

    public void eat(String food){
        System.out.println("eat..."+food);
    }
}


public class ReflectDemo {


    public static void main(String[] args) throws Exception {

        //0.获取Person的Class对象
        Class personClass = Person.class;

        //1.Field[] getFields()获取所有public修饰的成员变量
        Field[] fields = personClass.getFields();
        for (Field field : fields) {
            System.out.println(field);
        }

        System.out.println("------------");
        //2.Field getField(String name)
        Field a = personClass.getField("a");
        //获取成员变量a 的值
        Person p = new Person();
        Object value = a.get(p);
        System.out.println(value);
        //设置a的值
        a.set(p,"张三");
        System.out.println(p);

        System.out.println("===================");

        //Field[] getDeclaredFields():获取所有的成员变量,不考虑修饰符
        Field[] declaredFields = personClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println(declaredField);
        }
        //Field getDeclaredField(String name)
        Field d = personClass.getDeclaredField("d");
        //忽略访问权限修饰符的安全检查
        d.setAccessible(true);//暴力反射
        Object value2 = d.get(p);
        System.out.println(value2);
        
        
        
        
    
    	//0.获取Person的Class对象
        Class personClass = Person.class;

        //Constructor<T> getConstructor(类<?>... parameterTypes)
        Constructor constructor = personClass.getConstructor(String.class, int.class);
        System.out.println(constructor);
        //创建对象
        Object person = constructor.newInstance("张三", 23);
        System.out.println(person);

        System.out.println("----------");

        Constructor constructor1 = personClass.getConstructor();
        System.out.println(constructor1);
        //创建对象
        Object person1 = constructor1.newInstance();
        System.out.println(person1);

        Object o = personClass.newInstance();
        System.out.println(o);

        //constructor1.setAccessible(true);
    
        
        
        
        

    	//0.获取Person的Class对象
        Class personClass = Person.class;

        //获取指定名称的方法
        Method eat_method = personClass.getMethod("eat");
        Person p = new Person();
        //执行方法
        eat_method.invoke(p);


        Method eat_method2 = personClass.getMethod("eat", String.class);
        //执行方法
        eat_method2.invoke(p,"饭");

        System.out.println("-----------------");

        //获取所有public修饰的方法
        Method[] methods = personClass.getMethods();
        for (Method method : methods) {
            System.out.println(method);
            String name = method.getName();
            System.out.println(name);
            //method.setAccessible(true);
        }

        //获取类名
        String className = personClass.getName();
        System.out.println(className);//cn.itcast.domain.Person
    }
}
11.6.静态的获取和使用
  • 静态方法和静态变量的获取与普通的没有任何区别,但是在使用上有一点区别
public class Demo {

    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
        // 获取目标类对应的Class对象
        Class clz = StaticClass.class;

        // 获取静态的目标成员变量
        //accessStaticField(clz);

        // 静态方法
        Method staticMethod = clz.getDeclaredMethod("staticMethod");
        staticMethod.setAccessible(true);
        staticMethod.invoke(null);
    }

    private static void accessStaticField(Class clz) throws NoSuchFieldException, IllegalAccessException {
        Field staticField = clz.getDeclaredField("staticField");

        // 访问静态成员变量
        staticField.setAccessible(true);
        int a = (int) staticField.get(null);
        System.out.println(a);

        // 修改静态成员变量的值
        staticField.set(null, 900);
        a = (int) staticField.get(null);
        System.out.println(a);
    }
}

class StaticClass {
    private static int staticField = 100;
    private static void staticMethod() {
        System.out.println("static method");
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/181070.html

(0)
飞熊的头像飞熊bm

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!