动态代理

2019-04-22  本文已影响0人  树心图物

先说静态代理,静态代理就是设计模式中的代理模式,形式上与装饰模式一致,但是涵义不同,装饰模式注重新加的修饰功能,代理模式注重对原对象的方法控制行为的改变。
静态代理需要编译时就将接口、实现类、代理类一股脑儿全部手动完成,但如果我们需要很多的代理,每一个都这么手动的去创建实属浪费时间,而且会有大量的重复代码,此时我们就可以采用动态代理,动态代理可以在程序运行期间根据需要动态的创建代理类及其实例,来完成具体的功能,相比于静态代理来说可以在调用时决定要对什么类进行代理,对代理类的具体方法更不依赖。

动态代理:
需要被代理的类target,实现InvocationHandler接口的代理类,Proxy.newProxyInstance(ClassLoader, target.class.getInterfaces(), handler)
例如

1. 接口:

public interface Person {
    void searchHouse();
}

2. 被代理类Master,实现上述接口:

public class Master implements Person {    
    void searchHouse() {
    System.out.println("我是主人,我要找房子,一室一厅!");
    }
}

3. 代理类HomeLine,实现InvocationHandler接口

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class HomeLink implements InvocationHandler{
    
    private Person target;
    
    public Object getInstance(Person target){
        this.target = target;
        Class clazz = target.getClass();
        Object obj = Proxy.newProxyInstance(clazz.getClassLoader(), clazz.getInterfaces(), this);
    //Object obj = Proxy.newProxyInstance(clazz.getClassLoader(), new Class[] {clazz}, this);
        return obj;
    }

    @Override
    public Object invoke(Object obj, Method method, Object[] args) throws Throwable {
        System.out.println("我是链家,我帮别人找房子..");
        //第一个参数是target,也就是被代理类的对象;第二个参数是方法中的参数
        method.invoke(target, args);
        System.out.println("我是链家,已经找完了..");
        return null;
    }
}

4.测试用例:

public class TestSearchHouse {
    public static void main(String[] args) {
        Person person = (Person) new HomeLink().getInstance(new Master());
        person.searchHouse();
    }
}

5. 结果:

我是链家,我帮别人找房子..
我是主人,我要找房子,一室一厅!
我是链家,已经找完了..
上一篇下一篇

猜你喜欢

热点阅读