代理模式

it2026-03-19  3

代理模式

代理模式中,代理类与主体类实现同样的接口,代理类持有实体类的引用,并接受客户端对代理类中实体引用的外部注入,并代理实体类的功能。 注:描述中的这种外部注入形式有个专有技术名词:依赖注入

静态代理模式

角色

抽象角色:一般会使用接口或者抽象类来解决真实角色:被代理的角色代理角色:代理真实角色,一般会有一些附属操作客户:访问代理对象的角色

优点

可以使真实角色更加集中本身的业务,不用去关注一些公共的业务公共业务交给代理角色,实现业务的分工公共业务发生扩展时,方便集中管理

缺点

每多一个真实角色就会产生一个代理角色:代码量会翻倍,开发效率低

代码步骤

1.接口

public interface UsersService { public void add(); public void delete(); public void update(); public void query(); }

2.真实角色

public class UserServiceImp implements UsersService{ public void add() { System.out.println("添加"); } public void delete() { System.out.println("删除"); } public void update() { System.out.println("修改"); } public void query() { System.out.println("查询"); } }

3.代理角色

public class UserServiceProxy implements UsersService{ private UserServiceImp userService; public void setUserService(UserServiceImp userService) { this.userService = userService; } public void add() { log("add"); userService.add(); } public void delete() { log("delete"); userService.delete(); } public void update() { log("update"); userService.update(); } public void query() { log("query"); userService.query(); } public void log(String msg){ System.out.println("使用了"+msg+"方法"); } }

4.客户访问代理角色

public class User { public static void main(String[] args) { UserServiceImp userService = new UserServiceImp(); UserServiceProxy proxy = new UserServiceProxy(); proxy.setUserService(userService); proxy.add(); } }

运行结果:

使用了add方法 添加 Process finished with exit code 0

动态代理模式

和静态代理模式中的角色一样动态代理的具体代理类的生成是在运行期动态产生的,而非编译期就已经静态存在;动态代理分为: 基于接口:JDK动态代理基于类:CGLIB动态代理Java字节码实现:javassist

​ JDK中的动态代理是通过反射类Proxy以及InvocationHandler回调接口实现的;但是,JDK中所要进行动态代理的类必须要实现一个接口,也就是说只能对该类所实现接口中定义的方法进行代理,这在实际编程中具有一定的局限性,而且使用反射的效率也并不是很高。

JDK动态代理核心代码:

1.实现这个接口InvocationHandler,创建代理对象

2.Proxy提供创建动态代理类和实例的静态方法:Proxy.newProxyInstance()

//自动生成代理类 public class DynamicProxy implements InvocationHandler { //被代理的接口 private Object target; //set方法 public void setTarget(Object target) { this.target = target; } //生成得到代理类 public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(),this); } //处理代理实例并返回结果 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { log(method.getName()); Object result = method.invoke(target,args); return result; } //代理角色的附属业务的例子 public void log(String msg){ System.out.println("使用了"+msg+"方法"); } }

target可表示任务场景中的真实角色,这是一个模板!

客户实现业务:

public class User { public static void main(String[] args) { //真实角色 UserServiceImp userService = new UserServiceImp(); //代理角色,因为是动态代理所以不存在 DynamicProxy dp = new DynamicProxy(); //设置代理对象 dp.setTarget(userService); //动态生成代理类 UsersService proxy =(UsersService) dp.getProxy(); //调用真实角色中的方法 proxy.delete(); } }
最新回复(0)