【Spring】代理模式(十)

来源:互联网 发布:淘宝联盟导出excel后 编辑:程序博客网 时间:2024/06/07 14:09

1. 代理模式

1.1 概述

 代理(Proxy)是一种设计模式, 提供了对目标对象另外的访问方式;即通过代理访问目标对象。 这样好处: 可以在目标对象实现的基础上,增强额外的功能操作。(扩展目标对象的功能)

举例:明星(邓紫棋)ß---经纪人<-------用户  

    目标           (代理)

 

 

 

代理模式的关键点: 代理对象与目标对象。

 

1.2 静态代理

静态代理,

1) 代理对象,要实现与目标对象一样的接口;

2) 举例:

保存用户(模拟)

Dao  ,  直接保存

DaoProxy, 给保存方法添加事务处理

 

UserDaoProxy.java

 

package cn.lfsenior.a_static; /** * 代理对象(静态代理) *    代理对象,要实现与目标对象一样的接口 * */public class UserDaoProxy implements IUserDao{ // 接收保存目标对象private IUserDao target;public UserDaoProxy(IUserDao target) {this.target = target;}@Overridepublic void save() {System.out.println("开始事务...");target.save(); // 执行目标对象的方法System.out.println("提交事务...");}} 

App.java


package cn.lfsenior.a_static; public class App { public static void main(String[] args) {// 目标对象IUserDao target = new UserDao();// 代理IUserDao proxy = new UserDaoProxy(target);proxy.save();  // 执行的是,代理的方法}}




 

 

总结静态代理:

1)可以做到在不修改目标对象的功能前提下,对目标对象功能扩展。

2)缺点:

--》  因为代理对象,需要与目标对象实现一样的接口。所以会有很多代理类,类太多。

--》  一旦接口增加方法,目标对象与代理对象都要维护。

 

解决:

代理工厂?  可以使用动态代理。

 

 

1.3 动态代理

动态代理,

1)代理对象,不需要实现接口;

2)代理对象的生成,是利用JDKAPI, 动态的在内存中构建代理对象(需要我们指定创建 代理对象/目标对象 实现的接口的类型;);

3)  动态代理, JDK代理, 接口代理;

 

JDK中生成代理对象的API

|-- Proxy

static Object newProxyInstance(

ClassLoader loader,       指定当前目标对象使用类加载器

 Class<?>[] interfaces,     目标对象实现的接口的类型

InvocationHandler h       事件处理器

)  

 ProxyFactory.java

package cn.lfsenior.b_dynamic; import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy; /** * 给所有的dao创建代理对象【动态代理】 * * 代理对象,不需要实现接口 * */public class ProxyFactory { // 维护一个目标对象private Object target;public ProxyFactory(Object target){this.target = target;}// 给目标对象,生成代理对象  public Object getProxyInstance() {return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args)throws Throwable {System.out.println("开启事务");// 执行目标对象方法Object returnValue = method.invoke(target, args);System.out.println("提交事务");return returnValue;}});}} 


App.java


package cn.lfsenior.b_dynamic; public class App { public static void main(String[] args) {// 目标对象IUserDao target = new UserDao();// 【原始的类型 class cn.lfsenior.b_dynamic.UserDao】System.out.println(target.getClass());// 给目标对象,创建代理对象IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance();// class $Proxy0   内存中动态生成的代理对象System.out.println(proxy.getClass());// 执行方法   【代理对象】proxy.save();}}



代理对象不需要实现接口,但是目标对象一定要实现接口;否则不能用动态代理!
动态代理总结:

(class  $Proxy0  implements IuserDao)

 

 

 

思考:

有一个目标对象,想要功能扩展,但目标对象没有实现接口,怎样功能扩展?

Class  UserDao{}

// 子类的方式

Class subclass  extends  UserDao{}

以子类的方式实现(cglib代理)

 

 

1.4 Cglib代理

Cglib代理,也叫做子类代理。在内存中构建一个子类对象从而实现对目标对象功能的扩展。

 

JDK的动态代理有一个限制,就是使用动态代理的对象必须实现一个或多个接口。如果想代理没有实现接口的类,就可以使用CGLIB实现。

  CGLIB是一个强大的高性能的代码生成包,它可以在运行期扩展Java类与实现Java接口。它广泛的被许多AOP的框架使用,例如Spring AOPdynaop,为他们提供方法的interception(拦截)。

 CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。

 

 

Cglib子类代理:

1) 需要引入cglib– jar文件, 但是spring的核心包中已经包括了cglib功能,所以直接引入spring-core-3.2.5.jar即可。

2)引入功能包后,就可以在内存中动态构建子类

3)代理的类不能为final, 否则报错。

