JAVA 中的 动态代理

本文介绍了一个使用Java反射机制实现的动态代理示例,通过Proxy类和InvocationHandler接口为被代理对象的方法添加计时功能。

这几天温习了一下关于 java 反射 的内容,

发现JAVA 提供了 动态代理 的 默认实现(以前没用到过),

主要由 Proxy 类,InvocationHandler 接口(在 java.lang.reflect 包中)组成。

 

下面 演示一个使用 Proxy 和 InvocationHandler 类实现动态代理 的例子,

这个动态代理 对 被代理对象的方法执行时间进行计时,并将执行时间打印输出到控制台。

 

1,被代理类 的接口 Proxied

 

// 被代理类 需实现的 接口
public interface Proxied {
	void doSomething();
	void doSomethingElse(String str);
}
 

2,一个 Proxied接口 的实现类(被代理类)

 

public class ConcreteProxied implements Proxied {
	@Override
	public void doSomething() {
		try {
			Thread.sleep(100);
		} catch (InterruptedException e) {
			System.err.println("Error : InterruptedException");
		}
		System.out.println(this.getClass().getSimpleName()
				+ " >> doSomething .");
	}

	@Override
	public void doSomethingElse(String str) {
		try {
			Thread.sleep(150);
		} catch (InterruptedException e) {
			System.err.println("Error : InterruptedException");
		}
		System.out.println(this.getClass().getSimpleName()
				+ " >> doSomethingElse , argument = " + str + ".");
	}
}
 

3,TimingInvocationHandler 类,实现了 InvocationHandler 接口

 

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class TimeingInvocationHandler implements InvocationHandler{
	//被代理的对象
	private Object proxied;
	public TimeingInvocationHandler(Object proxied){
		this.proxied = proxied;
	}
	
	// 参数  proxy 表示代理类的对象
	// 参数  method 表示被代理类 和 代理类 都实现的接口 的方法对象
	// 参数  args 表示方法 method 的参数数组
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println(method.getDeclaringClass().getName());
		long currentTimeMillis = System.currentTimeMillis();
		Object ret = method.invoke(proxied, args);
		System.out.println(this.getClass().getSimpleName()+" >> wastes time : "
				+(System.currentTimeMillis() - currentTimeMillis)+"ms");
		return ret;
	}

}
 

4,测试类 Test

 

import java.lang.reflect.Proxy;

public class TestProxy {
	public static void main(String[] args) {
		Proxied proxied = new ConcreteProxied();
		proxied.doSomething();
		proxied.doSomethingElse("only a String");

		// 生成一个代理实例,这个代理实现了 Proxied 接口
		// 对这个代理(proxy)的方法的调用 会 重定向到 TimeingInvocationHandler 的 invoke 方法
		Proxied proxy = (Proxied) Proxy.newProxyInstance(Proxied.class
				.getClassLoader(), // 类加载器
				new Class[] { Proxied.class }, // 代理要实现的接口
				new TimeingInvocationHandler(proxied) // 调用处理器
				);
		proxy.doSomething();
		proxy.doSomethingElse("only a String");
	}

}

 

 运行Test类,输出如下:

 

ConcreteProxied >> doSomething .
ConcreteProxied >> doSomethingElse , argument = only a String.

ConcreteProxied >> doSomething .
TimeingInvocationHandler >> wastes time : 93ms
ConcreteProxied >> doSomethingElse , argument = only a String.
TimeingInvocationHandler >> wastes time : 157ms

 

 

