一个类继承另一个类
然后实例化子类
用子类覆写的方法调用super(),然后在super() 前后写开始和结束的功能
1 占用继承 有实现不了的情况
2 比如有的类是final怎么办?
代理:
更改UserDao 在功能执行前后添加新的功能
1 改源码
2 继承
3 静态代理
静态代理:
代理对象和被代理对象要实现同一个接口
需要保存被代理对象
实现相同的方法,然后在代理对象中,调用被代理的方法
在这个基础之上可以进行扩展
静态代理的问题:
如果接口中添加了新功能(方法),代理对象和被代理对象都需要维护这个功能
所以不是首选
测试静态代理
//代理对象
IUserDao userDao3=new UserDao3();
//被代理对象
IUserDao userDao=new UserDao();
userDao3.setUserDao(userDao);
动态代理:
代理类实现接口InvocationHandler
类体定义private Object target;
get set 方法
覆写invoke方法 参数:
proxy 代理对象 method 被调用的方法 args 方法参数
public Object invoke(Object proxy,Method method,Object[] args)
{
beforeMethod();
使用反射调用方法
method.invoke(target,args);
afterMethod();
return null;
}
//之前执行
public void beforeMethod(){}
//之后执行
public void afterMethod(){}
在测试类中测试
1 被代理的对象
例如 UserDao userDao=new UserDao();
创建代理类
例如 LogInterceptor logInterceptor=new LogInterceptor();
设置被代理对象
logInterceptor.setTarget(userDao);
创建代理对象
IUserDao userDaoProxy=Proxy.newProxyInstance(类加载器,被代理对象实现的接口类.class,代理类对象);(ClassLoader,new Class[] {IUserDao.class},logInterceptor)
第一个参数: 创建代理类对象到内存中需要类加载器,
第二个参数:需要相应的接口,去创建代理类对象
第三个参数: 前两个参数是无法得到被代理类对象的,所以,因为代理类 logInterceptor中定义了Object target 被代理对象,
还有覆写的invoke方法中传参的method调用了method.invoke(target,args)
所以到前两个参数创建了的代理类对象会去调用第三个参数的invoke方法,
就可以解决无法得到被代理类对象的尴尬问题
userDaoProxy.login(null);
AOP 面向切面
1 导包
2 引入约束
3 切面类
注解设置AOP
<context:annotation-config />
<context:component-scan base-package=“dsf” />
开启注解AOP
<aop:aspectj-autoproxy/>
类体中:
- Aspect : 切面
- Join point : 黏着点
- Pointcut : 切点
- Around : 真正执行
// 返回值 * 就是任意返回值 或者没有 都可以
@Pointcut("execution(public * com.service..*.login(..))")
public void myMethod() {
// 会调用注解里面的login方法
}
// . 表示子目录 .. 表示后代目录 * 表示所有
@Around("myMethod()")
// 环绕 : 前后都有,相当于 before和after一起使用
XML设置AOP
<!-- XML设置AOP -->
<!-- 实例化切面类 -->
<bean id="logInterceptor" class="dsf.aop.LogInterceptor"></bean>
<!-- 设置AOP -->
<aop:config>
<aop:aspect id="logAspect" ref="logInterceptor">
<!-- 切入点,也就是被代理的方法 -->
<aop:pointcut
expression="execution(public * dsf.service..*.login(..))"
id="pointcut" />
<!-- 用下面 pointcut这种方式 就不需要 aop:pointcut标签了 -->
<!-- <aop:after method="before" pointcut="execution(public * dsf.service..*.login(..))"/> -->
<!-- 也可以 通过pointcut-ref 去找 aop:pointcut标签-->
<aop:after method="after" pointcut-ref="pointcut" />
<aop:before method="before" pointcut-ref="pointcut" />
<!-- around 和 after+before 一般不会一起出现 -->
<!-- <aop:around method="aroundMethod" pointcut-ref="pointcut" /> -->
</aop:aspect>
</aop:config>