zoukankan      html  css  js  c++  java
  • day3 反射与动态代理

    import java.lang.reflect.Constructor;
    import java.lang.reflect.Field;
    import java.lang.reflect.Method;
    import java.util.ArrayList;
    import java.util.List;
    
    import org.junit.Before;
    import org.junit.Test;
    
    public class MyReflect {
        public String className = null;
        @SuppressWarnings("rawtypes")
        public Class personClass = null;
        /**
         * 反射Person类
         * @throws Exception 
         */
        @Before
        public void init() throws Exception {
            className = "cn.itcast_04_reflect.Person";
            personClass = Class.forName(className);
        }
        /**
         *获取某个class文件对象
         */
        @Test
        public void getClassName() throws Exception {
            System.out.println(personClass);
        }
        /**
         *获取某个class文件对象的另一种方式
         */
        @Test
        public void getClassName2() throws Exception {
            System.out.println(Person.class);
        }
        /**
         *创建一个class文件表示的实例对象,底层会调用空参数的构造方法
         */
        @Test
        public void getNewInstance() throws Exception {
            System.out.println(personClass.newInstance());
        }
        /**
         *获取非私有的构造函数
         */
        @SuppressWarnings({ "rawtypes", "unchecked" })
        @Test
        public void getPublicConstructor() throws Exception {
            Constructor  constructor  = personClass.getConstructor(Long.class,String.class);
            Person person = (Person)constructor.newInstance(100L,"zhangsan");
            System.out.println(person.getId());
            System.out.println(person.getName());
        }
        /**
         *获得私有的构造函数
         */
        @SuppressWarnings({ "rawtypes", "unchecked" })
        @Test
        public void getPrivateConstructor() throws Exception {
            Constructor con = personClass.getDeclaredConstructor(String.class);
            con.setAccessible(true);//强制取消Java的权限检测
            Person person2 = (Person)con.newInstance("zhangsan");
            System.out.println("**"+person2.getName());
        }
        /**
         *访问非私有的成员变量
         */
        @SuppressWarnings({ "rawtypes", "unchecked" })
        @Test
        public void getNotPrivateField() throws Exception {
            Constructor  constructor  = personClass.getConstructor(Long.class,String.class);
            Object obj = constructor.newInstance(100L,"zhangsan");
            
            Field field = personClass.getField("name");
            field.set(obj, "lisi");
            System.out.println(field.get(obj));
        }
        /**
         *访问私有的成员变量
         */
        @SuppressWarnings({ "rawtypes", "unchecked" })
        @Test
        public void getPrivateField() throws Exception {
            Constructor  constructor  = personClass.getConstructor(Long.class);
            Object obj = constructor.newInstance(100L);
            
            Field field2 = personClass.getDeclaredField("id");
            field2.setAccessible(true);//强制取消Java的权限检测
            field2.set(obj,10000L);
            System.out.println(field2.get(obj));
        }
        /**
         *获取非私有的成员函数
         */
        @SuppressWarnings({ "unchecked" })
        @Test
        public void getNotPrivateMethod() throws Exception {
            System.out.println(personClass.getMethod("toString"));
            
            Object obj = personClass.newInstance();//获取空参的构造函数
            Method toStringMethod = personClass.getMethod("toString");
            Object object = toStringMethod.invoke(obj);
            System.out.println(object);
        }
        /**
         *获取私有的成员函数
         */
        @SuppressWarnings("unchecked")
        @Test
        public void getPrivateMethod() throws Exception {
            Object obj = personClass.newInstance();//获取空参的构造函数
            Method method = personClass.getDeclaredMethod("getSomeThing");
            method.setAccessible(true);
            Object value = method.invoke(obj);
            System.out.println(value);
    
        }
        /**
         *
         */
        @Test
        public void otherMethod() throws Exception {
            //当前加载这个class文件的那个类加载器对象
            System.out.println(personClass.getClassLoader());
            //获取某个类实现的所有接口
            Class[] interfaces = personClass.getInterfaces();
            for (Class class1 : interfaces) {
                System.out.println(class1);
            }
            //反射当前这个类的直接父类
            System.out.println(personClass.getGenericSuperclass());
            /**
             * getResourceAsStream这个方法可以获取到一个输入流,这个输入流会关联到name所表示的那个文件上。
             */
            //path 不以’/'开头时默认是从此类所在的包下取资源,以’/'开头则是从ClassPath根下获取。其只是通过path构造一个绝对路径,最终还是由ClassLoader获取资源。
            System.out.println(personClass.getResourceAsStream("/log4j.properties"));
            System.out.println(personClass.getResourceAsStream("log4j.properties"));
            
            //判断当前的Class对象表示是否是数组
            System.out.println(personClass.isArray());
            System.out.println(new String[3].getClass().isArray());
            
            //判断当前的Class对象表示是否是枚举类
            System.out.println(personClass.isEnum());
            System.out.println(Class.forName("cn.itcast_04_reflect.City").isEnum());
            
            //判断当前的Class对象表示是否是接口
            System.out.println(personClass.isInterface());
            System.out.println(Class.forName("cn.itcast_04_reflect.TestInterface").isInterface());
            
            
        }
    
    }

    .1.1.    动态代理

             在之前的代码调用阶段,我们用action调用service的方法实现业务即可。

             由于之前在service中实现的业务可能不能够满足当先客户的要求,需要我们重新修改service中的方法,但是service的方法不只在我们这个模块使用,在其他模块也在调用,其他模块调用的时候,现有的service方法已经能够满足业务需求,所以我们不能只为了我们的业务而修改service,导致其他模块授影响。

             那怎么办呢?

             可以通过动态代理的方式,扩展我们的service中的方法实现,使得在原油的方法中增加更多的业务,而不是实际修改service中的方法,这种实现技术就叫做动态代理。

             动态代理:在不修改原业务的基础上,基于原业务方法,进行重新的扩展,实现新的业务。

             例如下面的例子:

    1、  旧业务

    买家调用action,购买衣服,衣服在数据库的标价为50元,购买流程就是简单的调用。

    2、  新业务

    在原先的价格上可以使用优惠券,但是这个功能在以前没有实现过,我们通过代理类,代理了原先的接口方法,在这个方法的基础上,修改了返回值。

    PS:我的印象里动态代理必须实现接口

    传统的方法

    package cn.itcast_05_proxy.service;
    /**
     * 这是一个业务的接口,这个接口中的业务就是返回衣服的价格
     * @author wilson
     *
     */
    public interface IBoss {//接口
        int yifu(String size);
    }
    package cn.itcast_05_proxy.service.impl;
    
    import cn.itcast_05_proxy.service.IBoss;
    
    
    /**
     * 实现了卖衣服的接口
     * 自定义了自己的业务,卖裤子
     * @author wilson
     *
     */
    public class Boss implements IBoss{
        public int yifu(String size){
            System.err.println("天猫小强旗舰店,老板给客户发快递----衣服型号:"+size);
            //这件衣服的价钱,从数据库读取
            return 50;
        }
        public void kuzi(){
            System.err.println("天猫小强旗舰店,老板给客户发快递----裤子");
        }
    }
    package cn.itcast_05_proxy.action;
    
    import org.junit.Test;
    
    import cn.itcast_05_proxy.service.IBoss;
    import cn.itcast_05_proxy.service.impl.Boss;
    
    public class SaleAction {
        /**
         * 不使用代理,直接调用方法
         * 方法中规定什么业务,就只能调用什么业务,规定什么返回值,就只能输出什么返回值
         */
        @Test
        public void saleByBossSelf() throws Exception {
            IBoss boss = new Boss();
            System.out.println("老板自营!");
            int money = boss.yifu("xxl");// 老板自己卖衣服,不需要客服,结果就是没有聊天记录
            System.out.println("衣服成交价:" + money);
        }
    }

    老板自营!
    天猫小强旗舰店,老板给客户发快递----衣服型号:xxl
    衣服成交价:50

     

    动态代理基础上修改

    package cn.itcast_05_proxy.action;
    
    import org.junit.Test;
    
    import cn.itcast_05_proxy.proxyclass.ProxyBoss;
    import cn.itcast_05_proxy.service.IBoss;
    import cn.itcast_05_proxy.service.impl.Boss;
    
    /**
     * 什么是动态代理? 简单的写一个模板接口,剩下的个性化工作,好给动态代理来完成!
     */
    public class ProxySaleAction {
        
        /**
         *使用代理,在这个代理中,只代理了Boss的yifu方法
         *定制化业务,可以改变原接口的参数、返回值等
         */
        @Test
        public void saleByProxy() throws Exception {
            IBoss boss = ProxyBoss.getProxy(10,IBoss.class,Boss.class);// 将代理的方法实例化成接口
            //IBoss boss = new Boss();// 将代理的方法实例化成接口
            System.out.println("代理经营!");
            int money = boss.yifu("xxl");// 调用接口的方法,实际上调用方式没有变
            System.out.println("衣服成交价:" + money);
        }
    }
    package cn.itcast_05_proxy.proxyclass;
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    public class ProxyBoss {
        /**
         * 对接口方法进行代理
         */
        @SuppressWarnings("unchecked")
        public static <T> T getProxy(final int discountCoupon,
                final Class<?> interfaceClass, final Class<?> implementsClass)
                throws Exception {
            return (T) Proxy.newProxyInstance(interfaceClass.getClassLoader(),
                    new Class[] { interfaceClass }, new InvocationHandler() {
                        public Object invoke(Object proxy, Method method,
                                Object[] args) throws Throwable {
                            Integer returnValue = (Integer) method.invoke(
                                    implementsClass.newInstance(), args);// 调用原始对象以后返回的值
                            return returnValue - discountCoupon;
                        }
                    });
        }
    }

    代理经营!
    天猫小强旗舰店,老板给客户发快递----衣服型号:xxl
    衣服成交价:40

     
  • 相关阅读:
    eclipse中开发android程序时,打开layout配置文件自动关闭的问题
    成功用WAP登陆ZBlog发表文章
    java环境变量配置
    Eclipse快捷键大全
    Android SDK 2.2 开发环境搭建
    用EnterpriseLibrary来自动管理与数据库的连接
    一个普通网站发展成大型网站过程中的架构演变史
    有关Silverlight TabControl组件的研究
    有关Silverlight浮动窗体组件的研究
    强大的DataGrid组件[9]_自定义头模板
  • 原文地址:https://www.cnblogs.com/bee-home/p/7846167.html
Copyright © 2011-2022 走看看