静态代理和动态代理

一、静态代理:就是直接由程序员写好的代理类或者在代理类.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、运行结果如下:
店铺服务的实现类,出售商品功能
-----------分割线------------
出售之前处理:比如进行商品出售前的库存验证。。。
店铺服务的实现类,出售商品功能
出售之后处理:比如给客户发送一封邮件,通知客户已生成订单。。。




### 静态代理动态代理的概念、区别及使用场景 #### 1. 概念 静态代理动态代理代理模式的两种实现方式。代理模式的核心思想是通过一个代理对象来控制对目标对象的访问,从而实现功能扩展或访问控制。 - **静态代理**是指在编译时就已经确定了代理代理类的关系[^1]。代理类需要手动编写,并且必须实现与被代理类相同的接口。 - **动态代理**则是在运行时动态生成代理类[^2],无需在编译前手动编写代理类代码。动态代理通常通过 Java 的 `java.lang.reflect.Proxy` 类或其他框架(如 CGLIB)实现。 #### 2. 区别 静态代理动态代理的主要区别体现在以下几个方面: - **定义时间**: - 静态代理代理类在编译时就已经确定[^3]。 - 动态代理代理类是在运行时动态生成的[^1]。 - **灵活性**: - 静态代理由于代理类是固定的,因此不够灵活,每次新增接口或修改接口时都需要重新编写代理类。 - 动态代理可以适应多种接口的变化,具有更高的灵活性可扩展性[^3]。 - **性能**: - 静态代理因为代理类是提前编译好的,所以在执行效率上可能略高于动态代理。 - 动态代理由于需要在运行时生成代理类,可能会引入一定的性能开销,但现代 JVM 已经对此进行了优化[^1]。 - **适用场景**: - 静态代理适合于接口较少且相对固定的情况。 - 动态代理更适合于接口较多或经常变化的场景,能够减少代码冗余并提高开发效率[^3]。 #### 3. 使用场景 - **静态代理**: - 当代理类的功能较为简单,且接口不常变化时,可以使用静态代理[^1]。 - 示例:日志记录、权限校验等简单的功能增强[^2]。 - **动态代理**: - 在需要为多个类提供统一的代理逻辑时,动态代理更加合适[^3]。 - 示例:AOP(面向切面编程)、事务管理、远程方法调用等复杂场景。 #### 示例代码 以下分别展示了静态代理动态代理的实现方式。 ##### 静态代理示例 ```java // 定义接口 interface Rent { void rent(); } // 真实主题类 class RealRent implements Rent { @Override public void rent() { System.out.println("房东出租房子"); } } // 静态代理类 class ProxyRent implements Rent { private Rent realRent; public ProxyRent(Rent realRent) { this.realRent = realRent; } @Override public void rent() { System.out.println("中介开始找房..."); realRent.rent(); System.out.println("中介收取佣金"); } } // 测试 public class StaticProxyTest { public static void main(String[] args) { Rent realRent = new RealRent(); Rent proxyRent = new ProxyRent(realRent); proxyRent.rent(); // 输出:中介开始找房... 房东出租房子 中介收取佣金 } } ``` ##### 动态代理示例 ```java import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; // 定义接口 interface Rent { void rent(); } // 真实主题类 class RealRent implements Rent { @Override public void rent() { System.out.println("房东出租房子"); } } // 动态代理类 class DynamicProxyHandler implements InvocationHandler { private Object target; public DynamicProxyHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("中介开始找房..."); Object result = method.invoke(target, args); System.out.println("中介收取佣金"); return result; } } // 测试 public class DynamicProxyTest { public static void main(String[] args) { Rent realRent = new RealRent(); InvocationHandler handler = new DynamicProxyHandler(realRent); Rent proxyRent = (Rent) Proxy.newProxyInstance( realRent.getClass().getClassLoader(), realRent.getClass().getInterfaces(), handler ); proxyRent.rent(); // 输出:中介开始找房... 房东出租房子 中介收取佣金 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值