美文网首页
反射以及动态代理

反射以及动态代理

作者: 暴躁的心 | 来源:发表于2019-11-21 11:55 被阅读0次

    反射

      反射之中包含了一个'反'字,那就肯定有正,那我们就先从'正'开始
    一般情况下,我们使用类的时候必定知道他是什么类,是用来做什么的,于是我们直接对这个类进行实例化,之后使用类的对象进行操作。
      反射则是一开始我们并知道我要初始化的这个类的对象是什么,自然也无法使用new关键字来创建对象了,这个时候,我们就要引入JDK 给我们提供的反射api 进行反射调用,反射就是在运行时才知道要操作的类是什么,并且可以再运行时获取类的完整构造,并调用对应的方法。
    Reflection(反射)是java被视为动态语言的关键,反射几只允许程序在执行期借助Reflection API 取得任何类的内部信息,并直接操作任意对象的内部属性及方法。

    Java反射机制主要提供了一下功能

    • 在运行时构造任意一个类的对象
    • 在运行时获取任意一个类所具有的成员变量和方法
    • 在运行时调用任意一个对象的方法(属性)

      java是一门面向对象的语言,在面向对象的世界里,万事万物皆对象,既然万事万物皆对象,那么我们的类是不是对象呢?我们写的每一个类都可以看成一个对象,是java.lang.Class类的对象,每一个类对应的Class放在哪里呢?当我们写完一个类的java文件,编译成class文件的时候,编译期都会将这个类的对应的calss对象放在class的末尾,里面都保存了些什么?大家可以理解保存了类的元数据信息,一个类的元数据信息包括什么?有哪些属性,方法,构造器,实现了哪些接口等等,那么这些信息在java里都有对应的类来表示。

    Class

    Class是一个类,封装了当前对象多对应的类的信息

      一个类中有属性,方法,构造器等,比如说有一个person类,一个Order类,一个Book类,这些都是不同的类,现在需要一个称呼用来描述类,这就是Class,他应该有类名,属性,方法,构造器等,Class是用来描述类的类(这个有点拗口)
      Class类就像是对象本身自己照镜子一样,可以看到自己的全部,有哪些属性,方法,构造器,实现了哪些接口等
      对于每个类而且,JRE都为期保留一个不变的Class类型的对象,一个Class对象包含了特定某个类的有关信息。
      对象只能由系统建立对象,一个类(而不是一个对象)在JVM中只会有一个Class实例

    获取Class对象的三种方式
    名称 方式
    1.通过类名获取 类型.class
    2.通过对象获取 对象名.getClass()
    3.通过全类名获取 Class.forName(全类名)
            //实例化对象的标准用法,也就是所谓的正
            Servant servant = new Servant();
            //获取class三种方法
            Class servantClass = Servant.class;
            Class servantClass1 = servant.getClass();
            Class servantClass3 = Class.forName("com.company.Reflect.Servant");
    
    Class类的常用方法
    方法

    接下来我们就来使用反射来获取属性,方法,构造器等。

    ps:因为都是固定的调一些API 此处就直接上代码了

    public class Person {
        String name;
        private int age;
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
            System.out.println("this is setName()");
        }
        public int getAge() {
            return age;
        }
        public void setAge(int age) {
            this.age = age;
            System.out.println("this is setAge()");
        }
        //包含一个带参的构造器和一个不带参的构造器
        public Person(String name, int age) {
            super();
            this.name = name;
            this.age = age;
        }
        public Person() {
            super();
        }
        private void privateMethod(){
            System.out.println("this is private method");
        }
    }
    
    构造器相关
    内容 方法
    获取全部constructor对象 getConstructors()
    获取某一个constructor对象 需要传入参数列表 getConstructor(Object...)
    调用构造器的 newInstance() 方法创建对象 有参数需传入 构造器.newInstance(Object...)
    public class TestConstructor {
        /*构造器相关*/
        public void testConstructor() throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
            String className ="com.company.Reflect.Person";
            Class<Person> clazz = (Class<Person>) Class.forName(className);
            System.out.println("获取全部constructor对象");
            Constructor<Person>[] constructors = (Constructor<Person>[]) clazz.getConstructors();
            for (Constructor<Person> constructor:constructors) {
                System.out.println(constructor);
            }
    
            System.out.println("获取某一个constructor对象 需要参数列表-----");
            Constructor<Person> constructor = clazz.getConstructor(String.class, int.class);
            System.out.println(constructor);
    
            System.out.println("调用构造器的 newInstance() 方法创建对象-----");
            Person hellow = constructor.newInstance("hellow", 18);
            System.out.println(hellow.getName());
        }
    }
    
    

    接下来我们看一下打印结果


    打印结果
    方法相关
    内容 方法
    获取对应类中的所有方法,不能获取private方法,且获取从父类继承来的所有方法 getMethods()
    获取所有方法,包括私有方法 所有声明的方法,都可以获取到,且只获取当前类的方法 getDeclaredMethods()
    获取指定的方法 需要参数名称和参数列表,无参数则不需要写 clazz.getDeclaredMethod(String name,Class<?>... paramterTypes)
    执行方法,第一个参数表示执行那个对象的方法,剩下的参数时执行方法时需要的参数 invoke(Object obj,Obkect... obj)
    执私有方法的执行,必须在调用invoke之前加上一句 method.setAccessible(ture)不然会报错 setAccessible(ture)
    public class TestMethod {
        public void testMethod() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
            String className ="com.company.Reflect.Person";
            Class clazz =Class.forName(className);
            System.out.println("获取clazz中对应类中的素有方法 不能获取private方法,且获取从父类继承来的所有方法");
            Method[] methods = clazz.getMethods();
            for (Method method :methods) {
                System.out.println(" "+method.getName()+"()");
            }
            System.out.println("--------------------------");
            System.out.println("获取所有方法,包括私有方法 所有声明的方法,都可以获取到,且只获取当前类的方法");
            Method[] declaredMethods = clazz.getDeclaredMethods();
            for (Method method :declaredMethods) {
                System.out.println(" "+method.getName()+"()");
            }
            System.out.println("--------------------------");
            System.out.println("获取指定的方法 需要参数名称和参数列表,无参数则不需要写");
            Method method = clazz.getDeclaredMethod("setAge", int.class);
            System.out.println(method);
            System.out.println("--------------------------");
    
            System.out.println("执行方法,第一个参数表示执行那个对象的方法,剩下的参数时执行方法时需要的参数");
            Object instance = clazz.newInstance();
            method.invoke(instance,18);
    
            System.out.println("执私有方法的执行,必须在调用invoke之前加上一句 method.setAccessible(ture)");
            Method privateMethod = clazz.getDeclaredMethod("privateMethod");
            System.out.println(privateMethod);
            System.out.println("--------------------------");
            System.out.println("执行私有方法");
            privateMethod.setAccessible(true);
            privateMethod.invoke(instance);
    
        }
    
    打印结果

    当不加setAccessible(true)时的结果


    不加时报错结果
    属性相关
    内容 方法
    获取公用和私有的所有字段 但是不能获取父类的字段 getDeclaredFields()
    获取指定字段 getDeclaredField(String name)
    获取指定字段的值 字段.get(Object obj)
    设置指定对象指定字段的值 字段.set(Object obj)
    字段是私有的,不管是读还是写都必须先调用 setAccessible(ture) 方法 setAccessible(true)

    动态模式

    代理模式和静态代理

      代理模式 给某一个对象提供一个代理对象,并有代理对象控制对原对象的引用,通俗的来讲代理模式就是我们生活中常见的中介
      举个例子:张三想买某种生活用品,虽然他可以自己去找,但是有点浪费时间和精力,或者自己不好意思去买,于是张三就通过中介Mark来买,Mark来帮张三买一个,张三只是负责选择喜欢的尺寸,然后付钱给Mark就可以了

    目的:
    1.通过引入代理对象的方式来间接访问目标对象,防止直接访问目标对象给系统带来的不必要复杂性;
    2.通过代理对象对原有的业务增强;
    代理模式一般会有三个角色:

    • 抽象角色:指代理角色和真实角色对外提供的公共方法,一般为一个接口
    • 真实角色:需要实现抽象角色接口,定义了真实角色所要实现的业务逻辑,以便供代理角色调用。也就是真正的业务逻辑在此。
    • 代理角色:需要实现抽象角色接口,是真实角色的代理,通过真实角色的业务逻辑方法来实现抽象方法,并可以附加自己的操作。将统一的流程控制都放到代理角色中处理!
        而访问者不再访问真实角色,而是去访问代理角色。
        静态代理在使用时,需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者是继承相同父类。一般来说,被代理对象和代理对象是一对一的关系,当然一个代理对象对应多个被代理对象也是可以的。
        静态代理,一对一则会出现时静态代理对象量多、代码量大,从而导致代码复杂,可维护性差的问题,一对多则代理对象会出现扩展能力差的问题。

    接下来我们用代码来简单的表述一下静态代理

    //代表真实角色
    public class AaFactory implements ManToolsFactory {
        @Override
        public void saleManTools(String size) {
            System.out.println("按需求定制了一个size为"+size+"的女model");
        }
    }
    //抽象角色
    public interface ManToolsFactory {
        void saleManTools(String size);
    }
    
    /*代理角色*/
    public class Mark1 implements ManToolsFactory {
        
        public ManToolsFactory factory;
    
        public Mark1(ManToolsFactory factory) {
            this.factory = factory;
        }
        
        /*后置处理器*/
        private void doSthAfter() {
            System.out.println("精美包装,快递一条龙服务");
        }
    
        /*前置处理器*/
        private void doSthBefore() {
            System.out.println("根据需求,进行市场调研和产品分析");
        }
    
        @Override
        public void saleManTools(String size) {
            doSthBefore();
            factory.saleManTools(size);
            doSthAfter();
        }
    }
    
    public static void main(String[] args) {
            ManToolsFactory factory = new AaFactory();
            Mark mark = new Mark(factory);
            mark.saleManTools("D");
        }
    
    
    静态代理运行结果

      通过运行我们发现确实完成了整个工作逻辑,但是有一个问题,如果这时张三的老婆来了,但是需求不同,需要另一种生活用品,这时怎么办呢?智能是再去增加一个新的抽象接口,去做新的逻辑,这样就造成了一个问题,需要不停去新增,不停的去修改,这就违反了面向对象的开闭原则。同时扩展能力差 可维护性差,所以在一般的正常使用中我们都是采用动态代理来实现。

    动态代理

      是指在使用时再创建代理类和实例
      优点
      只需要1个动态代理类就可以解决创建多个静态代理的问题,避免重复、多余代码,更强的灵活性
      缺点
      效率低,相比静态代理中 直接调用目标对象方法,动态代理则需要先通过Java反射机制 从而 间接调用目标对象方法
      应用场景局限,因为 Java 的单继承特性(每个代理类都继承了 Proxy 类),即只能针对接口 创建 代理类,不能针对类创建代理类。
      在java的动态代理机制中,有两个重要的类或接口,一个是InvocationHandler接口、另一个则是 Proxy类,这个类和接口是实现我们动态代理所必须用到的。
      InvocationHandler接口是给动态代理类实现的,负责处理被代理对象的操作的,而Proxy是用来创建动态代理类实例对象的,因为只有得到了这个对象我们才能调用那些需要代理的方法。

    下面简单的通过代码来实现一下

    public class AaFactory implements ManToolsFactory {
        @Override
        public void saleManTools(String size) {
            System.out.println("按需求定制了一个size为"+size+"的女model");
        }
    }
    
    public class BbFactory implements WomanToolsFactory {
        @Override
        public void saleWomanTools(float length) {
            System.out.println("按需求定制了一个高度为"+length+"的男model");
        }
    }
    
    public class MarkCompany implements InvocationHandler {
    
        /*持有的真实对象*/
        private Object factory;
    
        public Object getFactory() {
            return factory;
        }
    
        public void setFactory(Object factory) {
            this.factory = factory;
        }
    
        /*通过proxy获得动态代理对象*/
        public Object getProxyInstance(){
            return Proxy.newProxyInstance(factory.getClass().getClassLoader(),factory.getClass().getInterfaces(),this);
        }
    
    
        /*通过动态代理对象方法进行增强*/
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            doSthBefore();
            Object result = method.invoke(factory, args);
            doSthAfter();
            return result;
        }
    
        /*后置处理器*/
        private void doSthAfter() {
            System.out.println("精美包装,快递一条龙服务");
        }
    
        /*前置处理器*/
        private void doSthBefore() {
            System.out.println("根据需求,进行市场调研和产品分析");
        }
    }
    
    public interface ManToolsFactory {
        void saleManTools(String size);
    }
    
    public interface WomanToolsFactory {
    
        void saleWomanTools(float length);
    
    }
    
    public static void main(String[] args) {
            /*动态代理*/
            ManToolsFactory manToolsFactory = new AaFactory();
            MarkCompany company = new MarkCompany();
            company.setFactory(manToolsFactory);
            ManToolsFactory employee = (ManToolsFactory) company.getProxyInstance();
            employee.saleManTools("E");
    
            WomanToolsFactory womanToolsFactory = new BbFactory();
            company.setFactory(womanToolsFactory);
            WomanToolsFactory employee1 = (WomanToolsFactory) company.getProxyInstance();
            employee1.saleWomanTools(1.9f);
        }
    
    
    
    动态代理运行结果

    通过打印结果可以看到我们同样的实现了需求,但是工作量要比静态代理少,同时后期维护的话要更省事

      今天的介绍,对于庞大的java来说只是冰山一角,只是通过了一个小的demo来介绍一下反射和代理,所说的这些并不能面面俱到,只能对自己的学习所一个总结,以便于以后的查阅。

    相关文章

      网友评论

          本文标题:反射以及动态代理

          本文链接:https://www.haomeiwen.com/subject/mrnoictx.html