参考博文:https://blog.csdn.net/jiankunking/article/details/52143504

内容

一、动态代理解析

1. 代理模式

Java 这门语言有许多种设计模式,其中一种设计模式为代理模式,所谓代理模式,就是通过代理方来操作目标对象,而不是自己直接调用。代理模式又分为静态代理和动态代理。
静态代理:针对每个被代理对象写一个代理类,当有多个代理对象时需要写多个代理类,操作不够优雅;
动态代理:可以根据接口动态的生成代理类,这动态生成的类不需要自己书写,jdk帮我们完成了,代码变得简洁。
无论是动态代理还是静态代理,最终都会产生一个代理类(class文件),里面都含有对被代理对象的封装,只是诞生的途径不一样。下面我主要介绍 JDK 动态代理 的实现和原理。

2. 为什么要使用动态代理

动态代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问。代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理。

3. JDK 动态代理简单结构图

4. JDK 动态代理实现步骤

  1. 创建一个实现接口InvocationHandler的类,它必须实现invoke方法
  2. 创建被代理的类以及接口
  3. 通过Proxy的静态方法 newProxyInstance(ClassLoaderloader, Class[] interfaces, InvocationHandler h)创建一个代理
  4. 通过代理调用方法

注意: 真实类(被代理的类)必须实现接口

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 动态代理实现和代码分析的更多相关文章

  1. JAVA设计模式-动态代理(Proxy)源码分析

    在文章:JAVA设计模式-动态代理(Proxy)示例及说明中,为动态代理设计模式举了一个小小的例子,那么这篇文章就来分析一下源码的实现. 一,Proxy.newProxyInstance方法 @Cal ...

  2. Java JDK 动态代理使用及实现原理分析

    转载:http://blog.csdn.net/jiankunking   一.什么是代理? 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问.代理类负责为委托类预处理 ...

  3. Java,JDK动态代理的原理分析

    1. 代理基本概念: 以下是代理概念的百度解释:代理(百度百科) 总之一句话:三个元素,数据--->代理对象--->真实对象:复杂一点的可以理解为五个元素:输入数据--->代理对象- ...

  4. Java JDK动态代理解析

    动态代理虽不常自己实现,但在Spring或MyBatis中都有重要应用.动态代理的意义在于生成一个占位(又称代理对象),来代理真实对象,从而控制真实对象的访问.Spring常JDK和CGLIB动态代理 ...

  5. java jdk动态代理学习记录

    转载自: https://www.jianshu.com/p/3616c70cb37b JDK自带的动态代理主要是指,实现了InvocationHandler接口的类,会继承一个invoke方法,通过 ...

  6. JDK动态代理实现源码分析

    JDK动态代理实现方式 在Spring框架中经典的AOP就是通过动态代理来实现的,Spring分别采用了JDK的动态代理和Cglib动态代理,本文就来分析一下JDK是如何实现动态代理的. 在分析源码之 ...

  7. JDK动态代理案例与原理分析

    一.JDK动态代理实现案例 Person接口 package com.zhoucong.proxy.jdk; public interface Person { // 寻找真爱 void findlo ...

  8. java jdk动态代理模式举例浅析

    代理模式概述 代理模式是为了提供额外或不同的操作,而插入的用来替代”实际”对象的对象,这些操作涉及到与”实际”对象的通信,因此代理通常充当中间人角色. java中常用的动态代理模式为jdk动态代理和c ...

  9. java jdk动态代理

    在面试的时候面试题里有一道jdk的动态代理是原理,并给一个事例直接写代码出来,现在再整理一下 jdk动态代理主要是想动态在代码中增加一些功能,不影响现有代码,实现动态代理需要做如下几个操作 1.首先必 ...

随机推荐

  1. ELK日志收集(SpringBoot)

    目录 环境&准备 ES安装 Kibana安装 Logstash安装 Logstash配置 SpringBoot中logback-spring.xml配置 测试 启动 ES\Kibana\Log ...

  2. Triple Shift

    来源:Atcoder ARC 136 B - Triple Shift (atcoder.jp) 题解:这道题我们不可能去硬模拟(大多数这种题都不能这样去模拟的),然后我们就要去发现特性, 发现把 a ...

  3. think php 图像加水印

    1.将下载好的字体引入至 thinkPHP 框架的public/static 下(这里我建了一个文件夹叫font) (1.) (2.)   2.将字体路径写入config.php中 'font'=&g ...

  4. gitee中项目到运行操作,包括:打包、热部署、数据库操作

    使用的工具:window10.IDEA 2018.2.3 .navicat110_premium.Git-2.23 1.idea导入gitee代码 复制项目地址 选择git工具 粘贴地址,点击clon ...

  5. Linux特殊权限之suid、sgid、sbit权限

    文件权限管理之特殊命令 一:特殊权限 昨天所学的Linux基本权限为为9个:分别是rwx rwx rwx.但有时会发现系统中会有一些特殊的权限位符号: 例如: Linux系统一共有12个特殊权限符: ...

  6. For循环案例练习一基础版

    输出1-10之间的数据 1 public class LX1 { 2 public static void main(String[] args) { 3 for (int x=1;x<=10; ...

  7. 如何让 if/else 更优雅?

    if else 是我们写代码时,使用频率最高的关键词之一,然而有时过多的 if else 会让我们感到脑壳疼,有没有什么方法可以让我们避免来写这么多的 if else 呢? 1.使用 return 我 ...

  8. CentOS7 yum源修改为阿里,配置阿里epel源

    镜像下载.域名解析.时间同步请点击 阿里巴巴开源镜像站 一.概念/区分: yum源 什么是yum源: yum是一个在CentOS.RedHat和Fedora操作系统中使用的Shell前端软件包管理器. ...

  9. 数据库上云实践:使用Ora2pg进行数据库迁移

    目录 概述 重要 前置条件 配置环境 Win环境配置 linux环境配置 定义ORACLE_HOME环境变量 Ora2Pg使用方法 配置文件ora2pg_table.conf 配置文件解释:Oracl ...

  10. 搭建nuget服务器(二):制作nuget包

    生成nuget包可以使用nuget.exe或者下载nuget package explorer工具 nuget package explorer 下载地址:https://github.com/NuG ...