Spring_Day02
一、IoC容器装配Bean_基于注解配置方式
1. 基于xml配置的方式装配Bean
1. 导包,4个核心包:cores,context,expression,beans. 2个日志包:loging,log4j
2. 编写service层与dao层
3. 配置applicationContext.xml,中service层与dao层的两个bean。
4. web层(这里junit模拟测试)调用业务方法
==========================================================================
CustomerServiceImpl:
声明CustomerDAO对象用于接收属性注入的CustomerDAOImpl对象的引用。
必须提供UserDAO的setter方法,用于Spring框架根据配置进行依赖注入。
public class CustomerServiceImpl implements CustomerService {
// 声明CustomerDAO
private CustomerDAO customerDAO;
// 提供setter方法Spring框架属性注入
public void setCustomerDAO(CustomerDAO customerDAO) {
this.customerDAO = customerDAO;
}
@Override
// 模拟用户注册
public void save() {
// 业务逻辑
System.out.println("业务层......CustomerServiceImpl....save...");
// 调用dao层
customerDAO.save();
}
}
---------------------------------------------------------------------------
CustomerDAOImpl:
操作数据库的代码。模拟用户注册。
public class CustomerDAOImpl implements CustomerDAO {
@Override
//模拟客户注册
public void save() {
//数据库操作
System.out.println("dao层...CustomerDAOImpl....save");
}
}
---------------------------------------------------------------------------
配置核心文件applicationContext.xml:
---------------------------------------------------------------------------
web层:(Junit模拟调用业务层)
@Test
public void test() {
//创建Spring的ApplicationContext工厂对象
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
//获取CustomerServiceImpl的Bean对象
CustomerService customerService = (CustomerService) applicationContext.getBean("customerService");
//调用save方法
customerService.save();
}
2. 基于Spring注解装配Bean
1. 导包,4个核心包:cores,context,expression,beans. 2个日志包:loging,log4j
1. 额外添加一个aop的jar包
2. 编写service层与dao层,并完成注解注册
3. 核心配置文件applicationContext.xml中开启注解扫描
4. web层(这里junit模拟测试)调用业务方法
============================================================================
2.1. 导jar包
注解装配Bean与XML方式装配的jar包相同,但还需要额外导入一个aop的jar包。
2.2. 编写service层与dao层的注册
xml的做法:
注解做法 : spring2.5引入 @Component 注解
CustomerServiceImpl:
//注册bean,表明这是被Spring管理的bean.括号内相当于xml配置中的id/name
//@Componet //如果不写value,默认类名第一个字母小写.习惯写接口
@Component("customerService")
//@Service("customerService")
public class CustomerServiceImpl implements CustomerService {
@AutoWired// 声明CustomerDAO,用于接收属性注入的CustomerDAOImpl对象的引用
@Qualifier("customerDAO")
private CustomerDAO customerDAO;
@Override
// 模拟用户注册
public void save() {
// 业务逻辑
System.out.println("业务层......CustomerServiceImpl....save...");
// 调用dao层
customerDAO.save();
}
}
--------------------------------------------------------------------------
//注册bean,表明这是被Spring管理的bean,相当于xml配置中的
//如果不写value,默认是第一个字母小写的类名.
//@Repository是dao层的bean类注册的注解,是Component的子注解
@Component("customerDAO")
//@Repository("customerDAO")
public class CustomerDAOImpl implements CustomerDAO {
@Override
//模拟客户注册
public void save() {
//数据库操作
System.out.println("dao层...CustomerDAOImpl....save");
}
}
2.3. applicationContext.xml中开启注解扫描
1. 引入名称空间:
找到xsd-configuration.html中contextschema的头约束引入。
配置本地提示:XML Catalog
注意:所有的名称空间都要基于Beans的名称空间。
2. 编写applicationContext.xml开启注解
优化:
1. 注解扫描配置
在配置组件扫描时,Spring会自动开启注解功能,所以开启注解功能可以不配置。
即: 具有 作用!
2. 衍生注解
实际开发中,使用的是@Component三个衍生注解(“子注解”)
子注解的作用:有分层的意义(分层注解)。功能与Component一致,以后会增强
@Repository用于对DAO实现类进行标注
@Service用于对Service实现类进行标注
@Controller用于对Controller实现类进行标注
3. Bean属性的依赖注入
3.1 简单数据类型依赖注入(了解)
Spring3.0后,提供 @Value注解,可以完成简单数据的注入
3.3 复杂类型数据依赖注入【掌握】
注解实现属性依赖注入,将注解加在setXxx方法上 或者 属性定义上。
四种方式:
1. 使用@Value结合pElL表达式
@Value("#{customerDAO}") // 找到设置id/name为customer的bean
private CustomerDAO customerDAO;
2. 使用@Autowired 结合 @Qualifier
// 声明CustomerDAO,用于接收属性注入的CustomerDAOImpl对象的引用
@Autowired
@Qualifier("customerDAO")
private CustomerDAO customerDao;
单独使用@Autowired 按照类型注入
3. JSR-250标准(jdk) 提供@Resource
@Resource(name="customerDAO")
private CustomerDAO customerDao;
4. SR-330标准(jdk) 提供 @Inject
需要先导入 javax.inject 的 jar
@Inject
@Named("customerDAO")
private CustomerDAO customerDao;
总结:
4. Bean的初始化与销毁(注解)
使用 @PostConstruct 注解, 标明初始化方法 ---相当于 init-method 指定初始化方法
使用 @PreDestroy 注解, 标明销毁方法 ----相当于 destroy-method 指定对象销毁方法
==========================================================================
//不写value默认类名第一个字母小写作为该bean的id/name
@Component
public class InitAndDestroyDemo {
//执行的方法
public void save() {
System.out.println("InitAndDestroyDemo.....save..........");
}
@PostConstruct
public void init() {
System.out.println("InitAndDestroyDemo.......初始化 .......");
}
//销毁前调用的方法
@PreDestroy
public void destroy() {
System.out.println("InitAndDestroyDemo......销毁..........");
}
}
测试:
@Test
public void test() {
//加载配置文件获取Spring工厂对象
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
//获取Bean对象
InitAndDestroyDemo demo =(InitAndDestroyDemo) applicationContext.getBean("initAndDestroyDemo");
//调用业务逻辑
demo.save();
//手动关闭Spring容器,使demo对象销毁
ClassPathXmlApplicationContext context = (ClassPathXmlApplicationContext)applicationContext;
context.close();
}
注意:
要执行对象的销毁方法的条件:
1. 必须是单例bean(Spring容器close时,单例bean执行销毁方法),多例bean不归Spring容器管理
2. 必须调用Spring容器的close方法
结果:
5. Bean的作用域
通过@Scope注解,指定Bean的作用域(默认是 singleton 单例)
@Component //默认注册bean的id/name为第一个字母小写的类名
@Scope("prototype")//多例
public class ScopeDemo {
}
测试:
另外,XML和注解也可以混合配置
二、Spring的junit测试集成
Spring提供 test.jar 可以整合junit。
好处:可以简化测试代码(不需要手动创建上下文)
1. 导入jar包,Spring的基本包.Junit包.spring-test.jar
2. 通过@RunWith注解,使用junit整合spring
在测试类运行前的初始化的时候,会自动创建ApplicationContext对象
3. 通过@Autowired注解,注入需要测试的对象
4. 测试
=====================================================================================
//Junit和Spring整合,会自动开启注解功能,自动构建Spring容器,将当前类注册为bean
@RunWith(SpringJUnit4ClassRunner.class)
//为Spring容器指定配置文件
@ContextConfiguration(locations="classpath:applicationContext.xml")
public class SaveAndDestoryTest {
//提供接收属性注入的需要测试的bean对象的引用
@Autowired
@Qualifier("initAndDestroyDemo")
private InitAndDestroyDemo demo;
@Test
public void test() {
demo.save();
}
}
Spring整合Junit测试后,无需手动销毁,最后会closeSpring容器。
三、 AOP面向切面编程的相关概念
3.1 什么是AOP?
AOP (Aspect Oriented Programing) 称为:面向切面编程,它是一种编程思想。
AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码的编写方式(性能监视、事务管理、安全检查、缓存,日志记录等)。
传统的纵向体系代码复用:
横向抽取机制(AOP思想):
开闭原则:
AOP 思想: 基于代理思想,对原来目标对象,创建代理对象,在不修改原对象代码情况下,通过代理对象,调用增强功能的代码,从而对原有业务方法进行增强 !
切面:需要代理一些方法和增强代码 。
3.2 AOP的应用场景
场景一: 记录日志
场景二: 监控方法运行时间 (监控性能)
场景三: 权限控制
场景四: 缓存优化 (第一次调用查询数据库,将查询结果放入内存对象, 第二次调用, 直接从内存对象返回,不需要查询数据库 )
场景五: 事务管理 (调用方法前开启事务, 调用方法后提交关闭事务 )
3.3 Spring AOP编程两种方式
Spring内部支持两套AOP编程的方案:
- Spring 1.2 开始支持AOP编程 (传统SpringAOP 编程),编程非常复杂
- Spring 2.0 之后支持第三方 AOP框架(AspectJ ),实现另一种 AOP编程 -- 推荐
3.4 AOP编程相关术语
JointPoint(连接点) : 在Spring中即指 拦截到的所有方法。
PointCut(切入点) : 指对哪些JointPoint进行拦截的定义。即对哪些方法进行增强。
Advice(通知/增强) : 就是对切入点(要增强的方法)做怎样的增强。
Target(目标对象) : 代理的目标对象
Weaving(织入) : 将增强应用到目标对象来创建代理对象的过程
Proxy(代理) : 一个类被AOP织入增强后,返回的就是一个代理对象
Aspect : 切入点和通知的结合,就是定义对哪个方法进行怎么样的增强
4. AOP编程底层实现机制
AOP 就是要对目标进行代理对象的创建。
Spring AOP是基于动态代理的,基于两种动态代理机制: JDK动态代理和CGLIB动态代理
动态代理和静态代理区别?
动态代理:在虚拟机内部,运行的时候,动态生成代理类(运行时生成,runtime生成) ,并不是真正存在的类, Proxy$$ (Proxy$$Customer)
静态代理:存在代理类 (例如:struts2 Action的代理类 ActionProxy)
4.1. JDK动态代理
JDK动态代理,针对目标对象的接口进行代理 ,动态生成接口的实现类(必须有接口)
过程:
1. 必须对接口生成代理
2. 用Proxy对象,通过newProxyInstance方法为目标创建代理对象.该方法接收三个参数 :
1. 目标对象类加载器
2. 目标对象实现的接口
3. 代理后的处理程序InvocationHandler
3. 实现InvocationHandler接口中invoke方法,在目标对象每个方法调用时,都会执行invoke
例:
1. 编写UserService接口与UserServiceImpl实现类.(JDK动态代理必须有接口)
2. 编写JDK动态代理工厂生成动态代理,对目标对象进行增强
3. 测试
UserService:
public interface UserService {
//模拟用户注册
public void save();
}
UserServiceImpl:
public class UserServiceImpl implements UserService {
@Override
public void save() {
// 目标:使用jdk动态代理,增强该方法
System.out.println("用户注册.........");
}
}
--------------------------------------------------------------------------
JDK动态代理工厂类:
public class JdkProxyFactory implements InvocationHandler {
//用于接收目标对象
private Object taget;
//构造方法传入目标对象
public JdkProxyFactory(Object taget) {
this.taget = taget;
}
// 获取目标的代理子对象
public Object getJdkProxy() {
// 参数1:目标对象的类加载器
// 参数2:目标对象实现的所有接口(jdk通过目标接口生成代理子对象)
// 参数3: 方法调用处理程序InvocationHandler,目标对象调用方法时会执行该处理器的invoke方法(可以写增强的代码)
Object proxy = Proxy.newProxyInstance(taget.getClass().getClassLoader(), //
taget.getClass().getInterfaces(), //
this);
return proxy;
}
@Override
// 参数1:代理对象
// 参数2: 目标对象调用的方法
// 参数3:目标对象调用的方法的参数
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 增强的业务
writeLog();
// 执行目标对象原来的方法
Object object = method.invoke(taget, args);
// 还可以进行后置增强
return object;// 返回目标对象
}
// 通知/增强的代码
public void writeLog() {
System.out.println("开始记录日志........Jdk动态代理.....");
}
}
----------------------------------------------------------------------------
测试:
@Test
// 使用jdk动态代理增强目标对象
public void jdkProxyTest() {
// 1.获取目标对象
UserService userService = new UserServiceImpl();
// 2. 通过代理工厂获取代理对象
// 2.1 获取Jdk代理对象的工厂
JdkProxyFactory jdkProxyFactory = new JdkProxyFactory(userService);
// 将通知/增强应用到目标对象,生成代理对象的过程(织入)
// 代理对象是目标对象的接口的子对象
UserService jdkProxy = (UserService) jdkProxyFactory.getJdkProxy();
// 3. 调用方法
jdkProxy.save();
}
jdk动态代理的缺点: 只能面向接口代理,不能直接对目标类进行代理 (如果没有接口,则不能使用)。
4.2. cglib动态代理
Cglib的引入为了解决类的直接代理问题(生成代理子类),不需要接口也可以代理。
该代理方式需要相应的jar包,但不需要导入。因为Spring core包已经包含cglib ,而且同时包含了cglib 依赖的asm的包(动态字节码的操作类库)
示例:
1. 编写ProductService(没有接口)
2. 编写cglib代理工厂用于生成代理对象,并对目标对象进行增强
3. 测试
-----------------------------------------------------------------------------
ProductService:
public class ProductService {
// 模拟用户登录
public void login() {
System.out.println("ProductService.....用户要登陆了............");
}
}
-----------------------------------------------------------------------------
cglib代理工厂类:
public class CglibProxyFactory implements MethodInterceptor {
// 用于接收存储目标对象
private Object taget;
// 构造方法传入目标对象
public CglibProxyFactory(Object taget) {
this.taget = taget;
}
// 获得cglib动态代理对象
public Object getCglibProxy() {
// 1. 获得代理对象增强器
Enhancer enhancer = new Enhancer();
// 2. 设置代理对象的父类为目标对象
enhancer.setSuperclass(taget.getClass());
// 3. 设置每次代理对象调用方法时,都会执行的回调函数
enhancer.setCallback(this);
// 返回代理对象
return enhancer.create();
}
@Override
// 代理对象执行目标对象方法时,调用该方法
// 参数1:代理对象 , 参数2 : 调用的目标对象的方法, 参数3:调用的目标对象的方法的参数 , 参数4:代理对象的方法(已经被增强过的,目标对象的子类)
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
// 通知/增强,对目标对象进行增强
writeLog();
// 执行原来的方法,获得目标对象
Object object = method.invoke(taget, args);
// 返回目标对象
return object;
}
// 通知/增强.的方法
public void writeLog() {
System.out.println("开始写日志了......cglib......");
}
}
--------------------------------------------------------------------------
测试:
@Test
// 使用cglib动态代理增强目标对象(不需要有接口)
public void cglibProxyTest() {
// 1. 获取目标对象
ProductService productService = new ProductService();
// 2. 通过cglib代理对象工厂获得代理对象
CglibProxyFactory cglibProxyFactory = new CglibProxyFactory(productService);
// 织入:将增强应用到目标对象,获得代理对象
// cglib代理对象是目标对象的子类
ProductService cglibProxy = (ProductService) cglibProxyFactory.getCglibProxy();
//3. 调用方法
cglibProxy.login();
}
4.3.代理知识小结
区别 :
- jdk动态代理:基于接口的代理,会生成目标对象的接口类型的子对象。
- cglib动态代理:基于类的代理,会生成目标对象类型的子对象。
1、Spring AOP 优先对接口进行代理 (使用Jdk动态代理)
2、如果目标对象没有实现任何接口,才会对类进行代理 (使用cglib动态代理)
5. 基于AspectJ的SpringAOP编程(xml配置)
Spring的aop有两套编程方案:传统的aop和aspectj的aop。
5.1 快速入门
使用AspectJ的方式进行Spring的AOP开发的三个步骤:
1. 确定目标对象(bean),注册bean
2. 编写通知/增强(advice),注册bean
3. 配置切入点(Pointcut)和切面(aspect),切面即为建立切入点与通知的关联
4. 测试
------------------------------------------------------------------------
在测试用例时,可以不用配置 开启注解,当然配置也可以。
1. 确定目标对象UserServiceImpl(有接口),即增强UserServiceImpl中的所有方法
public class UserServiceImpl implements UserService {
@Override
// 使用Spring的AOP编程增强
public void save() {
System.out.println("用户注册.........");
}
}
目标对象ProductService(没有接口):
public class ProductService {
//模拟添加商品..对该方法进行增强
public void save() {
System.out.println("ProductService.......添加商品...........");
}
}
为目标对象注册bean:
2. 编写通知/增强类,并注册bean
//通知/增强类,对切入点进行增强
public class MyAspect {
public void writeLog() {
System.out.println("开始写日志了......Spring..aop开发");
}
}
为通知注册bean:
3. 配置切面.建立切入点 与 通知的关联。确定哪些方法要怎样进行增强
4. 测试
1.一定要导入Junit包!! 以及spring的test包
2. 注意ContextConfiguration中的写法(locations="classpath:appliactionContext.xml")
3. 测试必须写在一个test中,写两个test分开测试会报错
@RunWith(SpringJUnit4ClassRunner.class)//Spring框架整合Junit
@ContextConfiguration(locations="classpath:applicationContext.xml")//Spring容器加载配置文件
public class SpringAopTest {
//注入的测试bean
@Autowired
@Qualifier("userService")//指定id/名称的bean注入
private UserService userService;
@Autowired//以默认的类型ProductService注入
private ProductService productService;
@Test
public void test() {
//有接口
userService.save();
//没有接口
productService.save();
}
}
结果:
小结:
1. 有接口的使用的是jdk动态代理,没有接口的使用的是cglib动态代理
2. 这里使用的都是全局pointcut切入点,也可以将它们写在aop-aspect内部,那么这些pointcut就只针对这个aspect切面有效
3. 一个通知可以增强多个连接点,一个连接点可以被多次增强。
5.2 配置文件applicationConext.xml的细节
- 因为用到了注解,因此在导入头约束时要有context的名称空间
- 因为使用了aop开发,因此导入头约束时也要有aop的名称空间
- 所有的名称空间都要基于beans
头约束从官方提供的docs/spring-framework-reference/html/xsd-configuration.html
中查找
5.3. 切入点表达式语法
切入点表达式:
-
bean(bean Id/bean name)
例如: bean(customerService) 增强customerService的bean中所有方法,支持通配符 bean(*Service)增强以Service结尾的bean中的所有方法
-
execution(<访问修饰符>?<返回类型>空格<方法全名>(<参数>)<异常>?)
例如: execution(* com.itdream.spring.a_jdkproxy.CustomerServiceImpl.*(..)) 增强该bean对象中的所有方法 execution(* com.itdream.spring..*.*(..)) 增强spring包和子包所有bean所有方法
-
within(包.类)
例如: within(cn.itcast.spring..*) 增强spring包和子包所有bean“所有方法 ”
-
this(完整类型)/target(完整类型)
this对某一个类-(代理对象有效),target对代理对象无效(只对目标对象有效) this(cn.itcast.spring.a_jdkproxy.CustomerServiceImpl) 增强类型所有方法(对代理对象有效)
【AspectJ类型匹配的通配符】:
*:匹配任何数量字符;
..:匹配任何数量字符的重复,如在类型模式中匹配任何数量子包;而在方法参数模式中匹配任何数量参数。
+:匹配指定类型的子类型;仅能作为后缀放在类型模式后边。
5.4 AspectJ的通知类型(5+种)
- 前置通知(Before advice):在目标方法执行之前进行增强
- 后置通知(After returning advice):在目标方法执行之后进行增强
- 环绕通知(Around advice):在目标方法执行的前后进行增强
- 抛出通知(After throwing advice):在目标方法执行时抛出异常的时候进行增强
- 最终通知(After(finally) advice):在目标方法执行之后,不管是否有异常,都会进行增强
- 引介通知(Decare Parents):在目标类中增加一些新的方法和属性
5.5 各种通知的应用场景
5.5.1. Before前置通知
应用: 权限控制 (权限不足,抛出异常)、 记录方法调用信息日志
模拟:权限控制
1. 确定目标对象bean(要增强的对象的某些方法)
2. 编写通知/增强方法
3. 配置切面
目标对象bean:
public class UserServiceImpl implements UserService {
@Override
public void save() {
System.out.println("用户注册.........");
}
public int findCount() {
System.out.println("余额查询........");
return 100;
}
}
注册bean:
-----------------------------------------------------------------------------
编写通知/增强类:
//通知/增强类
public class MyAspect {
//前置增强
//参数:连接点:就是可以拦截到的方法
//需求:应用:权限控制(权限不足就抛出异常)、记录方法调用信息日志
public void before(JoinPoint joinPoint) {
System.out.println("前置增强..............");
System.out.println("增强的对象是:"+joinPoint.getTarget().getClass().getName());
//模拟权限控制
//连接点的前面Signature就是方法
if("findCount".equals(joinPoint.getSignature().getName())) {
throw new RuntimeException("对不起,您没有这个权限!");
}
}
}
注册bean:
-----------------------------------------------------------------------------
配置切面:
-----------------------------------------------------------------------------
测试:
方法运行到findCount时抛出异常。达到了权限控制的目的。
5.5.2. AfterReturing 后置通知
特点:在目标方法运行后,返回值后执行增强代码逻辑。
应用场景:与业务相关的,如网上营业厅查询余额后,自动下发短信。
分析: 后置通知可以获取到目标方法返回值,如果想对返回值进行操作,使用后置通知(但不能修改目标方法返回 )
还是以上面的目标bean作为目标进行后置增强:
1. 目标对象bean
2. 通知/增强类(一个类中可以写多个通知方法,配置中method可以选择通知方法)
//后置增强
//参数1:连接点 , 参数2:返回值类型Object,参数名随意,但配置时,需要配置返回值的名称与其一致
public void afterReturnning(JoinPoint joinPoint,Object returnValue) {
System.out.println("后置通知:系统日志:当前下发了短信的方法是:"+joinPoint.getSignature().getName());
//获取到查询结果,向用户下发短信
System.out.println("尊敬的用户,您的余额是:"+returnValue);
}
注册bean之前已经完成。
--------------------------------------------------------------------------------------
3. 配置切面
这里的pointcut的定义使用了局部定义,只针对这个切面有效
测试:
5.5.3. Around 环绕通知
特点:目标执行前后,都进行增强(控制目标方法执行)
应用场景:日志、缓存、权限、性能监控、事务管理
环绕通知的增强代码的方法要求:
接受的参数:ProceedingJoinPoint(可执行的连接点)
返回值:Object返回值
抛出Throwable异常。
=======================================================================
1. 确定目标bean,注册目标bean
2. 编写通知
3. 配置切面(切入点与通知的关联),即目标对象哪些方法需要通知增强。
通知:
//环绕通知
//需求:日志,缓存,权限,性能监控,"事务管理"
//三个特点:
//参数:正在执行的连接点(方法)
//必须抛出异常Throwable
//必须返回目标对象
public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
//在执行方法之前开启事务
System.out.println("开启事务......");
//执行目标方法
Object object = joinPoint.proceed();
//目标方法执行完毕后提交事务
System.out.println("提交事务");
//将目标对象返回
return object;
}
配置切面:
测试:
5.5.4. AfterThrowing 抛出通知
作用:目标代码出现异常,通知执行。记录异常日志、通知管理员(短信、邮件)
应用场景:处理异常(一般不可预知),记录日志
例子:
1. 确定目标对象bean
2. 编写通知/增强方法,类
//抛出通知:在目标对象的方法发生异常情况下,拦截增强
//需求:处理异常(一般不可预知),记录日志,通知管理员(短信,邮件)
//参数1:连接点信息,参数2:异常,Throwable类型,参数名称随意,配置这个名字
public void afterThrowing(JoinPoint joinPoint,Throwable ex) {
//发生异常时,通知管理员
System.out.println("尊敬的管理员,发生异常了,发生异常的类是:"+joinPoint.getTarget().getClass().getName()
+",发生异常的方法是:"+joinPoint.getSignature().getName()
+",发生异常的信息是:"+ex.getMessage());
}
---------------------------------------------------------------------------------------------------
3. 配置切面(切入点与增强的关联)
测试:
创造异常:
public class UserServiceImpl implements UserService {
@Override
public void save() {
System.out.println("用户注册.........");
}
public int findCount() {
int i = 1/0;//创造异常
System.out.println("余额查询........");
return 100;
}
}
测试结果:
5.5.5. After 最终通知
作用:不管目标方法是否发生异常,最终通知都会执行(类似于finally代码功能)
应用场景:释放资源 (关闭文件、 关闭数据库连接、 网络连接、 释放内存对象 )
1. 确定目标对象bean
2. 编写通知/增强类/方法
//最终通知:不管方法是否成功执行,都会拦截增强
//需求:释放资源 (关闭文件、 关闭数据库连接、 网络连接、 释放内存对象 )
public void after(JoinPoint joinPoint) {
//释放资源
System.out.println("开始释放资源了,该连接点的信息是:"+joinPoint.toLongString());
}
3. 配置切面(切入点与通知之间的关联)
测试:不管是否有异常都会执行最终通知,进行资源的释放)
五种通知小结:
只要掌握Around(环绕通知)通知类型,就可实现其他四种通知效果。
可以在环绕通知的方法中编写如下代码:
try {
//前置通知
Object result = proceedingJoinPoint.proceed();
//后置通知
}catch(Exception){
//抛出通知
}finally{
//最终通知
}
5.5.6. 各种Advice方法可接收的参数和返回值小结(参考)
方法格式:
public returnType method (param)
public 返回值类型 方法名 (参数类型 参数名)
其中参数类型为JoinPoint接口类型,返回值类型为void或Object
注意:JoinPoint是org.aspectj.lang.JoinPoint包下的