设计模式课程 设计模式精讲 16-4 代理模式Coding-动态代理
1 代码演练
1.1 动态代理
2 疑难解答
2.1 动态代理invoke怎么执行的?
2.2 感觉这块理解的不是很好,下边有时间再看看
1 代码演练
1.1 动态代理
重点:
重点关注动态代理类
测试类:
- package com.geely.design.pattern.structural.proxy.dynamicproxy;
- import com.geely.design.pattern.structural.proxy.IOrderService;
- import com.geely.design.pattern.structural.proxy.Order;
- import com.geely.design.pattern.structural.proxy.OrderServiceImpl;
- public class Test {
- public static void main(String [] args){
- Order order = new Order();
- order.setUserID(1);
- /**
- * new OrderServiceDynamicProxy(order) 该方法已经生成了一个新的代理类
- * 它的bind方法返回了原目标类,强转之后变成了原目标类。
- */
- IOrderService orderServiceDynamicProxy = (IOrderService) new OrderServiceDynamicProxy(new OrderServiceImpl()).bind();
- //注意,执行saveOrder方法,最终会执行invode方法。
- orderServiceDynamicProxy.saveOrder(order);
- }
- }
动态代理类:
- package com.geely.design.pattern.structural.proxy.dynamicproxy;
- import com.geely.design.pattern.structural.proxy.Order;
- import com.geely.design.pattern.structural.proxy.db.DataSourceContextHolder;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- import java.lang.reflect.Proxy;
- /**
- * 目的:抽奖信息和订单等不同的类都可以通过这一个动态代理进行复用,不用每一个都写一个静态代理。
- * 这就是静态代理和动态代理的区别
- * 动态代理是自动生成的,静态代理需要显式的来描述和coding
- */
- public class OrderServiceDynamicProxy implements InvocationHandler {
- //目标对象
- public Object target;
- //通过构造方法传入目标对象
- public OrderServiceDynamicProxy(Object target) {
- this.target = target;
- }
- /**
- * 主方法, 调用前置方法,主要方法,以及后置方法
- * @param proxy
- * @param method
- * @param args
- * @return
- * @throws Throwable
- */
- @Override
- public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
- //取得目标对象,argObject是目标类,也就是静态代理demo中的订单类
- Object argObject = args[0];
- beforeMethod(argObject);
- Object object = method.invoke(target,args);
- afterMethod();
- return object;
- }
- public Object bind(){
- //得到目标对象的class类
- Class cls = target.getClass();
- //这里边有三个参数,classLoader,复数的interface,它的类型是class,第三个是invoccationHandler 因为本类本身实现了InvocationHandler,所以把本类自己传过去即可。
- return Proxy.newProxyInstance(cls.getClassLoader(),cls.getInterfaces(),this);
- }
- /**
- * 前置方法,用来取模运算
- * @param obj
- */
- private void beforeMethod(Object obj){
- int userID = 0;
- System.out.println("动态代理 before code");
- if(obj instanceof Order){//如果该对象属于Order类
- Order order = (Order) obj;//强转成Order 类
- userID = order.getUserID();
- }
- int dbRouter = userID%2;
- System.out.println("动态代理分配到 【db"+dbRouter+"】数据库进行处理数据!");
- DataSourceContextHolder.setDBType("db"+String.valueOf(dbRouter));
- }
- /**
- * 后置方法
- */
- private void afterMethod(){
- System.out.println("动态代理 after code");
- }
- }
订单类:
- package com.geely.design.pattern.structural.proxy;
- /**
- * 建立订单实体类
- */
- public class Order {
- private Object orderInfo;
- //之所以选择integer类型,是为了方便OrderServiceStaticProxy静态代理类进行分库
- private Integer userID;
- public Object getOrderInfo() {
- return orderInfo;
- }
- public void setOrderInfo(Object orderInfo) {
- this.orderInfo = orderInfo;
- }
- public Integer getUserID() {
- return userID;
- }
- public void setUserID(Integer userID) {
- this.userID = userID;
- }
- }
订单dao:
- package com.geely.design.pattern.structural.proxy;
- public interface IOrderDao {
- int insertOrder(Order order);
- }
订单daoIMPL:
- package com.geely.design.pattern.structural.proxy;
- public class OrderDaoImpl implements IOrderDao{
- @Override
- public int insertOrder(Order order) {
- System.out.println("新增一条订单!");
- return 1;
- }
- }
订单Service:
- package com.geely.design.pattern.structural.proxy;
- public interface IOrderService {
- int saveOrder(Order order);
- }
订单ServiceIMPL:
- package com.geely.design.pattern.structural.proxy;
- public class OrderServiceImpl implements IOrderService {
- private IOrderDao orderDao;
- @Override
- public int saveOrder(Order order) {
- //Spring会自己注入,这里我们直接new了
- orderDao = new OrderDaoImpl();
- System.out.println("Service层调用dao层添加Order");
- return orderDao.insertOrder(order);
- }
- }
打印日志:
- Connected to the target VM, address: '127.0.0.1:12906', transport: 'socket'
- 动态代理 before code
- 动态代理分配到 【db1】数据库进行处理数据!
- Disconnected from the target VM, address: '127.0.0.1:12906', transport: 'socket'
- Service层调用dao层添加Order
- 新增一条订单!
- 动态代理 after code
- Process finished with exit code 0
2 疑难解答
2.1 动态代理invoke怎么执行的?
1.Proxy.newProxyInstance(//参数省略了...)的部分源码
2.程序运行时产生一个类$proxyQ
3.Sproxy0类继承自Proxy类,实现了目标对象的父类接口(借鉴的百度提供的源码
4.Sproxy0类有多个Method成员变量,它的静态代码块给Method赋值为我们自己的接口的实现类的对应的Method对象
5.Sproxyo实现接口的方法调用了super.h.invoke(参数),这里的参数包括Method变量
这样就缕顺了,整体流程:
代理对象调接口中的方法---代理对象的真身是$proxy0
调用了对应的方法---此方法内部调用其父类的成员h调用h的invoke方法---就是调用传入了InvocationHandler的invoke方法,至于返回值,那就看我们的InvocationHandler的实现类怎么写了。
这部分参考:https://www.jianshu.com/p/774c65290218
设计模式课程 设计模式精讲 16-4 代理模式Coding-动态代理的更多相关文章
- Java设计模式-代理模式之动态代理(附源代码分析)
Java设计模式-代理模式之动态代理(附源代码分析) 动态代理概念及类图 上一篇中介绍了静态代理,动态代理跟静态代理一个最大的差别就是:动态代理是在执行时刻动态的创建出代理类及其对象. 上篇中的静态代 ...
- (转)轻松学,Java 中的代理模式及动态代理
背景:讲到反射机制,肯定会想到动态代理. 轻松学,Java 中的代理模式及动态代理 代理模式可以在不修改被代理对象的基础上,通过扩展代理类,进行一些功能的附加与增强.值得注意的是,代理类和被代理类应该 ...
- java代理模式及动态代理类
1. 代理模式 代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用 ...
- 《Java设计模式》之代理模式 -Java动态代理(InvocationHandler) -简单实现
如题 代理模式是对象的结构模式.代理模式给某一个对象提供一个代理对象,并由代理对象控制对原对象的引用. 代理模式可细分为如下, 本文不做多余解释 远程代理 虚拟代理 缓冲代理 保护代理 借鉴文章 ht ...
- 动态代理模式——JDK动态代理
今天,我就来讲一下动态代理的设计模式. 动态代理的意义在于生成一个代理对象,来代理真实对象,从而控制真实对象的访问.操作动态代理需要两个步骤:一.代理对象和真实对象建立代理关系.二.实现代理对象的代理 ...
- Java的三种代理模式(Spring动态代理对象)
Java的三种代理模式 1.代理模式 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩 ...
- JAVA代理模式与动态代理模式
1.代理模式 所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动.在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之前起到中介的作用.代理模式给某 ...
- Java代理模式之动态代理
动态代理类的源码是程序在运行期间由JVM根据反射等机制动态生成的,所以不存在代理类的字节码文件.代理角色和真实角色的联系在程序运行时确定! Java中有两种动态代理,一种是JDK自带的,另一种的CGL ...
- 代理模式与动态代理之JDK实现和CGlib实现
静态代理 静态代理中的代理类和委托类会实现同一接口或是派生自相同的父类. 由业务实现类.业务代理类 两部分组成.业务实现类 负责实现主要的业务方法,业务代理类负责对调用的业务方法作拦截.过滤.预处理, ...
- 代理模式-jdk动态代理
IDB package com.bjpowernode.proxy; /** * 代理类和目标类都必须使用同一个接口. */ public interface IDB { int insert(); ...
随机推荐
- java中路径的问题
在java中,涉及路径的问题有很多,不管在windows还是linux系统中,不要纠结”/“分隔符的使用,在windows系统中,资源加载器会自动的将”/“转换成”\“. 在java中获取资源的方式有 ...
- 吴裕雄 python 神经网络TensorFlow实现LeNet模型处理手写数字识别MNIST数据集
import tensorflow as tf tf.reset_default_graph() # 配置神经网络的参数 INPUT_NODE = 784 OUTPUT_NODE = 10 IMAGE ...
- Cookie API介绍
一.Java提供的操作Cookie的API Java中的javax.servlet.http.Cookie类用于创建一个Cookie Cookie类的主要方法 No. 方法 类型 描述 Cookie( ...
- 2020年英特尔CPU供应短缺将持续
导读 有着相当靠谱的爆料历史的台媒 DigiTimes 报道称,其预计英特尔的 CPU 供应短缺问题,将持续到 2020 年末.对于这样的预测,我们其实早已见怪不怪,毕竟该公司首席执行官 Bob Sw ...
- MYSQL客户端管理工具
双击安装navicact客户端管理工具: 双击启动脚本,并指定管理工具的navicat.exe文件 最后打开
- 深入剖析Windows专业版安装Docker引擎和Windows家庭版Docker引擎安装的区别
原创声明:作者:Arnold.zhao 博客园地址:https://www.cnblogs.com/zh94 公司使用的电脑是Windows专业版,所以配置本机的Docker时会方便许多,后续由于需 ...
- Java连载65-自定义手动抛出异常、子类的异常范围、数组初探
一.手动抛出异常1.自定义无效名字异常: (1)编译时异常,直接继承Exception (2)运行时异常,直接继承RuntimeException 举例子:注意点:throws会向上抛出异常,跑到最上 ...
- 前x个数据中至少有m个元素最小值与最大值之差不超过K
题意 给一组数据,从左到右开始,寻找最小的x,使得第1个元素到第x个元素中,至少存在m个数据,最小值与最大值之差不超过K. INPUT 第一行是T,代表数据组数 每组数据的第一行是三个整数,n.m.k ...
- elasticsearch ik解析器
ik解析器 1. ik解析器 The IK Analysis plugin integrates Lucene IK analyzer (http://code.google.com/p/i ...
- Java 9 New Features
Java 9 概述 1. jdk 9 的发布.经过 4 次跳票,历经曲折的 java 9 终于终于在 2017 年 9 月 21 日发布. 2. Java 9 中哪些不得不说的新特性?java 9 提 ...