代码转载自:https://pan.quark.cn/s/7f503284aed9 Hibernate的核心组件总数达到五个,具体包括:Session、SessionFactory、Transaction、Query以及Configuration。 这五个核心组件在各类开发项目中都具有普遍的应用性。 借助这些组件,不仅可以高效地进行持久化对象的读取与存储,还能够实现事务管理功能。 接下来将通过图形化的方式,逐一阐述这五个核心组件的具体细节。 依据所提供的文件内容,可以总结出以下几个关键知识点:### 1. SSH框架详细架构图尽管标题提及“SSH框架详细架构图”,但在描述部分并未直接呈现关于SSH的详细内容,而是转向介绍了Hibernate的核心接口。 然而,在此我们可以简要概述SSH框架(涵盖Spring、Struts、Hibernate)的核心理念及其在Java开发中的具体作用。 #### Spring框架- **定义**:Spring框架是一个开源架构,其设计目标在于简化企业级应用的开发流程。 - **特点**: - **分层结构**:该框架允许开发者根据实际需求选择性地采纳部分组件,而非强制使用全部功能。 - **可复用性**:Spring框架支持创建可在不同开发环境中重复利用的业务逻辑和数据访问组件。 - **核心构成**: - **核心容器**:该部分包含了Spring框架的基础功能,其核心在于`BeanFactory`,该组件通过工厂模式运作,并借助控制反转(IoC)理念,将配置和依赖管理与具体的应用代码进行有效分离。 - **Spring上下文**:提供一个配置文件,其中整合了诸如JNDI、EJB、邮件服务、国际化支持等企业级服务。 - **Spring AO...
<think>好的,用户想了解Java动态代理的应用场景和示例。首先,我需要回忆一下动态代理的基本概念,确保自己理解正确。动态代理主要是在运行时动态生成代理类,而不需要像静态代理那样手动编写代理类。这通常通过Javajava.lang.reflect.Proxy和InvocationHandler来实现。 接下来,用户需要的是应用场景和示例。根据已有的引用资料,比如引用[1]提到了AOP、远程方法调用、延迟加载和事务管理,引用[5]也提到了框架中的应用。需要把这些场景整理出来,每个场景配一个具体的例子,最好是代码示例。 然后要注意用户提到的格式要求:行内数学表达式用$...$,独立公式用$$...$$。不过这个问题可能不涉及数学公式,但代码块需要用```python或者其他语言标签。不过用户的问题是关于Java的,所以代码示例应该用Java。但原示例中的代码用了Python,可能需要调整。 用户还要求生成相关问题,所以需要最后添加相关问题部分。同时,引用标识要自然地添加在引用的段落末尾,比如引用[^1]、[^2]等。需要确保每个引用对应的内容正确,比如引用[1]关于AOP的应用,引用[5]关于框架中的使用。 另外,回答的结构要清晰,分步骤或分点说明。用户可能希望看到每个应用场景的具体解释和代码示例,这样更容易理解。比如,AOP的场景可以结合Spring框架中的事务管理,但可能需要简化,用更基础的例子来说明动态代理如何实现方法增强。 还要注意使用中文回答,保持语言流畅自然。需要检查是否有遗漏的应用场景,比如远程调用(如RMI)、延迟加载(如Hibernate的延迟加载)和事务管理,这些都需要例子支持。 最后,生成的相关问题应该与动态代理相关,比如原理、与静态代理的区别、性能比较等,确保问题有针对性,帮助用户进一步学习。</think>### Java动态代理的应用场景及示例 Java动态代理主要用于**运行时动态生成代理对象**,无需手动编写代理类,适用于以下典型场景: --- #### 1. **面向切面编程(AOP)** 动态代理可在方法执行前后插入通用逻辑(如日志、事务、权限校验)。例如,为所有Service层方法添加事务控制: ```java // 接口 public interface UserService { void saveUser(User user); } // 动态代理实现事务管理 public class TransactionHandler implements InvocationHandler { private Object target; public TransactionHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { System.out.println("开启事务"); Object result = method.invoke(target, args); System.out.println("提交事务"); return result; } catch (Exception e) { System.out.println("回滚事务"); throw e; } } } // 使用代理 UserService userService = (UserService) Proxy.newProxyInstance( UserService.class.getClassLoader(), new Class[]{UserService.class}, new TransactionHandler(new UserServiceImpl()) ); userService.saveUser(user); // 代理对象自动管理事务 ``` 此模式在Spring框架的事务管理中广泛应用[^5]。 --- #### 2. **远程方法调用(RPC/RMI)** 动态代理可隐藏网络通信细节,例如Dubbo等框架通过代理封装远程调用: ```java // 客户端调用远程服务接口 public interface OrderService { Order getOrderById(int id); } // 代理类处理网络请求 public class RpcProxy implements InvocationHandler { @Override public Object invoke(Object proxy, Method method, Object[] args) { // 序列化参数并发送网络请求 return sendRequest(method.getName(), args); } } // 客户端透明调用 OrderService orderService = (OrderService) Proxy.newProxyInstance(...); Order order = orderService.getOrderById(123); // 实际触发远程调用 ``` --- #### 3. **延迟加载(Lazy Loading)** 动态代理可延迟资源加载,例如Hibernate中关联对象的延迟加载: ```java public class LazyLoader implements InvocationHandler { private Object realObject; private Supplier<Object> loader; public LazyLoader(Supplier<Object> loader) { this.loader = loader; } @Override public Object invoke(Object proxy, Method method, Object[] args) { if (realObject == null) { realObject = loader.get(); // 首次调用时加载真实对象 } return method.invoke(realObject, args); } } // 使用代理延迟加载大对象 ImageService imageService = (ImageService) Proxy.newProxyInstance(...); // 真实对象仅在调用时加载 imageService.getHighResolutionImage(); ``` --- #### 4. **接口适配与装饰器模式** 动态代理可统一处理多个接口的公共逻辑,例如为多个Service添加性能监控: ```java public class MetricsHandler implements InvocationHandler { private Object target; @Override public Object invoke(Object proxy, Method method, Object[] args) { long start = System.currentTimeMillis(); Object result = method.invoke(target, args); System.out.println("方法耗时: " + (System.currentTimeMillis() - start) + "ms"); return result; } } // 为任意Service接口添加监控 UserService userService = Proxy.newProxyInstance(...); ProductService productService = Proxy.newProxyInstance(...); ``` --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值