首页>>后端>>java->Java如何实现动态代理

Java如何实现动态代理

时间:2023-11-30 本站 点击:1

代理,指的是使用代理对象代替对其它对象的访问,简单点说,你求职时找的中介就是代理,那么在Java中,代理如何体现呢?

静态代理

我们首先需要知道,何为静态代理?静态代理指的是在编译期就对目标对象的方法进行增强,例如:

publicclassTestDemo{interfaceEmailService{voidsendEmail(StringemailContent);}staticclassEmailServiceImplimplementsEmailService{@OverridepublicvoidsendEmail(StringemailContent){System.out.println("发送了一封邮件,内容为:"+emailContent);}}publicstaticvoidmain(String[]args){EmailServiceemailService=newEmailServiceImpl();emailService.sendEmail("hello");}}

现在若是想在发送邮件之前获取一下当前的时间,则可以使用代理类对发邮件的方法进行增强:

publicclassTestDemo{interfaceEmailService{voidsendEmail(StringemailContent);}staticclassEmailServiceImplimplementsEmailService{@OverridepublicvoidsendEmail(StringemailContent){System.out.println("发送了一封邮件,内容为:"+emailContent);}}staticclassEmailProxyimplementsEmailService{privatefinalEmailServiceemailService;publicEmailProxy(EmailServiceemailService){this.emailService=emailService;}@OverridepublicvoidsendEmail(StringemailContent){System.out.println(LocalDateTime.now());emailService.sendEmail(emailContent);}}publicstaticvoidmain(String[]args){EmailServiceemailProxy=newEmailProxy(newEmailServiceImpl());emailProxy.sendEmail("hello");}}

静态代理的缺点非常明显,编写麻烦,且可扩展性不强,而动态代理的出现,将彻底解决这些问题。

动态代理

动态代理与静态代理恰恰相反,动态代理是在运行期对目标对象的某个方法进行增强,比如仍然是发邮件的服务,使用动态代理,即可这样实现:

publicclassTestDemo{interfaceEmailService{voidsendEmail(StringemailContent);}staticclassEmailServiceImplimplementsEmailService{@OverridepublicvoidsendEmail(StringemailContent){System.out.println("发送了一封邮件,内容为:"+emailContent);}}publicstaticvoidmain(String[]args){EmailServiceemailService=newEmailServiceImpl();EmailServiceemailProxy=(EmailService)Proxy.newProxyInstance(EmailServiceImpl.class.getClassLoader(),EmailServiceImpl.class.getInterfaces(),newInvocationHandler(){@OverridepublicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{System.out.println(LocalDateTime.now());Objectresult=method.invoke(emailService,args);returnresult;}});emailProxy.sendEmail("hello");}}

使用JDK提供的Proxy和InvocationHandler类能够轻松实现动态代理,但这种方式也是有局限性的,就是被增强的类必须实现了接口,因为Proxy的参数中需要接收类的接口信息。

CGLib实现动态代理

CGLib的出现,打破了这一僵局,使用CGLib,能够增强任意的对象方法,即使你没有实现任何接口,因为它是通过继承的方式进行增强的。 下面就来演示一下如何使用CGLib,首先引入依赖:

<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>3.3.0</version></dependency>

实现如下:

publicclassTestDemo{staticclassEmailServiceImpl{publicvoidsendEmail(StringemailContent){System.out.println("发送了一封邮件,内容为:"+emailContent);}}publicstaticvoidmain(String[]args){EmailServiceImplemailService=newEmailServiceImpl();EmailServiceImplemailProxy=(EmailServiceImpl)Enhancer.create(emailService.getClass(),newMethodInterceptor(){@OverridepublicObjectintercept(Objecto,Methodmethod,Object[]args,MethodProxymethodProxy)throwsThrowable{System.out.println(LocalDateTime.now());Objectobj=methodProxy.invokeSuper(o,args);returnobj;}});emailProxy.sendEmail("hello");}}

它的写法与JDK提供的方式类似,通过Enhancer类的create()方法即可增强一个对象,并传入对象的Class对象和一个MethodInterceptor接口的实现类,并在intercept()方法中对原方法进行增强。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:/java/4929.html