java学习笔记(中级篇)—JDK动态代理
2019-08-16 12:39:55来源:博客园 阅读 ()
java学习笔记(中级篇)—JDK动态代理
一、什么是代理模式
相信大家都知道代理商这个概念,在商业中,代理商无处不在。假设你要去买东西,你不可能去找真正的厂家去买,也不可能直接跟厂家提出需求,代理商就是这中间的一桥梁,连接买家和厂商。你要买或者定制什么产品,需要什么规格样式和价格的直接跟代理商沟通就好了,由代理商与真实的厂商沟通,这样买家有什么问题都可以咨询代理商,真实厂商也就可以专心做产品,不需要管其他事务了,买家也插手不了厂商的事。
在以上的关系中,厂商就是被代理对象,代理商就是代理对象,而买家就是调用者了。java中有多种动态代理的技术,包括JDK、CGLIB、Javassist等,这里我会拿CGLIB与JDK动态代理作比较。
二、意义
动态代理的意义就是在与生成一个占位(代理对象),来代理真实对象,从而达到控制真实对象的目的。要了解动态代理,首先要具备反射的知识。
三、实现动态代理的步骤
代理的实现分为两个主要步骤:
1.代理对象和真实对象建立代理关系
2.实现代理对象的逻辑方法
四、JDK动态代理
JDK动态代理,JDK自带的功能,在java.lang.reflect.*包中。要实现JDk动态代理,必须要借助接口才能产生代理对象。
1.先来定义一个简单的接口HelloWorld.java
public interface HelloWorld {
void sayHello();
}
2.HelloWorld的实现类HelloWorldImpl.java
public class HelloWorldImpl implements HelloWorld {
@Override
public void sayHello() {
System.out.println("hello world");
}
}
3.建立代理关系,必须实现接口InvocationHandler。invoke是接口里唯一需要实现的方法,里面实现的代理逻辑,当代理对象调度方法时,就会映射到invoke方法来,实际是通过反射来实现的。而bind()方法就是建立代理关系,通过Proxy的newProxyInstance来创建代理对象,第一个参数类加载器,第二个参数下挂的接口,第三个参数则是实现了代理逻辑方法invoke的类。
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class JdkProxy implements InvocationHandler {
//真实对象
private Object target = null;
/**
* 建立真实对象与代理对象之间的关系
* @param target 传入真实对象
* @return
*/
public Object bind(Object target){
this.target = target;
return Proxy.newProxyInstance
(target.getClass().getClassLoader(),
target.getClass().getInterfaces(),this);
}
/**
* 代理逻辑
* @param proxy 代理对象
* @param method 当前调度的方法
* @param args 调度方法的参数
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("进入代理逻辑方法");
System.out.println("在调用真实对象之前的服务");
Object object = method.invoke(target,args);
System.out.println("在调用真实对象之后的服务");
return object;
}
}
4.测试
import org.junit.Test;
public class JdkProxyTest {
@Test
public void test_JdkProxy(){
JdkProxy proxy = new JdkProxy();
HelloWorld helloWorld = null;
try {
helloWorld = (HelloWorld) proxy
.bind(Class.forName
("com.xcl.ssm.chapter2.jdkproxy.HelloWorldImpl")
.newInstance());
} catch (Exception e) {
e.printStackTrace();
}
helloWorld.sayHello();
}
}
五、CGLIB动态代理
上面说JDK动态代理的实现必须借助接口才能实现,而CGLIB则不用,只需要有一个非抽象类即可。我们以上面的HelloWorldImpl类为例,现在没有HelloWorld接口了。JDK动态代理必须要实现invocationHandler接口,CGLIB则必须要实现MethodInterceptor接口,里面也只有一个需要实现的方法intercept(),需要在里面实现代理逻辑。写代码之前记得导包哦,这是第三方提供的技术。
public class CglibProxy implements MethodInterceptor {
/**
* 生成CGLIB代理对象
* @return
*/
public Object getProxy(Class clz){
//CGLIB增强类对象
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(clz);
//设置当前对象为代理逻辑对象
enhancer.setCallback(this);
//生成并返回代理对象
return enhancer.create();
}
/**
* 代理逻辑方法
* @param proxy 代理对象
* @param method 方法
* @param args 参数
* @param methodProxy 方法代理
* @return
* @throws Throwable
*/
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
System.out.println("调用真实对象前");
//通过反射调用真实对象的方法
Object obj = methodProxy.invokeSuper(proxy,args);
System.out.println("调用真实对象后");
return obj;
}
}
六、总结
要真正的参透动态代理就要非常熟悉反射机制,从以上的代码可以得知,动态代理的底层实现还是基于反射的,我们知道反射是很消耗性能的,这带来了一些性能问题,但是为了开发上的简便,这种牺牲是值得的。我们学的框架,比如spring,mybatis等,都大量使用了动态代理技术,有兴趣的童鞋可以去阅读源码(表示本人也不是看过很多源码),这里面用到了很多的设计模式,对提高自己的代码水平很有帮助。好啦,就说到这里吧~
喜欢我的小伙伴记得扫描关注呦~
原文链接:https://www.cnblogs.com/chlinlearn/p/11360205.html
如有疑问请与原作者联系
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- 国外程序员整理的Java资源大全(全部是干货) 2020-06-12
- 2020年深圳中国平安各部门Java中级面试真题合集(附答案) 2020-06-11
- 2020年java就业前景 2020-06-11
- 04.Java基础语法 2020-06-11
- Java--反射(框架设计的灵魂)案例 2020-06-11
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash