java proxy 代理

本文介绍Spring AOP中使用JDK动态代理技术在运行期织入增强代码的过程。通过实现InvocationHandler接口定义横切逻辑,并通过Proxy类创建接口的代理实例,实现了对业务方法的前后增强。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

spring AOP 使用动态代理技术在运行期织入增强的代码,动态代理分为基于jdk的和基于CGLib两种,jdk本身只提供接口的代理,而cglib支持类的代理.

1.jdk动态代理

jdk1.3以后,java提供了动态代理技术,允许开发者在运行期创建接口的代理实例.jdk动态代理主要涉及到java.lang.reflect包中的两个类:Proxy和InvocationHandler.

其中InvocationHandler是一个接口,可卡因通过实现该接口定义横切逻辑,并通过反射机制调用目标类的代码,动态将横切逻辑和业务逻辑编织在一起.而Proxy利用InvocationHandler动态的创建一个符合某一接口的实例,生成目标类的代理对象.


以下是示例代码:



①接口

public interface Count {
	/**
	 * 查看账户方法
	 */
	public void queryCount();

}

②实现类

public class CountImpl implements Count {

	@Override
	public void queryCount() {
		System.out.println("query method...");
	}

}

③代理类

public class CountProxy2 implements InvocationHandler {
	//要代理的目标类实例
	Object target = null;
	public CountProxy2(Object target) {
		this.target = target;
	}
	
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		//调用目标类方法前
		System.out.println("before.....");
		Object result = method.invoke(target, args);
		//调用目标类方法后
		System.out.println("after.....");
		//返回代理类
		return result;
	}

④测试代码

/**
*
* Proxy.newProxyInstance方法会做如下几件事:
* 
* 1,根据传入的第二个参数interfaces动态生成一个类,实现interfaces中的接口,
* 	该例中即BusinessProcessor接口的processBusiness方法
* 	。并且继承了Proxy类,重写了hashcode,toString,equals等三个方法。具体实现可参看
* 	ProxyGenerator.generateProxyClass(...); 该例中生成了$Proxy0类
* 
* 2,通过传入的第一个参数classloder将刚生成的类加载到jvm中。即将$Proxy0类load
* 
* 3,利用第三个参数,调用$Proxy0的$Proxy0(InvocationHandler)构造函数
* 	创建$Proxy0的对象,并且用interfaces参数遍历其所有接口的方法,并生成Method对象初始化对象的几个Method成员变量
*
*/
public class CountTest {

	public static void main(String[] args) {
		
		CountImpl countImpl = new CountImpl();
		CountProxy2 handler = new CountProxy2(countImpl);
		Count count = (Count) Proxy.newProxyInstance(countImpl.getClass().getClassLoader(), 
				countImpl.getClass().getInterfaces(), handler);
		count.queryCount();
		
	}
	
}

⑤打印结果

before.....
query method...
after.....


### Java 中设置和使用 Proxy 代理网络 #### 使用 `java.net.Proxy` 访问网络资源 为了通过代理服务器发起 HTTP 请求,可以利用 `java.net.Proxy` 来指定代理信息。具体来说,可以通过创建一个 `Proxy` 对象并将其传递给 `URL.openConnection()` 方法。 ```java import java.io.IOException; import java.net.InetSocketAddress; import java.net.Proxy; import java.net.URL; import java.net.HttpURLConnection; public class HttpProxyExample { public static void main(String[] args) throws IOException { String host = "proxy.example.com"; int port = 8080; // 创建代理对象 Proxy proxy = new Proxy(Proxy.Type.HTTP, new InetSocketAddress(host, port)); // 打开连接到目标网站,并指定了使用的代理 URL url = new URL("http://example.com"); HttpURLConnection connection = (HttpURLConnection) url.openConnection(proxy); // 发起请求 connection.connect(); System.out.println("Response Code: " + connection.getResponseCode()); } } ``` 这段代码展示了如何定义一个 HTTP 协议型的代理,并通过该代理建立与远程主机之间的连接[^1]。 #### 动态代理机制简介 除了上述方式外,在某些场景下可能还需要自定义接口行为或者增强现有功能时,则需要用到动态代理技术。这通常涉及到反射 API 和字节码操作工具包如 ASM 或 CGLIB 来生成新的文件。对于 JDK 自带的支持而言: 当调用 `Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)` 方法时,内部会先基于传入参数构建相应的代理字节数组,再加载成实际可用的 Class 实例返回给开发者用于后续逻辑处理[^2]。 #### 高版本中的 HttpClient 支持 从 Java 11 开始引入了一个更现代化且易用的新客户端库——`HttpClient`,它不仅支持同步还提供了异步编程模型;更重要的是简化了很多复杂配置项的操作流程,包括但不限于 SSL/TLS 握手过程以及各种认证方案等特性集成。下面是一个简单的例子说明怎样借助这个新组件完成相同任务: ```java import java.net.http.HttpClient; import java.net.http.HttpRequest; import java.net.http.HttpResponse; import java.net.InetSocketAddress; import java.net.ProxySelector; import java.util.concurrent.CompletableFuture; public class ModernHttpClientWithProxy { private static final String PROXY_HOST = "proxy.example.com"; private static final int PROXY_PORT = 8080; public static CompletableFuture<HttpResponse<String>> fetchPageAsync() { var clientBuilder = HttpClient.newBuilder() .version(HttpClient.Version.HTTP_2) .followRedirects(HttpClient.Redirect.NORMAL); if (!PROXY_HOST.isEmpty()) { clientBuilder.proxy( ProxySelector.of(new InetSocketAddress(PROXY_HOST, PROXY_PORT)) ); } var httpClient = clientBuilder.build(); HttpRequest request = HttpRequest.newBuilder() .uri(java.net.URI.create("https://example.org")) .GET() .build(); return httpClient.sendAsync(request, HttpResponse.BodyHandlers.ofString()); } public static void main(String[] args) throws Exception { fetchPageAsync().thenAccept(response -> System.out.println("Status code: " + response.statusCode()) ).join(); } } ``` 此段程序片段体现了现代风格下的代理设定方法及其优势所在[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值