4) 目标对象的方法如果为final/static,那么就不会被拦截,即不会执行目标对象额外的业务方法。

ProxyFactory.java

package cn.itcast.c_cglib; import java.lang.reflect.Method; import org.springframework.cglib.proxy.Enhancer;import org.springframework.cglib.proxy.MethodInterceptor;import org.springframework.cglib.proxy.MethodProxy; /** * Cglib子类代理工厂 * (对UserDao 在内存中动态构建一个子类对象) * @author Jie.Yuan * */public class ProxyFactory implements MethodInterceptor{// 维护目标对象private Object target;public ProxyFactory(Object target){this.target = target;}// 给目标对象创建代理对象public Object getProxyInstance(){//1. 工具类Enhancer en = new Enhancer();//2. 设置父类en.setSuperclass(target.getClass());//3. 设置回调函数en.setCallback(this);//4. 创建子类(代理对象)return en.create();} @Overridepublic Object intercept(Object obj, Method method, Object[] args,MethodProxy proxy) throws Throwable {System.out.println("开始事务.....");// 执行目标对象的方法Object returnValue = method.invoke(target, args);System.out.println("提交事务.....");return returnValue;} }


App.java

package cn.itcast.c_cglib; public class App { public static void main(String[] args) {// 目标对象UserDao target = new UserDao();// class cn.itcast.c_cglib.UserDaoSystem.out.println(target.getClass());// 代理对象UserDao proxy = (UserDao) new ProxyFactory(target).getProxyInstance();// UserDao子类:class cn.itcast.c_cglib.UserDao$$EnhancerByCGLIB$$25d4aeabSystem.out.println(proxy.getClass());// 执行代理对象的方法proxy.save();}}

 

SpringAOP编程中,

如果加入容器的目标对象有实现接口,用JDK代理;

如果目标对象没有实现接口,用Cglib代理;

 

 

原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 弟媳妇天天在家吵架怎么办 弟媳妇不和我说话怎么办 我想和弟媳妇做一次怎么办 老公对弟媳妇有非分之想怎么办 跟弟媳妇有矛盾怎么办 学生考试前不认真怎么办 有人雇凶要杀我怎么办 高考理综数学英语都没考好怎么办 母亲判刑孩子未成年无人监管怎么办 高三了数学30分怎么办 母猪发烧耳朵放血止不住了怎么办? 青春期孩子动手打父母该怎么办 20岁睡觉不老实怎么办 孕妇梦见钱掉了怎么办 小孩晚上睡觉鼻子塞怎么办 宝宝晚上睡觉鼻子塞怎么办 按摩后吹冷气发冷怎么办 碰到特别细心敏感的下属怎么办 睡觉压的肩膀疼怎么办 睡觉感觉被压住动弹不得怎么办 越想睡觉越睡不着怎么办 腿被裤子染黑了怎么办 肚子上的松皮怎么办 在公司天天背锅怎么办 职场老实背锅怎么办 三星a8充不了电怎么办 果6软件连不上网怎么办 孩子在幼儿园磕伤了怎么办 小孩在幼儿园摔骨折了怎么办 孩子在幼儿园摔骨折了怎么办 大腿被撞了很痛怎么办 马面褶子坏了怎么办 纱料衣服有褶怎么办 裙子如果后背那里小了怎么办 湖州耳朵鸣很严重怎么办 预授权撤销错了怎么办 打酒店里小卡片电话被骗怎么办 本科错过了选导师该怎么办 本科导师威胁学生不让毕业怎么办 意向导师名额满了怎么办? 扔的瓶子被限制怎么办