设计模式课程 设计模式精讲 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(); ...
随机推荐
- 以C语言为例完成简单的网络聊天程序以及关于socket在Linux下系统调用的分析
套接字是网络编程中的一种通信机制,是支持TCP/IP的网络通信的基本操作单元,可以看做是不同主机之间的进程进行双向通信的端点,简单的说就是通信的两方的一种约定,用套接字中的相关函数来完成通信过程. 端 ...
- bootstrap当中,实现div居中
文本内容居中:利用bootstrap自带CSS样式表当中 的 text-center 样式来实现 <div class="row form-group text-center&qu ...
- ubuntu安装与设置
为学习Linux,在虚拟机中安装类ubuntu18.04,刚装完系统时间是不对的,系统中也没有gcc,g++. 关于安装软件无非就是: sudo apt-get install gcc sudo ap ...
- Linux上FTP部署:基于mariadb管理虚拟用户
FTP原理 FTP 采用 Internet 标准文件传输协议 FTP 的用户界面, 向用户提供了一组用来管理计算机之间文件传输的应用程序.图1 FTP 的基本模型 FTP 是基于客户---服务器(C/ ...
- JS中的数组创建,初始化
JS中没有专门的数组类型.但是可以在程序中利用预定义的Array对象及其方法来使用数组. 在JS中有三种创建数组的方法: var arr = new Array(1,2,3,4); var arr = ...
- Java工作流引擎关于数据加密流程(MD5数据加密防篡改)
关键字: 驰骋工作流程快速开发平台 工作流程管理系统 工作流引擎 asp.net工作流引擎 java工作流引擎. 开发者表单 拖拽式表单 工作流系统 流程数据加密 md5 数据保密流程数据防篡改 ...
- laravel 模拟数据批量添加
模拟User表结构: database/factories/UserFactory.php(模型工厂) <?php use App\Models\User; use Illuminate\Sup ...
- Django框架之图书管理系统(一)
图书管理系统共分为两篇博客进行讲解,该篇博客主要记录图书与出版社之间的关系(一对一),记录图书的增删查改操作 ============================================= ...
- 无刷新的批量图片上传插件.NET版
啥都不说,先上效果图: 这是一个网上的第三方组件,原版是php的,我用.NET重写了图片上传的处理,下面贴上代码 using System; using System.Collections.Gene ...
- Vue 项目 在局域网内访问
之前在百度上看到过很多中 关于局域网访问项目的方法, 在这里,个人推荐两种比较喜欢的方法 一.直接在项目的package.json文件中进行配置 ,代码如下 "scripts": ...