Java JDK 动态代理实现和代码分析
JDK 动态代理
参考博文:https://blog.csdn.net/jiankunking/article/details/52143504
内容
一、动态代理解析
1. 代理模式
Java 这门语言有许多种设计模式,其中一种设计模式为代理模式,所谓代理模式,就是通过代理方来操作目标对象,而不是自己直接调用。代理模式又分为静态代理和动态代理。
静态代理:针对每个被代理对象写一个代理类,当有多个代理对象时需要写多个代理类,操作不够优雅;
动态代理:可以根据接口动态的生成代理类,这动态生成的类不需要自己书写,jdk帮我们完成了,代码变得简洁。
无论是动态代理还是静态代理,最终都会产生一个代理类(class文件),里面都含有对被代理对象的封装,只是诞生的途径不一样。下面我主要介绍 JDK 动态代理 的实现和原理。
2. 为什么要使用动态代理
动态代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问。代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理。
3. JDK 动态代理简单结构图

4. JDK 动态代理实现步骤
- 创建一个实现接口InvocationHandler的类,它必须实现invoke方法
- 创建被代理的类以及接口
- 通过Proxy的静态方法 newProxyInstance(ClassLoaderloader, Class[] interfaces, InvocationHandler h)创建一个代理
- 通过代理调用方法
注意: 真实类(被代理的类)必须实现接口
5. JDK 动态代理 API
5.1 java.lang.reflect.Proxy
Java 动态代理机制生成的所有动态代理类的父类,它提供了一组静态方法来为一组接口动态地生成代理类及其对象。
主要方法: public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler hanlder)
- 方法职责:
为指定类加载器、一组接口及调用处理器生成动态代理类实例。 - 方法参数:
loader : 类加载器,一般传递真实对象的类加载器;
interfaces: 代理类需要实现的接口;
handler: 代理执行处理器,说人话就是生成代理对象帮你要做什么。 - 方法返回: 创建的代理对象。
5.1 java.lang.reflect.InvocationHandler
主要方法:public Object invoke(Object proxy, Method method, Object[] args)。
- 方法职责:
代理类实现该接口,负责集中处理动态代理类上的所有方法调用,让使用者自定义做什么事情, 对原来方法增强(加什么功能)。 - 方法参数:
proxy : 生成的代理对象;
method: 当前调用的真实方法对象;
args : 当前调用方法的实参。 - 方法返回: 真实方法的返回结果。
二、JDK 动态代理的实现(代码)
1. 项目结构图

