学习内容
1. Spring框架的IOC基于注解的方式
2. Spring框架整合JUnit单元测试
3. AOP的概述
4. AOP的底层实现原理(了解)
5. AOP功能基于AspectJ的配置文件方式
Spring框架的IOC之注解方式的快速入门
更加简单,替换传统配置文件的方式。
1. 步骤一:导入注解开发所有需要的jar包
引入IOC容器必须的6个jar包
多引入一个:Spring框架的AOP的jar包,spring-aop的jar包
2. 步骤二:创建对应的包结构,编写Java的类
* UserService – 接口
* UserServiceImpl – 具体的实现类
3.步骤三:在src的目录下,创建applicationContext.xml的配置文件,然后引入约束。注意:因为现在想使用注解的方式,那么引入的约束发生了变化
* 需要引入context的约束,具体的约束如下
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!-- bean definitions here -->
</beans>
4.步骤四:在applicationContext.xml配置文件中开启组件扫描
Spring的注解开发:组件扫描
<context:component-scan base-package="com.oath.demo01"/>
注意:可以采用如下配置
<context:component-scan base-package="com.oath"/>
这样是扫描com.oath包下所有的内容
5.步骤五:在UserServiceImpl的实现类上添加注解
@Component(value="userService") -- 相当于在XML的配置方式中 <bean id="userService" class="...">
6.步骤六:编写测试代码
public class SpringDemo01 {
@Test
public void run1() {
ApplicationContext a = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService us = (UserService) a.getBean("userService");
us.save();
}
}
Spring框架中Bean管理的常用注解
@Component:组件.(作用在类上)
Spring中提供@Component的三个衍生注解:(功能目前来讲是一致的)
*@Controller – 作用在WEB层
*@Service – 作用在业务层
*@Repository – 作用在持久层*说明:这三个注解是为了让标注类本身的用途清晰,Spring在后续版本会对其增强
属性注入的注解(说明:使用注解注入的方式,可以不用提供set方法)
*如果是注入的普通类型,可以使用value注解
*@Value – 用于注入普通类型
@Value(value = "小明")
private String name;
*如果注入的是对象类型,使用如下注解
*@Autowired -- 默认按类型进行自动装配
*如果想按名称注入,再添加
*@Qualifier -- 强制使用名称注入
@Autowired
@Qualifier(value = "userDao")
private UserDao userDao;
*@Resource -- 相当于@Autowired和@Qualifier一起使用
* 强调:Java提供的注解
* 属性使用name属性
Bean的作用范围和生命周期的注解
Bean的作用范围注解
*注解为@Scope(value=”prototype”),作用在类上。值如下:
*singleton – 单例,默认值
*prototype – 多例Bean的生命周期的配置(了解)
*注解如下:
*@PostConstruct – 相当于init-method
*@PreDestroy – 相当于destroy-method
Spring框架整合JUnit单元测试
- 为了简化了JUnit的测试,使用Spring框架也可以整合测试
具体步骤
*要求:必须先有JUnit的环境(即已经导入了JUnit4的开发环境)!!*步骤一:在程序中引入:spring-test.jar
*步骤二:在具体的测试类上添加注解
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SpringDemo02 {
@Resource(name = "userService")
public UserService userService;
@Test
public void run01() {
userService.save();
}
}
(需求分析:使用AOP技术对DAO层操作进行增强功能)
Spring框架的核心功能之AOP技术
- 什么是AOP的技术?
*在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程
*AOP是一种编程范式,隶属于软工范畴,指导开发者如何组织程序结构
*AOP最早由AOP联盟的组织提出的,制定了一套规范.Spring将AOP思想引入到框架中,必须遵守AOP联盟的规范
*通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术
*AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型
*利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率 - AOP:面向切面编程.(思想.—解决OOP遇到一些问题)
传统纵向继承:编写一个类,继承该类,重写该类中的这种需要增强的方法,这样确实可以达到我们的目的,但是一旦需要修改的方法所属的类不是一个类,那么就需要在多写很多子类,增加很多方法,编程量就是一个很大的问题,并且后期要修改,工作量也很大。
横向抽取机制::要增强所用到的代码提取到一个类中,然后对需要增强的方法通过代理类去将其增强即可。
- AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码(性能监视、事务管理、安全检查、缓存)
- 为什么要学习AOP
*可以在不修改源代码的前提下,对程序进行增强!!
Spring框架的AOP的底层实现
Srping框架的AOP技术底层也是采用的代理技术,代理的方式提供了两种
基于JDK的动态代理
*必须是面向接口的,只有实现了具体接口的类才能生成代理对象基于CGLIB动态代理
*对于没有实现了接口的类,也可以产生代理,产生这个类的子类的方式
Spring的传统AOP中根据类是否实现接口,来采用不同的代理方式
1. 如果实现类接口,使用JDK动态代理完成AOP
2. 如果没有实现接口,采用CGLIB动态代理完成AOP
JDK的动态代理(代码了解,理解原理)
- 使用Proxy类来生成代理对象的一些代码如下:
*使用JDK的方式生成代理对象
public class MyProxyUtils {
public static UserDao getProxy(final UserDao dao) {
// 使用proxy类生成代理对象
UserDao proxy = (UserDao) Proxy.newProxyInstance(dao.getClass().getClassLoader(),
dao.getClass().getInterfaces(), new InvocationHandler() {
// 代理对象方法一执行,invoke方法就会执行一次
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if ("update".equals(method.getName())) {
System.out.println("!!!!!!记录日志!!!!!!");
// 开启事务
}
// 提交事务
// 让dao类的save或者update方法正常的执行下去
return method.invoke(dao, args);
}
});
// 返回代理对象
return proxy;
}
}
调用:
// 使用工具类,获取到代理对象
UserDao proxy = MyProxyUtils.getProxy(dao);
// 调用代理对象的方法
proxy.save();
proxy.update();
输出结果:
保存客户。。。
!!!!!!记录日志!!!!!!
修改客户。。。。
CGLIB的代理技术(代码了解)
1.引入CBLIB的开发包
*如果想使用CGLIB的技术来生成代理对象,那么需要引入CGLIB的开发的jar包,在Spring框架核心包中已经引入了CGLIB的开发包了。所以直接引入Spring核心开发包即可!
2.编写相关的代码
创建代理的工厂:
public class MyCglibUtils {
public static BookDaoImpl getProxy() {
//创建CGLIB核心的类
Enhancer enhancer = new Enhancer();
// 设置父类
enhancer.setSuperclass(BookDaoImpl.class);
// 设置回调函数
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy)
throws Throwable {
if (method.getName().equals("save")) {
System.out.println("记录日志");
}
return methodProxy.invokeSuper(obj, args);
}
});
//生成代理对象
BookDaoImpl proxy = (BookDaoImpl) enhancer.create();
return proxy;
}
}
BookDaoImpl没有接口(有接口使用jdk代理方式)
public class BookDaoImpl {
public void save() {
System.out.println("保存图书。。。");
}
public void update() {
System.out.println("修改图书。。。");
}
}
测试:
public class Demo02 {
@Test
public void run1(){
BookDaoImpl proxy = MyCglibUtils.getProxy();
proxy.save();
proxy.update();
}
}
结果:
记录日志
保存图书。。。
修改图书。。。
Spring基于AspectJ的AOP的开发
AOP的相关术语
- Joinpoint(连接点) – 所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。标类上需要被增强的方法,(这些方法可以被增强,也可以不增强,也就是说目标类中所有的方法都可以称为是连接点)
- Pointcut(切入点) – 所谓切入点是指我们要对哪些Joinpoint进行拦截的定义。被增强的方法(已经确定这个方法要被增强),切入点就是一个连接点的子集
- Advice(通知/增强) – 所谓通知是指拦截到Joinpoint之后所要做的事情就是通知.通知分为前置通知,后置通知,异常通知,最终通知,环绕通知(切面要完成的功能)
- Introduction(引介) – 引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或Field
- Target(目标对象) – 代理的目标对象。要被增强的类,也就是上面我们写的BookDaoImpl 。
- Weaving(织入) – 是指把增强应用到目标对象来创建新的代理对象的过程。从没被增强到已经增强的过程
- Proxy(代理) – 一个类被AOP织入增强后,就产生一个结果代理类
- Aspect(切面) – 是切入点和通知的结合,以后咱们自己来编写和配置的。切入点 和 通知点 结合形成面(特殊情况:一个切入点 和 一个通知)
AspectJ的XML方式完成AOP的开发
步骤一:创建JavaWEB项目,引入具体的开发的jar包
*先引入Spring框架开发的基本开发包
*再引入Spring框架的AOP的开发包
spring的传统AOP的开发的包:
spring-aop-4.2.4.RELEASE.jar
com.springsource.org.aopalliance-1.0.0.jaraspectJ的开发包:
com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
spring-aspects-4.2.4.RELEASE.jar步骤二:创建Spring的配置文件,引入具体的AOP的schema约束
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
3.步骤三:创建包结构,编写具体的接口和实现类
*com.oath.demo2
*CustomerDao – 接口
*CustomerDaoImpl – 实现类
public void save() {
System.out.println("保存客户。。。");
}
public void update() {
System.out.println("修改客户。。。");
}
4.步骤四:将目标类配置到Spring中
<bean id="customerDao" class="com.oath.demo03.CustomerDaoImpl" />
5.步骤五:定义切面类
public class MyAspectXml {
// 定义通知
public void log() {
System.out.println("!!!!!!记录日志!!!!!!!");
}
}
6.步骤六:在配置文件中定义切面类
<bean id="myAspectXml" class="com.oath.demo03.MyAspectXml" />
7.步骤七:在配置文件中完成aop的配置
<aop:config>
<!-- 引入切面类 -->
<aop:aspect ref="myAspectXml">
<!-- 定义通知类型:切面类的方法和切入点的表达式 -->
<aop:before method="log"
pointcut="execution(public void com.oath.demo03.CustomerDaoImpl.update())" />
</aop:aspect>
</aop:config>
8.完成测试
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class Demo03 {
@Resource(name = "customerDao")
private CustomerDao customerDao;
@Test
public void run01() {
customerDao.save();
customerDao.update();
}
}
结果:
保存客户。。。
!!!!!!记录日志!!!!!!!
修改客户。。。
切入点的表达式
再配置切入点的时候,需要定义表达式,重点的格式如下:execution(public * *(..)),具体展开如下:
*切入点表达式的格式如下:
execution([修饰符] 返回值类型 包名.类名.方法名(参数))修饰符可以省略不写,不是必须要出现的。
返回值类型是不能省略不写的,根据你的方法来编写返回值。可以使用 代替。
包名例如:com.itheima.demo3.BookDaoImpl
首先com是不能省略不写的,但是可以使用 代替
中间的包名可以使用 号代替
如果想省略中间的包名可以使用 ..类名也可以使用 * 号代替,也有类似的写法:*DaoImpl
方法也可以使用 * 号代替
参数如果是一个参数可以使用 * 号代替,如果想代表任意参数使用 ..
<!-- 配置AOP -->
<aop:config>
<aop:aspect ref="myAspectXml">
<!-- 切入点的表达式
1. execution() 固定的,不能不写
2. public 可以省略不写
3. void,返回值可以出现 * 表示任意的返回值,返回值类型不能不写
4. 可以使用 * 代替的,不能不编写的,简写方式:*..*方法
5. *DaoImpl
6. 方法 save*
7. 方法的参数:
-->
<!-- <aop:before method="log" pointcut="execution(public void com.itheima.demo3.CustomerDaoImpl.save())"/> -->
<!-- public 可以省略不写 -->
<!-- <aop:before method="log" pointcut="execution(void com.itheima.demo3.CustomerDaoImpl.save())"/> -->
<!-- void,返回值可以出现 * 表示任意的返回值,返回值类型不能不写 -->
<!-- <aop:before method="log" pointcut="execution(* com.itheima.demo3.CustomerDaoImpl.save())"/> -->
<!-- 包名可以使用 * 代替,不能不写 -->
<!-- <aop:before method="log" pointcut="execution(* com.itheima.*.CustomerDaoImpl.save())"/> -->
<!-- 包的简写的方式,任意的包的结构 -->
<!-- <aop:before method="log" pointcut="execution(* *..*.CustomerDaoImpl.save())"/> -->
<!-- 编写类的写法 -->
<!-- <aop:before method="log" pointcut="execution(* *..*.*DaoImpl.save())"/> -->
<!-- 方法编写 -->
<!-- <aop:before method="log" pointcut="execution(* *..*.*DaoImpl.save*())"/> -->
<!-- 参数列表:出现一个*,表示一个参数,任意参数使用 .. -->
<aop:before method="log" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>
</aop:aspect>
</aop:config>
AOP的通知类型
1. 前置通知
*在目标类的方法执行之前执行。
*配置文件信息:<aop:after method="before" pointcut-ref="myPointcut3"/>
*应用:可以对方法的参数来做校验
2. 最终通知
*在目标类的方法执行之后执行,如果程序出现了异常,最终通知也会执行。
*在配置文件中编写具体的配置:<aop:after method="after" pointcut-ref="myPointcut3"/>
*应用:例如像释放资源、清理现场
3. 后置通知
*方法正常执行后的通知。
*在配置文件中编写具体的配置:<aop:after-returning method="afterReturning" pointcut-ref="myPointcut2"/>
*应用:可以修改方法的返回值
4. 异常抛出通知
*在抛出异常后通知
*在配置文件中编写具体的配置:<aop:after-throwing method="afterThorwing" pointcut-ref="myPointcut3"/>
*应用:包装异常的信息
5. 环绕通知
*方法的执行前后执行。
*在配置文件中编写具体的配置:<aop:around method="around" pointcut-ref="myPointcut2"/>
*要注意:目标的方法默认不执行,需要使用ProceedingJoinPoint对来让目标对象的方法执行。
public class MyAspectXml {
// 定义通知
public void log() {
System.out.println("!!!!!!记录日志!!!!!!!");
}
public void after() {
System.out.println("最终通知");
}
/**
* 环绕通知:方法执行之前和方法执行之后进行通知,默认的情况下,目标对象的方法不能执行的。 需要手动让目标对象的方法执行
*/
public void around(ProceedingJoinPoint joinPoint) {
System.out.println("环绕通知1####");
// 手动让目标对象的方法执行
try {
joinPoint.proceed();
} catch (Throwable e) {
e.printStackTrace();
}
System.out.println("环绕通知2####");
}
}
结果: 保存客户。。。
环绕通知1####
修改客户。。。
环绕通知2####