利用转账演示事务控制
动态代理:

特点:字节码随用随创建,随用岁加载
作用:不修改源码的基础上对方法增强
分类:
基于接口的动态代理
基于子类的动态代理
基于接口的动态代理:
涉及的类:Proxy
提供者:JDK官方
如何创建代理对象:
使用Proxy类中的newProxyInstance方法
创建代理对象的要求:
被代理类最少实现一个接口,如果没有则不能使用
newProxyInstance方法的参数:
ClassLoader:类加载器
它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。固定写法。
Class[]:字节码数组
它是用于让代理对象和被代理对象有相同方法。固定写法。
InvocationHandler:用于提供增强的代码
它是让我们写如何代理。一般都是写一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。
此接口的实现都是谁用谁写。
public static void main(String[] args) {final Producer producer = new Producer();/*** 动态代理*/producer.saleProduct(113f);Proxy.newProxyInstance(producer.getClass().getClassLoader(), producer.getClass().getInterfaces(), new InvocationHandler() {/*** 作用:执行被代理对象的任何接口方法都会经过该方法* @param proxy 代理对象的引用* @param method 当前执行的方法* @param args 当前执行方法所需的参数* @return 和被代理对象方法有相同的返回值* @throws Throwable*/public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {return null;}});}
基于子类的动态代理
<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>2.1_3</version></dependency>
涉及的类:Enhancer
提供者:cglib
如何创建代理对象:
使用Enhancer类中的create方法
创建代理对象的要求:
被代理类不能是最终类
create方法的参数:
class:字节码
它是用于指定被代理对象的字节码。
callback:用于提供增强的代码
它是让我们写如何代理。一般都是写一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。
此接口的实现都是谁用谁写。
一般写的都是该接口的子接口实现类:MethodInterceptor
public static void main(String[] args) {final Producer producer = new Producer();Producer cglibProducer = (Producer) Enhancer.create(producer.getClass(), new MethodInterceptor() {/*** @param proxy 代理对象的引用* @param method 当前执行的方法* @param args 当前执行方法所需的参数* @param methodProxy 当前执行方法的代理对象* @return* @throws Throwable*/public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {Object returnValue = null;//获取方法执行的参数Float money = (Float) args[0];// 判断当前方法是不是销售if ("saleProduct".equals(method.getName())) {returnValue = method.invoke(producer, money * 0.8f);}return returnValue;}});cglibProducer.saleProduct(12000f);}}
