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

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();
	}


你可能感兴趣的:(dynamicProxy)