【黑马程序员济南中心】代理模式-动态代理

【黑马程序员济南中心】代理模式-动态代理

我们上一次和大家聊了聊代理模式里的静态代理,今天和大家聊一个代理模式的另一种方式即动态代理。

什么是动态代理?

当想要给实现了某个接口的类中的方法,加一些额外的处理。比如说加日志,加事务等。可以给这个类创建一个代理,故名思议就是创建一个新的类,这个类不仅包含原来类方法的功能,而且还在原来的基础上添加了额外处理的新类。这个代理类并不是定义好的,是动态生成的。具有解耦意义,灵活,扩展性强。

大家会发现在静态代理和动态代理的区别吗,我们上一章讲得静态代理大家会发现一个问题,静态代理类只能为特定的接口(service)服务,如果想要为多个接口服务则需要建立很多个代理类。在这种情况下,我们就需要引入动态代理类了,因为动态代理是在程序运行时,通过反射机制实现动态代理,并且能够代理各种类型的对象。

我们通过代码给大家演示一下动态代理的应用:

具体实现类:

public class StuManagerImpl implements StuManager {

 

@Override

public void addStu(String studentId, String studentName) {

System.out.println("添加学员");

}

 

@Override

public void delStu(String studentId) {

System.out.println("删除学员");

}

 

@Override

public String findStu(String studentId) {

System.out.println("通过ID查找学员");

return "张三";

}

 

}

 

动态创建代理对象的类:

//动态代理类只能代理接口(不支持抽象类),代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类

   

public class LogHandler implements InvocationHandler {

 

// 目标对象

private Object targetObject;

//绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。            

public Object newProxyInstance(Object targetObject){

this.targetObject=targetObject;

//该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例  

//第一个参数指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器

//第二个参数要实现和目标对象一样的接口,所以只需要拿到目标对象的实现接口

//第三个参数表明这些被拦截的方法在被拦截时需要执行哪个InvocationHandler的invoke方法

//根据传入的目标返回一个代理对象

return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),

targetObject.getClass().getInterfaces(),this);

}

@Override

//关联的这个实现类的方法被调用时将被执行

/*InvocationHandler接口的方法,proxy表示代理,method表示原对象被调用的方法,args表示方法的参数*/

public Object invoke(Object proxy, Method method, Object[] args)

throws Throwable {

System.out.println("start-->>");

for(int i=0;i<args.length;i++){

System.out.println(args[i]);

}

Object ret=null;

try{

/*原对象方法调用前处理日志信息*/

System.out.println("satrt-->>");

 

//调用目标方法

ret=method.invoke(targetObject, args);

/*原对象方法调用后处理日志信息*/

System.out.println("success-->>");

}catch(Exception e){

e.printStackTrace();

System.out.println("error-->>");

throw e;

}

return ret;

}

 

}

客户端代码:

public class Client {

 

public static void main(String[] args){

LogHandler logHandler=new LogHandler();

UserManager userManager=(UserManager)logHandler.newProxyInstance(new UserManagerImpl());

//UserManager userManager=new UserManagerImpl();

userManager.addUser("1111", "张三");

}

}

 

我们可以看到,其实动态代理类并不是仅仅局限于我们这个例子中的类对象,他可以代理不同类型的类对象。

有的同学可能理解不了动态代理的概念,现在看完了可能还在模糊状态,不过不要紧,我们可以多问问度娘,多多练习,最终我们肯定能搞定动态代理这个设计模式。

转载于:https://my.oschina.net/u/3825479/blog/1799295

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值