设计模式- 代理模式 (静态代理 和 动态代理)

本文详细介绍了代理模式的概念及其应用场景,通过具体的静态代理和动态代理实例,展示了如何在数据库操作中运用代理模式来控制事务。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.代理

                                           

    

  (1)意图:

           为其他对象提供一种代理以控制对这个对象的访问。

  (2)适用性:

           在需要用比较通用和复杂的对象指针代替简单的指针的时候,使用Proxy模式。下面是一 些可以使用Proxy 模式常见情况: 
1) 远程代理(Remote Proxy )为一个对象在不同的地址空间提供局部代表。 NEXTSTEP[Add94] 使用NXProxy 类实现了这一目的。Coplien[Cop92] 称这种代理为“大使” (Ambassador )。 
2 )虚代理(Virtual Proxy )根据需要创建开销很大的对象。在动机一节描述的ImageProxy 就是这样一种代理的例子。 
3) 保护代理(Protection Proxy )控制对原始对象的访问。保护代理用于对象应该有不同 的访问权限的时候。例如,在Choices 操作系统[ CIRM93]中KemelProxies为操作系统对象提供 了访问保护。 
4 )智能指引(Smart Reference )取代了简单的指针,它在访问对象时执行一些附加操作。 它的典型用途包括:

对指向实际对象的引用计数,这样当该对象没有引用时,可以自动释放它(也称为SmartPointers[Ede92 ] )。当第一次引用一个持久对象时,将它装入内存。在访问一个实际对象前,检查是否已经锁定了它,以确保其他对象不能改变它。

2.静态代理实例

   (1)场景描述

           在操作数据库的时候,我们在进行增,修,删的时候,需要提交事务操作;

   (2)示例操作 :模拟数据库dao操作

          1)dao接口

public interface DocmentDao {
	
	void insertDoc();
	
	void updateDoc();

}

         2)dao实现接口

public class DocmentDaoImpl implements DocmentDao {

	@Override
	public void insertDoc() {
          System.out.println("添加文档成功");
	}

	@Override
	public void updateDoc() {
          System.out.println("更新文档成功");
	}

}

        3)事务实现

public class Transtion {
	
	public void beginTranstion(){
		System.out.println("开启事务");
	}
	
	public void closeTranstion(){
		System.out.println("提交事务");
	}

}

      4)dao代理实现

public class DocmentDaoProxy implements DocmentDao {

	private DocmentDao documentDao;
	private Transtion transtion;

	public DocmentDaoProxy(DocmentDao documentDaoImpl, Transtion transtion) {
		super();
		this.documentDao = documentDaoImpl;
		this.transtion = transtion;
	}

	@Override
	public void insertDoc() {
		this.transtion.beginTranstion();
		this.documentDao.insertDoc();
		this.transtion.closeTranstion();
	}

	@Override
	public void updateDoc() {
		this.transtion.beginTranstion();
		this.documentDao.updateDoc();
		this.transtion.closeTranstion();
	}

}

     5)测试

	@Test
	public void testProxy(){
		
		DocmentDao docmentDao=new DocmentDaoImpl();
		Transtion transtion=new Transtion();
		
		DocmentDaoProxy daoProxy = new DocmentDaoProxy(docmentDao, transtion);
		daoProxy.insertDoc();
		
	}

3.示例2 : 使用Spring ,模拟dao层事务提交

      (1)bean.xml 配置实现

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

    <bean id="docmentDaoImpl" class="cn.labelnet.proxy.spring.DocmentDaoImpl"></bean>

    <bean id="transtion" class="cn.labelnet.proxy.spring.Transtion"></bean>
    
    <bean id="docmentDaoProxy" class="cn.labelnet.proxy.spring.DocmentDaoProxy">
       <constructor-arg index="0" ref="docmentDaoImpl"></constructor-arg>
       <constructor-arg index="1" ref="transtion"></constructor-arg>
    </bean>
    
    
</beans>

      (2)测试实现

	@Test
	public void testSpringProxy(){
		ApplicationContext context=new ClassPathXmlApplicationContext("applicationContext.xml");
		DocmentDaoProxy daoProxy=(DocmentDaoProxy) context.getBean("docmentDaoProxy");
		daoProxy.updateDoc();
	}

4.静态代理总结

     静态代理模式的缺点:
         1)如果一个系统有多个dao,则需要创建多个代理对象;
2)如果igedao中有很多方法需要事物,则代理对象的方法中重复的代码很多;
3)proxy的重用行不强;

 

5.动态代理

    使用拦截器实现 ;

     场景描述 :同样是模拟数据库dao层实现

    (1)dao接口

public interface DocmentDao {
	
	void insertDoc();
	
	void updateDoc();

}
 

     (2)dao接口实现

public class DocmentDaoImpl implements DocmentDao {

	@Override
	public void insertDoc() {
          System.out.println("添加文档成功");
	}

	@Override
	public void updateDoc() {
          System.out.println("更新文档成功");
	}

}


    (3)事务实现

public class Transtion {
	
	public void beginTranstion(){
		System.out.println("开启事务");
	}
	
	public void closeTranstion(){
		System.out.println("提交事务");
	}

}

    (4)拦截器实现

/**
 *  事务动态代理拦截器
 *
 */
public class TranstionProxy implements InvocationHandler {
	

	private Object target;
	private Transtion transtion;
	
	
	
	public TranstionProxy(Object target, Transtion transtion) {
		super();
		this.target = target;
		this.transtion = transtion;
	}



	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		this.transtion.beginTranstion();
		method.invoke(target, args);
		this.transtion.closeTranstion();
		return null;
	}
	
}

   (5)测试

@Test
	public void testProxy(){
		
		//目标对象
		Object daoImpl = new DocmentDaoImpl();
		//
		Transtion transtion=new Transtion();
		//拦截器
		TranstionProxy transtionProxy = new TranstionProxy(daoImpl, transtion);
        //
        DocmentDao dao = (DocmentDao) Proxy.newProxyInstance(daoImpl.getClass().getClassLoader(),daoImpl.getClass().getInterfaces(),transtionProxy);
        dao.updateDoc();
		
	}


6.动态代理总结

    产生的代理对象和目标对象实现了共同的接口;(jdk动态代理)(例如5的示例)
    JDK的动态代理 : 
  1. 用Jdk的API做到的;
  2. 代理对象时动态产生的;
 
 
 2)代理对象时目标对象的子类;(spring:cglib动态代理) (后面实现)


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值