概念
举例:我要灭了传说中的山口组,我通知青龙帮,然后青龙帮 帮我灭了,最终达到了我要灭山口组的目的,我亲自灭和别人帮我灭结果都一样的,所以,青龙帮就相当于代替了我,这种方式叫代理模式。
2、技术点(反射)
了解下反射,可能只讲代理设计模式的话可能你不需要掌握,但是代理设计模式的应用场景会讲到,还是希望你能够了解下反射知识点。
3、代理设计模式
3.1 静态代理
1、概念:目标对象、和代理对象统一实现一样的接口 缺点,代理类太多的情况,一旦增加接口方法,那么目标对象与代理对象都要重新修改和维护。 2、代码讲解 类:接口IUserDao、实现接口的两个类,目标类和代理类 GirlUser和ProxyGirl,最后main函数执行(java9版本以上)
代码如下
public interface IUserDao { public static final String girlName ="七七"; void miss(String boyName); static void love(){ System.out.println("恋爱"); }; default void noLove(){ System.out.println("高冷,不谈恋爱"); } private static void securt(){ System.out.println("每个女孩子内心都有一个小秘密"); } }
public class GirlUser implements IUserDao{ @Override public void miss(String boyName) { System.out.println("我在思念岳飞将军"); } @Override public void noLove() { System.out.println("我跟谁也不谈恋爱,你们男的一个比一个丑"); } }
public class ProxyGirl implements IUserDao{ private IUserDao iUserDao; public ProxyGirl(IUserDao iUserDao){ this.iUserDao = iUserDao; } @Override public void miss(String boyName) { System.out.println("我告诉你们男生他在思念几百年前的岳飞大将军"); } @Override public void noLove() { System.out.println("她不会和你们恋爱的,你们死心吧"); } }
public class Main { public static void main(String[] args){ //可以做到在不修改目标对象的功能前提下,对目标功能扩展. //因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类,类太多.同时,一旦接口增加方法,目标对象与代理对象都要维护. GirlUser girlUser = new GirlUser(); ProxyGirl proxyGirl = new ProxyGirl(girlUser); proxyGirl.noLove(); } }
运行结果
3.2 jdk动态代理
1、概念:jdk动态代理是代理对象类不需要实现接口!但是目标对象一定要实现接口! 2、代码实现 用到的类 IUserDao GirlUser DongtaiProxy,由于3.1.2已经写了两个类了,我就最后在把DongtaiProxy 和重写的main函数代码贴一下
代码如下
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class DongtaiProxy { //传入的是目标对象 private Object object; public DongtaiProxy(Object object){ this.object = object; } public Object getProxyInstance(){ return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(), new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { return method.invoke(object,args); } }); } }
import proxy.jingtaiProxy.GirlUser; import proxy.jingtaiProxy.IUserDao; public class Main { public static void main(String[] args){ //代理对象不需要实现接口,但是目标对象一定要实现接口,否则不能用动态代理 IUserDao iUserDao = new GirlUser(); IUserDao proxy = (IUserDao) new DongtaiProxy(iUserDao).getProxyInstance(); proxy.noLove(); } }
运行结果如图:
3.3 cglib动态代理
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-core</artifactId> <version>5.0.4.RELEASE</version> </dependency>
3.3.1 Girl、CglibProxy、main函数,重新贴一下这个cglib代理的代码
public class Girl { public void love(){ System.out.println("爱家人爱朋友爱和平!"); } }
import org.springframework.cglib.proxy.Enhancer; import org.springframework.cglib.proxy.MethodInterceptor; import org.springframework.cglib.proxy.MethodProxy; import java.lang.reflect.Method; public class CglibProxy { private Object target; public CglibProxy(Object target){ this.target = target; } public Object getProxyInstance(){ //org.springframework.cglib.proxy.Enhancer cglib中的一个工具类 Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(target.getClass()); enhancer.setCallback(new MethodInterceptor() { public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { Object result = method.invoke(target,objects); return result; } }); //创建子类对象 return enhancer.create(); } }
public class main { public static void main(String[] args){ Girl girl = new Girl(); Girl proxy = (Girl) new CglibProxy(girl).getProxyInstance(); proxy.love(); } }
运行结果
4、应用场景
最典型的就是Spring AOP,它就是根据代理模式实现的,当然还有我上面讲的反射。
4.1、先说反射,只是大概说下,反射在我们工作中应用的还是比较多的,比如我们的eclipse、idea,当我们输入一个对象的时候,只要一点就会出现它相应的方法和属性,还有它能够开发各种通用的框架,struts2举例
我们前端请求的是ModifyLoginPasswordAction,那么我们就会解析这个struts.xml文件,然后找到方法是sms4LoginPwd的action,然后创建这个action的实例,去调用execute()执行后台代码,就讲到了这里,主要说设计模式。
4.2 spring aop 用到的代理设计模式
它主要用到jdk动态代理设计以及cglib动态代理设计,spring aop怎么用到的呢,它会判断你的目标对象类有木有实现接口,然后做一个判断,如图
在说下aop应用在那些场景呢?事物管理,日志,缓存等等...
说下事物配置?我们都是在xml文件配置相关属性,如图
然后看到我红色标出来的那个类了吗,进去看一下!
这个类里面封装了很多控制事物的方法,commit(),begin()...等等,因为文件太大我只截图一部分,所以我们在项目开发中就在事物方面一直用这些方法,但是绝对不会创建这个类的实例,因为这是spring核心内部的类,怎么可能是我们随随便便拿出来new的,所以我们只能通过代理模式用这个类的事物控制方法,同理好多类我们也如此!
希望大家记住代理设计模式在我看来就是,框架里面的部分核心类,官方的部分类,咱就不直接new,用另外一种方式用他们的方法,就是代理!为什么不直接 new,看下面
在实际的项目中,在你通往架构师的道路上,你要培养出一种感觉:要new一个实体对象是件很谨慎的事情(不是指值对象),不要随便new。最好不要自己new,让别人去new,传给你去调用。这样new错了也是别人的事,换而言之你的模块是好质量的,禁得起推敲的。那么都不愿意去new,谁去new?你可以通过代理模式获取该类实例,获取对应的方法,当然还有另外一种代替new的,那就是用抽象工厂模式,下一节讲解!
来源:oschina
链接:https://my.oschina.net/hlHuyNe/blog/3210553