静态代理和动态代理

本文详细介绍了静态代理和动态代理的概念及其应用场景。静态代理通过预先编写好的代理类来实现,而动态代理则在运行时动态生成代理类。通过具体实例展示了如何使用这两种代理方式,并比较了它们之间的区别。

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

一、静态代理:就是直接由程序员写好的代理类或者在代理类.class文件在编译期间就已经存在了
比如一般店铺具备商品出售功能,但店铺也可以把出售商品委托给超市。也就是说我们大家买商品直接到超市买就行了,不用管它是来自于哪个店铺。在这里,店铺(被代理类或者称之为委托对象)和超市(代理类)都必须实现同一个出售商品的接口。
1、先创建一个店铺服务接口:
/**
 * 店铺服务,具有出售商品功能
 * @author lichuang
 *
 */
public interface ShopService {
	
	/**
	 * 出售功能
	 */
	void sale();

}
2、再创建一个店铺实现类(被代理类,或者说委托类),也就真实的店铺服务类,实现出售商品接口
/**
 * 店铺服务的实现类
 * 
 * @author lichuang
 */
public class ShopServiceImpl implements ShopService{

	@Override
	public void sale() {
		System.out.println("店铺服务的实现类,出售商品功能");
	}

}
3、再创建一个超市类(代理类),实现出售商品接口,由于超市的出售商品功能本质上还是依赖于店铺的出售功能(相当于没有店铺提供商品出售功能,那么超市也就不会有出售商品功能了),所以需要持有一个被代理类对象。
/**
 * 店铺静态代理类,比如超市
 * 店铺可以直接出售商品,也可以让超市帮忙出售商品,也即由超市代理
 * @author lichuang
 *
 */
public class ShopStaticProxy implements ShopService {
	
//这里需要持有一个被代理类对象
	private ShopService shopService;
	
	public ShopStaticProxy(ShopService shopService) {
		 this.shopService = shopService;
	}

	@Override
	public void sale() {
		System.out.println("代理类:我是超市,能帮助店铺进行商品出售");
		shopService.sale();
	}

}
4、通过main方法进行测试

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

/**
 * 静态代理测试类,演示使用静态代理与不使用静态代理的区别
 * @author lichuang
 *
 */
public class StaticProxyTest {
	public static void main(String[] args) {
		notUseStaticProxy();
		System.out.println("-----------分割线------------");
		useStaticProxy();
	}
	
	//1、如果不用静态代理,则店铺需要自己进行出售商品
	private static void notUseStaticProxy() {
		ShopService shopService = new ShopServiceImpl();
		shopService.sale();
	}
	
	//2、如果使用静态代理,则店铺不需要自己出售商品,由代理类(比如超市)进行出售商品即可
	private static void useStaticProxy() {
		ShopService shopService = new ShopServiceImpl();
		ShopStaticProxy proxy = new ShopStaticProxy(shopService);
		proxy.sale();
	}
	
}
运行结果如下:
店铺服务的实现类,出售商品功能
    -----------分割线------------
    代理类:我是超市,能帮助店铺进行商品出售
    店铺服务的实现类,出售商品功能




二、动态代理:是在运行时动态生成的代理类
动态代理有2种,即JDK动态代理和cglib动态代理。前者是基于接口实现的,后者是基于继承一个类进行实现的。如果被代理类是一个类,那么只能使用cglib了,由于需要继承,所以要求该类不能被定义为final。在这里我暂时只演示JDK动态代理。
JDK动态代理涉及到的两个核心:
一个是InvocationHandler接口,需要重写其invoke(Object proxy, Method method, Object[] args)方法;
另一个是Proxy代理类,主要是其生成代理对象的静态方法newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
1、先创建一个店铺服务接口:
/**
 * 店铺服务,具有出售商品功能
 * @author lichuang
 *
 */
public interface ShopService {
	
	/**
	 * 出售功能
	 */
	void sale();

}
2、再创建一个店铺实现类(被代理类,或者说委托类),也就真实的店铺服务类,实现出售商品接口
/**
 * 店铺服务的实现类
 * 
 * @author lichuang
 */
public class ShopServiceImpl implements ShopService{

	@Override
	public void sale() {
		System.out.println("店铺服务的实现类,出售商品功能");
	}

}
3、创建一个代理类,并实现InvocationHandler接口,主要是实现invoke方法
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * 动态代理类
 * @author lichuang
 *
 */
public class ShopDynamicProxy implements InvocationHandler {
	
	/**
	 * 被代理对象
	 */
	private Object obj;
	
	public ShopDynamicProxy(Object obj) {
		this.obj = obj;
	}

	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("出售之前处理:比如进行商品出售前的库存验证。。。");
		
		method.invoke(obj, args);
		
		System.out.println("出售之后处理:比如给客户发送一封邮件,通知客户已生成订单。。。");
		return null;
	}

}
4、main方法测试
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

/**
 * 静态代理测试类,演示使用静态代理与不使用静态代理的区别
 * @author lichuang
 *
 */
public class StaticProxyTest {
	public static void main(String[] args) {
		notUseStaticProxy();
		/*System.out.println("-----------分割线------------");
		useDynamicProxy();*/
	}
	
	//1、如果不用静态代理,则店铺需要自己进行出售商品
	private static void notUseStaticProxy() {
		ShopService shopService = new ShopServiceImpl();
		shopService.sale();
	}
	
	//3、如果使用动态代理,则店铺不需要自己出售商品,由代理类 进行出售商品即可,并进行一些预处理
	private static void useDynamicProxy() {
		ShopService realShopService = new ShopServiceImpl();
		InvocationHandler handler = new ShopDynamicProxy(realShopService);
		Object object =	Proxy.newProxyInstance(realShopService.getClass().getClassLoader(),
				realShopService.getClass().getInterfaces(),handler );
		ShopService shopService = (ShopService) object;
		shopService.sale();
	}
}
5、运行结果如下:
店铺服务的实现类,出售商品功能
-----------分割线------------
出售之前处理:比如进行商品出售前的库存验证。。。
店铺服务的实现类,出售商品功能
出售之后处理:比如给客户发送一封邮件,通知客户已生成订单。。。




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值