前言
动态代理在Java中有着广泛的应用,比如Spring AOP、Hibernate数据查询、测试框架的后端mock、RPC远程调用、Java注解对象获取、日志、用户鉴权、全局性异常处理、性能监控,甚至事务处理等。
代理,指的是使用代理对象代替对其它对象的访问,简单点说,你求职时找的中介就是代理,那么在Java中,代理如何体现呢?
静态代理
我们首先需要知道,何为静态代理?静态代理指的是在编译期就对目标对象的方法进行增强,例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
public class TestDemo { interface EmailService { void sendEmail(String emailContent); } static class EmailServiceImpl implements EmailService{ @Override public void sendEmail(String emailContent) { System.out.println( "发送了一封邮件,内容为:" + emailContent); } } public static void main(String[] args) { EmailService emailService = new EmailServiceImpl(); emailService.sendEmail( "hello" ); } } |
现在若是想在发送邮件之前获取一下当前的时间,则可以使用代理类对发邮件的方法进行增强:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
public class TestDemo { interface EmailService { void sendEmail(String emailContent); } static class EmailServiceImpl implements EmailService{ @Override public void sendEmail(String emailContent) { System.out.println( "发送了一封邮件,内容为:" + emailContent); } } static class EmailProxy implements EmailService{ private final EmailService emailService; public EmailProxy(EmailService emailService) { this .emailService = emailService; } @Override public void sendEmail(String emailContent) { System.out.println(LocalDateTime.now()); emailService.sendEmail(emailContent); } } public static void main(String[] args) { EmailService emailProxy = new EmailProxy( new EmailServiceImpl()); emailProxy.sendEmail( "hello" ); } } |
静态代理的缺点非常明显,编写麻烦,且可扩展性不强,而动态代理的出现,将彻底解决这些问题。
动态代理
动态代理与静态代理恰恰相反,动态代理是在运行期对目标对象的某个方法进行增强,比如仍然是发邮件的服务,使用动态代理,即可这样实现:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
public class TestDemo { interface EmailService { void sendEmail(String emailContent); } static class EmailServiceImpl implements EmailService { @Override public void sendEmail(String emailContent) { System.out.println( "发送了一封邮件,内容为:" + emailContent); } } public static void main(String[] args) { EmailService emailService = new EmailServiceImpl(); EmailService emailProxy = (EmailService) Proxy.newProxyInstance(EmailServiceImpl. class .getClassLoader(), EmailServiceImpl. class .getInterfaces(), new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println(LocalDateTime.now()); Object result = method.invoke(emailService, args); return result; } }); emailProxy.sendEmail( "hello" ); } } |
使用JDK提供的Proxy和InvocationHandler类能够轻松实现动态代理,但这种方式也是有局限性的,就是被增强的类必须实现了接口,因为Proxy的参数中需要接收类的接口信息。
CGLib实现动态代理
CGLib的出现,打破了这一僵局,使用CGLib,能够增强任意的对象方法,即使你没有实现任何接口,因为它是通过继承的方式进行增强的。
下面就来演示一下如何使用CGLib,首先引入依赖:
1
2
3
4
5
|
< dependency > < groupId >cglib</ groupId > < artifactId >cglib</ artifactId > < version >3.3.0</ version > </ dependency > |
实现如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public class TestDemo { static class EmailServiceImpl { public void sendEmail(String emailContent) { System.out.println( "发送了一封邮件,内容为:" + emailContent); } } public static void main(String[] args) { EmailServiceImpl emailService = new EmailServiceImpl(); EmailServiceImpl emailProxy = (EmailServiceImpl) Enhancer.create(emailService.getClass(), new MethodInterceptor() { @Override public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { System.out.println(LocalDateTime.now()); Object obj = methodProxy.invokeSuper(o, args); return obj; } }); emailProxy.sendEmail( "hello" ); } } |
它的写法与JDK提供的方式类似,通过Enhancer类的create()方法即可增强一个对象,并传入对象的Class对象和一个MethodInterceptor接口的实现类,并在intercept()方法中对原方法进行增强。
总结
到此这篇关于Java实现动态代理的文章就介绍到这了,更多相关Java实现动态代理内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!
原文链接:https://juejin.cn/post/7012608239816671269