设计模式之代理模式

什么是代理模式

    代理模式是指给某一个对象提供一个代理或占位符,并由代理对象来控制对原对象的访问。代理对象在客户端和目标对象之间起到中介的作用。
    代理模式包含以下几个角色:
        Subject(抽象主题角色):声明了真实主题和代理主题的共同接口,这样在任何使用真实主题的地方都可以使用代理主题。客户端通常需要针对抽象主题角色编程。
        Proxy(代理主题角色):包含了对真实主题的引用,从而可以在任何时候操作真实主题对象。在代理主题角色中提供一个与真实主题角色相同的接口,以便在任何时候都可以替代真实主题。代理主题角色还可以控制对真实主题的使用,负责在需要的时候创建和删除真实主题,并对真实主题对象的使用加以限制。
        RealSubject(真实主题角色):定义了代理角色所代表的真实对象,在真实主题角色中实现了真实的业务操作。
    java中代理分为静态代理和动态代理:
        静态代理:代理对象和被代理对象实现同一个接口,代理类在编译器就生成。
        动态代理:在java运行时动态生成。

代理模式的优缺点

优点
  1. 降低了系统耦合。
  2. 客户端可以针对抽象主题角色编程,增加或者替换代理类无需修改源代码,符合开闭原则。
  3. 静态代理实现简单。
缺点
  1. 代理类和被代理类必须实现同一个接口。
  2. 静态代理中,接口发生修改,代理类也要进行修改。

代理模式的应用场景

  1. 隐藏某个类或者类中的某些方法。

代理模式的案例

// 1. 静态代理

// 抽象主题角色
public interface ITeacherDao {
    void teach();
}
// 真实主题角色
public class TeacherDao implements ITeacherDao {
	@Override
	public void teach() {
		System.out.println(" 老师授课中  。。。。。");
	}
}
// 代理主题角色
public class TeacherDaoProxy implements ITeacherDao {

    private ITeacherDao target;

    public TeacherDaoProxy(ITeacherDao target) {
        this.target = target;
    }


    @Override
    public void teach() {
        System.out.println("开始代理  完成某些操作。。。。。 ");
        target.teach();
        System.out.println("提交。。。。。");
    }

}

public static void main(String[] args) {
	//创建目标对象(被代理对象)
	TeacherDao teacherDao = new TeacherDao();

	//创建代理对象, 同时将被代理对象传递给代理对象
	TeacherDaoProxy teacherDaoProxy = new TeacherDaoProxy(teacherDao);

	//通过代理对象,调用到被代理对象的方法
	//即:执行的是代理对象的方法,代理对象再去调用目标对象的方法
	teacherDaoProxy.teach();
}

/******************************************************************/

// 2. JDK动态代理

// 抽象主题角色
public interface ITeacherDao {

    void teach();

    void sayHello(String name);
}
// 真实主题角色
public class TeacherDao implements ITeacherDao {

	@Override
	public void teach() {
		System.out.println(" 老师授课中.... ");
	}

	@Override
	public void sayHello(String name) {
		System.out.println("hello " + name);
	}

}

// 代理主题角色
public class ProxyFactory {

    private Object target;

    public ProxyFactory(Object target) {

        this.target = target;
    }

    public Object getProxyInstance() {

        //说明
		/*
		 *  public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)

            //1. ClassLoader loader : 指定当前目标对象使用的类加载器, 获取加载器的方法固定
            //2. Class<?>[] interfaces: 目标对象实现的接口类型,使用泛型方法确认类型
            //3. InvocationHandler h : 事情处理,执行目标对象的方法时,会触发事情处理器方法, 会把当前执行的目标对象方法作为参数传入
		 */
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                (proxy, method, args) -> {
                    System.out.println("JDK代理开始~~");
                    //反射机制调用目标对象的方法
                    Object returnVal = method.invoke(target, args);
                    System.out.println("JDK代理提交");
                    return returnVal;
                });
    }

}


public static void main(String[] args) {
    //创建目标对象
    ITeacherDao target = new TeacherDao();

    //给目标对象,创建代理对象, 可以转成 ITeacherDao
    ITeacherDao proxyInstance = (ITeacherDao) new ProxyFactory(target).getProxyInstance();

    // proxyInstance=class com.sun.proxy.$Proxy0 内存中动态生成了代理对象
    System.out.println("proxyInstance=" + proxyInstance.getClass());

    //通过代理对象,调用目标对象的方法
    proxyInstance.teach();
    proxyInstance.sayHello(" tom ");
}

代理模式在源码中的应用

DefaultAopProxyFactory
// 在AOP代理类中根据代理对象是否实现接口判断是jdk动态代理还是cglib代理
public class DefaultAopProxyFactory implements AopProxyFactory, Serializable {

	private static final long serialVersionUID = 7930414337282325166L;


	@Override
	public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
		if (!NativeDetector.inNativeImage() &&
				(config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config))) {
			Class<?> targetClass = config.getTargetClass();
			if (targetClass == null) {
				throw new AopConfigException("TargetSource cannot determine target class: " +
						"Either an interface or a target is required for proxy creation.");
			}
			if (targetClass.isInterface() || Proxy.isProxyClass(targetClass) || ClassUtils.isLambdaClass(targetClass)) {
				return new JdkDynamicAopProxy(config);
			}
			return new ObjenesisCglibAopProxy(config);
		}
		else {
			return new JdkDynamicAopProxy(config);
		}
	}
	......
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值