Java实现代理模式的三种方式

开发 前端
代理模式是项目中常用的一种设计模式。提供了间接访问目标对象的一种方式;即通过代理对象访问目标对象。这样做的好处是,可以在不改变原有目标对象的基础上,对目标对象增加额外的扩展功能。

什么是代理模式

代理模式是项目中常用的一种设计模式。提供了间接访问目标对象的一种方式;即通过代理对象访问目标对象。

这样做的好处是,可以在不改变原有目标对象的基础上,对目标对象增加额外的扩展功能。

代理模式又分为静态代理、jdk动态代理、cglib动态代理三种实现方式。

三种实现方式各有优点,以及适用的场景。

一、静态代理

被代理对象与代理对象需要实现相同的接口或者是继承相同父类,因此要定义一个接口或抽象类。

/**代理接口*/
public interface IHello {
String hi(String key);
}
/**代理接口实现类*/
public class HelloImpl implements IHello {
@Override
public String hi(String key) {
String str = "hello:" + key;
System.out.println("HelloImpl! " + str);
return str;
}
}
/**静态代理类*/
public class HelloStaticProxy implements IHello {

private IHello hello;

public HelloStaticProxy(IHello hello) {
this.hello = hello;
}

@Override
public String hi(String key) {
System.out.println(">>> static proxy start");
String result = hello.hi(key);
System.out.println(">>> static proxy end");
return result;
}
}
/**测试*/
public class DemoTest {

public static void main(String[] args) {
IHello helloProxy = new HelloStaticProxy(new HelloImpl());
helloProxy.hi("world");
}
}

输出结果:

>>> static proxy start
HelloImpl! hello:world
>>> static proxy end

二、jdk动态代理

jdk动态代理是基于接口的一种代理方式,目标对象一定要实现接口。

原理是,利用反射机制,动态生成匿名类继承Proxy类并且实现了要代理的接口,由于java不支持多继承,所以JDK动态代理不能代理类。

/**代理接口*/
public interface IHello {
String hi(String key);
}
/**代理接口实现类*/
public class HelloImpl implements IHello {
@Override
public String hi(String key) {
String str = "hello:" + key;
System.out.println("HelloImpl! " + str);
return str;
}
}


/**jdk动态代理类*/
public class JdkProxy implements InvocationHandler {

private Object target;

public JdkProxy(Object target) {
this.target = target;
}

/**
* 获取被代理接口实例对象
*
* @param <T>
* @return
*/
public <T> T getProxy() {
return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}

@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println(">>> JdkProxy start");
Object result = method.invoke(target, args);
System.out.println(">>> JdkProxy end");
return result;
}
}

/**测试*/
public class Demo2Test {

public static void main(String[] args) {
JdkProxy proxy = new JdkProxy(new HelloImpl());
IHello helloProxy = proxy.getProxy();
helloProxy.hi(" jdk proxy !");
}
}

输出结果:

>>> JdkProxy start
HelloImpl! hello: jdk proxy !
>>> JdkProxy end

三、cglib动态代理

目标对象可以不用实现接口,不能针对final类进行代理。

原理是,动态生成class继承目标对象。

使用cglib必须引入对应的jar包。

 <dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.2.7</version>
</dependency>
/**目标类*/
public class HelloImpl {
public String hi(String key) {
String str = "hello:" + key;
System.out.println("HelloImpl! " + str);
return str;
}
}

/**cglib代理类*/
public class CglibProxy implements InvocationHandler {

private Object target;

/**
* 获取被代理接口实例对象
*/
public <T> T getProxy() {
//1创建增强器对象
Enhancer e = new Enhancer();
//2设置增强器的类加载器
e.setClassLoader(target.getClass().getClassLoader());
//3设置代理对象父类类型
e.setSuperclass(target.getClass());
//4设置回调函数
e.setCallback(this);
//5创建代理对象
return (T) e.create();
}

public CglibProxy(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println(">>> cglib start");
Object obj = method.invoke(target, args);
System.out.println(">>> cglib end");
return obj;
}
}

/**测试*/
public class Demo3Test {

public static void main(String[] args) {
HelloImpl hello = new HelloImpl();
CglibProxy cglibProxy = new CglibProxy(hello);
HelloImpl proxy = cglibProxy.getProxy();
proxy.hi(" cglib ");
}
}

输出结果:

>>> cglib start
HelloImpl! hello: cglib
>>> cglib end

四、总结

静态代理,代理类必须非常明确,所以无法做到通用,但是效率也是最高的。

jdk动态代理,必须基于接口代理,有一定局限性;动态生成字节码文件,可以用于通用业务(性能日志等)。

cglig动态代理,也是动态生成字节码文件,生成的代理类继承了目标对象。

spring aop默认代理策略是:如果目标对象实现了接口,则使用jdk动态代理,否则使用cglib代理。

jdk8之后,jdk动态代理效率要高于cglib代理​。

责任编辑:武晓燕 来源: 今日头条
相关推荐

2018-04-02 14:29:18

Java多线程方式

2014-12-31 17:42:47

LBSAndroid地图

2021-06-24 08:52:19

单点登录代码前端

2021-11-05 21:33:28

Redis数据高并发

2010-03-12 17:52:35

Python输入方式

2023-12-04 09:31:13

CSS卡片

2022-01-20 08:38:02

Java接口Lambda

2022-01-17 08:19:51

Javascript 接口前端

2010-09-07 16:31:27

CSS

2012-07-17 09:16:16

SpringSSH

2022-06-06 13:41:27

区块链商业活动数字技术

2015-09-14 09:31:44

结对设计

2021-03-07 10:17:40

RDMA网络传输网络协议

2019-11-20 18:52:24

物联网智能照明智能恒温器

2010-07-12 15:40:53

BlackBerry开

2020-11-01 17:10:46

异步事件开发前端

2022-05-27 11:33:02

前端代码设计模式

2022-03-16 14:27:49

CSS三角形前端

2020-11-03 19:52:54

Java数组编程语言

2009-11-06 13:23:27

WCF模式
点赞
收藏

51CTO技术栈公众号