1. 代理模式简介:
代理模式是常用的Java 设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。按照代理类的创建时期,代理类可分为静态代理和动态代理两种。
2. 静态代理:
静态代理类: 由程序员创建或由特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。
简单静态代理实例代码:
定义一个接口:
public interface BookDao {
public void addBook();
}
再完成这个接口的实现:
public class BookDaoImpl implements BookDao {
@Override
public void addBook() {
System.out.println("添加书....代理测试");
}
}
BookDao代理类:
public class BookDaoProxy implements BookDao {
private BookDao bookDao;
/**
* 构造方法
*/
public BookDaoProxy(BookDao bookDao) {
super();
this.bookDao = bookDao;
}
@Override
public void addBook() {
this.before();
this.bookDao.addBook();
this.after();
}
public void before(){
System.out.println("前置处理..");
}
public void after(){
System.out.println("后置处理");
}
}
静态代理测试类:
public class TestProxy {
public static void main(String[] args){
BookDao bookDao=new BookDaoImpl();
BookDao bookDaoProxy=new BookDaoProxy(bookDao);
bookDaoProxy.addBook();
}
}
分析上面的代码,加入需要在BookDao这个接口中增加一个queryBook() 的方法,那么我们的实现类以及代理类都需要更改,这样既不利于代码的扩展,所以对应的出现了动态代理。
3. 动态代理:
动态代理类:动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。
动态代理示例代码:
定义一个接口:
public interface BookDao {
public void addBook();
}
再完成这个接口的实现:
public class BookDaoImpl implements BookDao {
@Override
public void addBook() {
System.out.println("添加书....代理测试");
}
}
代理类:
public class ObjectProxy implements InvocationHandler {
private Object target;
/**
* 绑定委托对象并返回一个代理类对象
* @param target
* @return
*/
public Object bind(Object target){
this.target=target;
return Proxy.newProxyInstance(target.getClass().getClassLoader(),
target.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] arg)
throws Throwable {
Object result=null;
result=method.invoke(target, arg);
return result;
}
}
动态代理测试类:
public class TestProxy {
public static void main(String[] args){
ObjectProxy op=new ObjectProxy();
BookDao bd=(BookDao) op.bind(new BookDaoImpl());
bd.addBook();
}
}
从上面的代码可以看出,即使修改接口,也不会影响到代理类,方便了代码的扩展。
并且动态代理类只能代理接口,代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类