2. IRentService 接口
package com.yy.homework.service;
public interface IRentService {
void rent();
}
3. LandlordServiceImpl 真实类
package com.yy.homework.service.impl;
import com.yy.homework.service.IRentService;
public class LandlordServiceImpl implements IRentService {
@Override
public void rent() {
System.out.println("我是房东,我以1000一个月的房价给中介帮我出租!");
}
}
4. TransactionInvocationHandler 代理类
package com.yy.homework.service.impl;
import com.yy.homework.tx.MyTransactionManager;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class TransactionInvocationHandler implements InvocationHandler {
private Object target;
@Autowired
private MyTransactionManager myTransactionManager;
public Object getTarget() {
return target;
}
public void setTarget(Object target) {
this.target = target;
}
/**
* @author YanYang
* @description: 负责集中处理动态代理类上的所有方法调用,让使用者自定义做什么事情,对原来方法增强(模拟一下事务)
* proxy:生成的代理对象
* method:调用真实对象的方法
* args:当前调用方法的实参
* return:返回真实方法的返回结果
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object retVal = null;
try {
// 开启事务
myTransactionManager.begin();
// 调用真实对象的方法
retVal = method.invoke(target, args);
// 提交事务
myTransactionManager.commit();
} catch (Exception e) {
// 回滚事务
myTransactionManager.rollback();
e.printStackTrace();
}
return retVal;
}
}
5. MyTransactionManager 增强类(模拟一下事务)
package com.yy.homework.tx;
import org.springframework.stereotype.Component;
/**
* @program: static-proxy
* @ClassName MyTransactionManager
* @description:
* @author: YanYang
**/
@Component
public class MyTransactionManager {
public void begin() {
System.out.println("开启事务");
}
public void commit() {
System.out.println("提交事务");
}
public void rollback() {
System.out.println("回滚事务");
}
}
6. applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="com.yy.homework"/>
<bean id="transactionInvocationHandler" class="com.yy.homework.service.impl.TransactionInvocationHandler">
<property name="target">
<!-- 把房东真实对象参起来藏起来 -->
<bean class="com.yy.homework.service.impl.LandlordServiceImpl"/>
</property>
</bean>
</beans>
7. TransactionInvocationHandlerTest 测试类
package com.yy.homework.service.impl;
import com.yy.homework.service.IRentService;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import java.lang.reflect.Proxy;
import java.util.Arrays;
/**
* @program: static-proxy
* @ClassName TransactionInvocationHandlerTest
* @description:
* @author: YanYang
**/
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class TransactionInvocationHandlerTest {
@Autowired
TransactionInvocationHandler handler;
@Test
public void invoke() {
// 动态生成代理类并创建对象
IRentService proxy = (IRentService) Proxy.newProxyInstance(
// 类加载器,一般传递真实对象的类加载器
handler.getTarget().getClass().getClassLoader(),
// 代理类需要实现的接口,获取真实类实现的接口,生成代理类也是实现什么接口
handler.getTarget().getClass().getInterfaces(),
// 代理执行处理器,也就是生成代理对象帮你要做什么
// 通过这个参数告诉 API 生成代理对象具体做什么
handler);
proxy.rent();
System.out.println("handler = " + Arrays.toString(handler.getTarget().getClass().getInterfaces()));
System.out.println("classLoader = " + handler.getTarget().getClass().getClassLoader());
System.out.println("handler = " + handler);
}
}
运行结果:
"C:\Program Files\Java\jdk-11.0.9\bin\java.exe"
com.intellij.rt.junit.JUnitStarter -ideVersion5 -junit4 com.yy.homework.service.impl.TransactionInvocationHandlerTest,invoke
开启事务
我是房东,我以1000一个月的房价给中介帮我出租!
提交事务
handler = [interface com.yy.homework.service.IRentService]
classLoader = jdk.internal.loader.ClassLoaders$AppClassLoader@2437c6dc
handler = com.yy.homework.service.impl.TransactionInvocationHandler@36328d33
Process finished with exit code 0
总结
以上就是对 JDK 动态代理 的总结了,代码仅供参考,欢迎讨论交流。
Java JDK 动态代理实现和代码分析的更多相关文章
- JAVA设计模式-动态代理(Proxy)源码分析
在文章:JAVA设计模式-动态代理(Proxy)示例及说明中,为动态代理设计模式举了一个小小的例子,那么这篇文章就来分析一下源码的实现. 一,Proxy.newProxyInstance方法 @Cal ...
- Java JDK 动态代理使用及实现原理分析
转载:http://blog.csdn.net/jiankunking 一.什么是代理? 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问.代理类负责为委托类预处理 ...
- Java,JDK动态代理的原理分析
1. 代理基本概念: 以下是代理概念的百度解释:代理(百度百科) 总之一句话:三个元素,数据--->代理对象--->真实对象:复杂一点的可以理解为五个元素:输入数据--->代理对象- ...
- Java JDK动态代理解析
动态代理虽不常自己实现,但在Spring或MyBatis中都有重要应用.动态代理的意义在于生成一个占位(又称代理对象),来代理真实对象,从而控制真实对象的访问.Spring常JDK和CGLIB动态代理 ...
- java jdk动态代理学习记录
转载自: https://www.jianshu.com/p/3616c70cb37b JDK自带的动态代理主要是指,实现了InvocationHandler接口的类,会继承一个invoke方法,通过 ...
- JDK动态代理实现源码分析
JDK动态代理实现方式 在Spring框架中经典的AOP就是通过动态代理来实现的,Spring分别采用了JDK的动态代理和Cglib动态代理,本文就来分析一下JDK是如何实现动态代理的. 在分析源码之 ...
- JDK动态代理案例与原理分析
一.JDK动态代理实现案例 Person接口 package com.zhoucong.proxy.jdk; public interface Person { // 寻找真爱 void findlo ...
- java jdk动态代理模式举例浅析
代理模式概述 代理模式是为了提供额外或不同的操作,而插入的用来替代”实际”对象的对象,这些操作涉及到与”实际”对象的通信,因此代理通常充当中间人角色. java中常用的动态代理模式为jdk动态代理和c ...
- java jdk动态代理
在面试的时候面试题里有一道jdk的动态代理是原理,并给一个事例直接写代码出来,现在再整理一下 jdk动态代理主要是想动态在代码中增加一些功能,不影响现有代码,实现动态代理需要做如下几个操作 1.首先必 ...
随机推荐
- 译<容器网络中OVS-DPDK的性能>
译<容器网络中OVS-DPDK的性能> 本文来自对Performance of OVS-DPDK in Container Networks的翻译. 概要--网络功能虚拟化(Network ...
- think php 未登录,禁止访问页面 + 退出登录
1.首先在Local创建一个基类控制器 D:\PHP\phpstudy_pro\WWW\1906A\pyg>php think make:controller goods/Base --plai ...
- 编写Python代码的注意事项
1.标识符 标识符是定义的名称,包括类名.变量名等等 标识符的大小写是敏感的,且第一个字符必须是字母表中的字母或"_" 在python3中,中文可被用作变量名 不能使用Python ...
- LGB+XGB+CNN一般写法
现在的比赛,想要拿到一个好的名次,就一定要进行模型融合,这里总结一下三种基础的模型: - lightgbm:由于现在的比赛数据越来越大,想要获得一个比较高的预测精度,同时又要减少内存占用以及提升训练速 ...
- kkFileView部署到windows服务出现问题解决
1.部署之后执行出现api-ms-win-crt-runtime-l1-1-0.dll丢失的办法 微软官网下载vc_redist.x64.exe vc_redist.x86.exe 64位的操作系统需 ...
- LGP3281口胡
当你看到一个东西的时候,GF 有可能比 DP 更方便.处理贡献也有可能比 DP 更方便. 这个题意明显是让我们计算 \(S(r)-S(l-1)\) 之类的东西( 所以直接考虑前缀的答案就好了( 考虑将 ...
- (acwing蓝桥杯c++AB组)1.1 递归
(acwing蓝桥杯c++AB组)1.课程介绍+递归 文章目录 (acwing蓝桥杯c++AB组)1.课程介绍+递归 课程介绍 第一讲 递归与递推 递归 引入 递归的底层调用顺序 例题与练习 课程介绍 ...
- Windows 10平台安装PostgreSQL 14.2详细教程
Windows 10平台安装postgreSQL 14.2.1,安装步骤很简单,基本上是点击下一步(next). 使用SQL Shell(psql)进行交互:使用pgAdmin工具进行管理. tips ...
- mac电脑sublime text3安装pretty json插件
因http://packagecontrol.io/地址被墙无法实现自动安装,导致sublime Text3安装插件非常麻烦,总是出现There Are No Packages Available F ...
- JavaWeb 11_文件上传
一.操作步骤 1.要有一个form标签,method=post 请求2.form标签的encType属性值必须为multipart/form-data值3.在form标签中使用input type=f ...