学游者

  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

JDK动态代理和cglib的动态代理

1、代理

代理在java中用多,主要用来对已有功能做扩展和适配。动态代理就涉及到了反射、jvm的类加载机制等信息。一般用在框架中。例如:spring的AOP就是用动态代理。下面做一些简单讲述。

  • a、接口
public interface Count {

	// 查看账户方法
	public void queryCount();

	// 修改账户方法
	public void updateCount();
}
  • b、实现
public class CountImpl implements Count{

	@Override
	public void queryCount() {
		System.out.println("查看账户方法...");
	}

	@Override
	public void updateCount() {
		System.out.println("修改账户方...");
	}

}
  • c、代理类
public class CountProxy implements Count {

	private CountImpl countImpl;

	public CountProxy(CountImpl countImpl) {
		this.countImpl = countImpl;
	}

	@Override
	public void queryCount() {
		System.out.println("事务处理之前");
		this.countImpl.queryCount();
		System.out.println("事物处理之后");
		System.out.println("");
	}

	@Override
	public void updateCount() {
		System.out.println("事务处理之前");
		this.countImpl.updateCount();
		System.out.println("事物处理之后");
		System.out.println("");
	}
}
  • d、测试类
@Test
	public void staticProxy() {
		CountProxy cp = new CountProxy(new CountImpl());
		cp.updateCount();
		cp.queryCount();
	}

普通的代理都需要实现但前功能的接口,然后在这个接口上实现一个其它功能。这就会造成在大量功能增强的过程中,创建很多的proxy类。

2、动态代理

动态代通过了JVM的反射和类加载机制,减少了proxy类的创建。我们只需要创建一个代理类,就可以在整个项目中使用。

  • JDK的动态代理:

JDK的动态代理只能对实现了接口的类来代理,对于没有实现接口的类不能使用JDK的动态代理。主要分两部分实现。

1、通过:Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h);加载我们需要实现的功能类。

2、通过下面的接口来操作我们的方法。:

public interface InvocationHandler { 
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable; 
}
  • a、接口
public interface BookFacade {
	public void addBook();
}
  • b、实现类
public class BookFacadeImpl implements BookFacade{
	@Override
	public void addBook() {
		System.out.println("增加图书方法...");
	}
}
  • c、代理类
public class BookFacadeProxy 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[] arg2) throws Throwable {
		Object result = null;
		System.out.println("事物开始");
		// 执行
		result = method.invoke(target, arg2);
		System.out.println("事物结束");
		return result;
	}
}
  • d、测试
@Test
	public void dynamicProxy() {
		BookFacadeProxy bfp = new BookFacadeProxy();
		BookFacade bookProxy = (BookFacade) bfp.bind(new BookFacadeImpl());
		bookProxy.addBook();
	}
  • CGLIB的动态代理:采用继承的方式,然后生成子类,并覆盖。
  • a、接口
public interface UserDAO {

	void addUser();
}
  • b、实现
public class UserDAOImpl implements UserDAO{

	@Override
	public void addUser() {
		System.out.println("添加了用户");
	}

}
  • c、代理
public class CglibProxy implements MethodInterceptor {

	private Object target;

	public Object getInstance(Object target) {
		this.target = target;
		Enhancer enhancer = new Enhancer();
		// 回调方法
		enhancer.setCallback(this);
		// 创建代理对象
		return enhancer.create();
	}

	@Override
	public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
		System.out.println("事物开始");
		proxy.invokeSuper(obj, args);
		System.out.println("事物结束");
		return null;
	}
}
  • d、测试
public void cglibProxy() {
		CglibProxy cp = new CglibProxy();
		UserDAO cglibProxy = (UserDAO) cp.getInstance(new UserDAOImpl());
		cglibProxy.addUser();
	}
posted on 2024-06-19 22:08  学游者  阅读(15)  评论(0)    收藏  举报