zoukankan      html  css  js  c++  java
  • 反射

         反射调用私有构造器创建对象
      Class c=hungrysingleton.class;
            Constructor constructor=c.getDeclaredConstructor();
            constructor.setAccessible(true);//入侵
            hungrysingleton h1= (hungrysingleton) constructor.newInstance();
    
    
    1、获取类(Class)对象
    	通过forName() -> 示例:Class.forName(“PeopleImpl”)
    	通过getClass() -> 示例:new PeopleImpl().getClass()
    	直接获取.class -> 示例:PeopleImpl.class
    2、类的常用方法
    getName():获取类完整方法;
    getSuperclass():获取类的父类;
    newInstance():创建实例对象;
    getFields():获取当前类和父类的public修饰的所有属性;
    getDeclaredFields():获取当前类(不包含父类)的声明的所有属性;
    getMethod():获取当前类和父类的public修饰的所有方法;
    getDeclaredMethods():获取当前类(不包含父类)的声明的所有方法;
    
    3、类方法调用
    
    1 静态方法调用
    // 核心代码(省略了抛出异常的声明)
    public static void main(String[] args) {
        Class myClass = Class.forName("example.PeopleImpl");
        // 调用静态(static)方法
        Method getSex = myClass.getMethod("getSex");
        getSex.invoke(myClass);
    }
    2 普通方法调用
    普通非静态方法调用,需要先获取类示例,通过“newInstance()”方法获取,核心代码如下:
    
    Class myClass = Class.forName("example.PeopleImpl");
    Object object = myClass.newInstance();
    Method method = myClass.getMethod("sayHi",String.class);
    method.invoke(object,"老王");
    getMethod 获取方法,可以声明需要传递的参数的类型。
    
    3 调用私有方法
    调用私有方法,必须使用“getDeclaredMethod(xx)”获取本类所有什么的方法,代码如下:
    
    Class myClass = Class.forName("example.PeopleImpl");
    Object object = myClass.newInstance();
    Method privSayHi = myClass.getDeclaredMethod("privSayHi");
    privSayHi.setAccessible(true); // 修改访问限制
    privSayHi.invoke(object);
    除了“getDeclaredMethod(xx)”可以看出,调用私有方法的关键是设置 setAccessible(true) 属性,修改访问限制,这样设置之后就可以
    
    进行调用了。
    
    
    1、JDK Proxy 动态代理
    JDK Proxy 是通过实现 InvocationHandler 接口来实现的,代码如下:
    
    interface Animal {
        void eat();
    }
    class Dog implements Animal {
        @Override
        public void eat() {
            System.out.println("The dog is eating");
        }
    }
    class Cat implements Animal {
        @Override
        public void eat() {
            System.out.println("The cat is eating");
        }
    }
    
    // JDK 代理类
    class AnimalProxy implements InvocationHandler {
        private Object target; // 代理对象
        public Object getInstance(Object target) {
            this.target = target;
            // 取得代理对象
            return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
        }
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("调用前");
            Object result = method.invoke(target, args); // 方法调用
            System.out.println("调用后");
            return result;
        }
    }
    
    public static void main(String[] args) {
        // JDK 动态代理调用
        AnimalProxy proxy = new AnimalProxy();
        Animal dogProxy = (Animal) proxy.getInstance(new Dog());
        dogProxy.eat();
    }
    如上代码,我们实现了通过动态代理,在所有请求之前和之后打印了一个简单的信息。
    
    注意: JDK Proxy 只能代理实现接口的类(即使是extends继承类也是不可以代理的)。
    
    
    2、Cglib 动态代理
    JDK 动态代理机制只能代理实现了接口的类,Cglib 是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实
    
    现增强,但因为采用的是继承,所以不能对 final 修饰的类进行代理。
    
    
    class Panda {
        public void eat() {
            System.out.println("The panda is eating");
        }
    }
    class CglibProxy implements MethodInterceptor {
        private Object target; // 代理对象
        public Object getInstance(Object target) {
            this.target = target;
            Enhancer enhancer = new Enhancer();
            // 设置父类为实例类
            enhancer.setSuperclass(this.target.getClass());
            // 回调方法
            enhancer.setCallback(this);
            // 创建代理对象
            return enhancer.create();
        }
        @Override
        public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
            System.out.println("调用前");
            Object result = methodProxy.invokeSuper(o, objects); // 执行方法调用
            System.out.println("调用后");
            return result;
        }
    }
    
    public static void main(String[] args) {
        // CGLIB 动态代理调用
        CglibProxy proxy = new CglibProxy();
        Panda panda = (Panda)proxy.getInstance(new Panda());
        panda.eat();
    }
    
    JDK Proxy 的优势:
    
    最小化依赖关系,减少依赖意味着简化开发和维护,JDK 本身的支持,更加可靠;
    平滑进行 JDK 版本升级,而字节码类库通常需要进行更新以保证在新版上能够使用;
    
    Cglib 框架的优势:
    
    可调用普通类,不需要实现接口;
    高性能;
    
    反射获取属性:
    			Class c=o.getClass();
    		 Field f=c.getDeclaredField(fieldname);
    		 f.setAccessible(true);
    		 Object dd=f.get(o);
    用反射完成对象属性复制:
    /**
         * 复制一个类对象属性到另一个类对象中
         * @param objA 需要复制的对象
         * @param objB 复制到的目标对象类型
         * @return 返回复制后的目标对象
         */
        private static void parseObj(Object objA,Object objB) throws Exception{
            if (objA == null){
                return;
            }
            //获取objA的类信息
            Class classA = objA.getClass();
            Class classB = objB.getClass();
            try {
                //获取objA的所有字段
                Field[] fieldsA = classA.getDeclaredFields();
                //获取objB的所有字段
                Field[] fieldsB = classB.getDeclaredFields();
                if (fieldsA == null || fieldsA.length <= 0 || fieldsB == null || fieldsB.length <= 0){
                    return;
                }
    
                //生成查询map
                Map<String,Field> fieldMap = new HashMap<>();
                for (Field field:fieldsA){
                    fieldMap.put(field.getName(),field);
                }
    
                //开始复制字段信息
                for (Field fieldB : fieldsB){
                    //查找是否在objB的字段中存在该字段
                    Field fielaA = fieldMap.get(fieldB.getName());
                    if (fielaA != null){
                        fieldB.setAccessible(true);
                        fieldB.set(objB,getFieldValue(objA,fielaA.getName()));
                    }
                }
            } catch (IllegalStateException e) {
                throw new IllegalStateException("instace fail: " ,e);
            }
        }
      public void test(){
            try{
                //生成Employee对象
                Employee employee = new Employee("Frank",6666.66);
                //生成一个Manager对象
                Manager manager = new Manager();
                //复制对象
                parseObj(employee,manager);
                System.out.println(manager.getName());
                System.out.println(manager.getSalary());
            }catch (Exception e){
                e.printStackTrace();
            }
        }
    
    
    	
    

      

  • 相关阅读:
    串匹配模式中的BF算法和KMP算法
    “隐藏与显示”的多种方法实现
    原生js实现tooltip提示框的效果
    心向旋转巧得木马 峰回路转偶得时钟
    jQuery 之 验证表单
    Java代码添加背景音乐
    svg动画 之 我的自制太阳系
    java_22 Map接口
    java_22.1 Map 的应用
    java_18 Collection接口
  • 原文地址:https://www.cnblogs.com/qinyios/p/11062162.html
Copyright © 2011-2022 走看看