Spring是一个开源框架,Spring是于2003年兴起的一个轻量级的Java开发框架,由Rod Johnson在其著做Expert One-On-One J2EE Development and Design中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而建立的。框架的主要优点之一就是其分层架构,分层架构容许使用者选择使用哪个组件,同时为J2EE应用程序开发提供集成的框架。Spring使用基本的JavaBean来完成之前只可能由EJB完成的事情。然而,Spring的用途不只限于服务器端的开发。从简单性、可测试性和松耦合的角度而言,任何Java应用均可以从Spring中受益。Spring的核心是控制反转(IoC)和面向切面(AOP)。简单来讲,Spring是一个分层的JavaSE/EEfull-stack(一站式)轻量级开源框架。java
为何说Spring是一个一站式的轻量级开源框架呢?EE开发可分红三层架构,针对JavaEE的三层结构,每一层Spring都提供了不一样的解决技术。程序员
• WEB层:SpringMVCspring
• 业务层:Spring的IoCexpress
• 持久层:Spring的JDBCTemplate(Spring的JDBC模板,ORM模板用于整合其余的持久层框架)编程
从上面的简要介绍中,咱们要知道Spring的核心有两部分:设计模式
• IoC:控制反转。安全
举例来讲,在以前的操做中,比方说有一个类,咱们想要调用类里面的方法(不是静态方法),就要建立类的对象,使用对象调用方法实现。对于Spring来讲,Spring建立对象的过程,不是在代码里面实现的,而是交给Spring来进行配置实现的。bash
AOP:面向切面编程。服务器
AOP: Aspect Oriented Programming 面向切面编程。 面向切面编程(也叫面向方面):Aspect Oriented Programming(AOP),是目前软件开发中的一个热点。利用AOP能够对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度下降,提升程序的可重用性,同时提升了开发的效率。 AOP是OOP的延续,是(Aspect Oriented Programming)的缩写,意思是面向切面(方面)编程。 主要的功能是:日志记录,性能统计,安全控制,事务处理,异常处理等等。 主要的意图是:将日志记录,性能统计,安全控制,事务处理,异常处理等代码从业务逻辑代码中划分出来,经过对这些行为的分离,咱们但愿能够将它们独立到非指导业务逻辑的方法中,进而改 变这些行为的时候不影响业务逻辑的代码。架构
能够经过预编译方式和运行期动态代理实如今不修改源代码的状况下给程序动态统一添加功能的一种技术。AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,AOP能够说也是这种目标的一种实现。
假设把应用程序想成一个立体结构的话,OOP的利刃是纵向切入系统,把系统划分为不少个模块(如:用户模块,文章模块等等),而AOP的利刃是横向切入系统,提取各个模块可能都要重复操做的部分(如:权限检查,日志记录等等)。因而可知,AOP是OOP的一个有效补充。
注意:AOP不是一种技术,其实是编程思想。凡是符合AOP思想的技术,均可以当作是AOP的实现。
Aop, aspect object programming 面向切面编程
功能: 让关注点代码与业务代码分离!
关注点,重复代码就叫作关注点;
关注点造成的类,就叫切面(类)!
面向切面编程,就是指 对不少功能都有的重复的代码抽取,再在运行的时候网业务方法上动态植入“切面类代码”。
执行目标对象方法,动态植入切面代码。
能够经过切入点表达式,指定拦截哪些类的哪些方法; 给指定的类在运行的时候植入切面类代码。
经过代理控制对象的访问,能够详细访问某个对象的方法,在这个方法调用处理,或调用后处理。既(AOP微实现) ,AOP核心技术面向切面编程。
SpringAOP、事物原理、日志打印、权限控制、远程调用、安全代理 能够隐蔽真实角色
静态代理(静态定义代理类)
动态代理(动态生成代理类)
Jdk自带动态代理
Cglib 、javaassist(字节码操做库)
由程序员建立或工具生成代理类的源码,再编译代理类。所谓静态也就是在程序运行前就已经存在代理类的字节码文件,代理类和委托类的关系在运行前就肯定了。
public interface IUserDao {
void save();
}
public class UserDao implements IUserDao {
public void save() {
System.out.println("已经保存数据...");
}
}
代理类
public class UserDaoProxy implements IUserDao {
private IUserDao target;
public UserDaoProxy(IUserDao iuserDao) {
this.target = iuserDao;
}
public void save() {
System.out.println("开启事物...");
target.save();
System.out.println("关闭事物...");
}
}
复制代码
1.代理对象,不须要实现接口
2.代理对象的生成,是利用JDK的API,动态的在内存中构建代理对象(须要咱们指定建立代理对象/目标对象实现的接口的类型)
3.动态代理也叫作:JDK代理,接口代理
1)原理:是根据类加载器和接口建立代理类(此代理类是接口的实现类,因此必须使用接口 面向接口生成代理,位于java.lang.reflect包下)
2)实现方式:
缺点:jdk动态代理,必须是面向接口,目标业务类必须实现接口
// 每次生成动态代理类对象时,实现了InvocationHandler接口的调用处理器对象
public class InvocationHandlerImpl implements InvocationHandler {
private Object target;// 这其实业务实现类对象,用来调用具体的业务方法
// 经过构造函数传入目标对象
public InvocationHandlerImpl(Object target) {
this.target = target;
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = null;
System.out.println("调用开始处理");
result = method.invoke(target, args);
System.out.println("调用结束处理");
return result;
}
public static void main(String[] args) throws NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
// 被代理对象
IUserDao userDao = new UserDao();
InvocationHandlerImpl invocationHandlerImpl = new InvocationHandlerImpl(userDao);
ClassLoader loader = userDao.getClass().getClassLoader();
Class<?>[] interfaces = userDao.getClass().getInterfaces();
// 主要装载器、一组接口及调用处理动态代理实例
IUserDao newProxyInstance = (IUserDao) Proxy.newProxyInstance(loader, interfaces, invocationHandlerImpl);
newProxyInstance.save();
}
}
复制代码
原理:利用asm开源包,对代理对象类的class文件加载进来,经过修改其字节码生成子类来处理。
使用cglib[Code Generation Library]实现动态代理,并不要求委托类必须实现接口,底层采用asm字节码生成框架生成代理类的字节码
public class CglibProxy implements MethodInterceptor {
private Object targetObject;
// 这里的目标类型为Object,则能够接受任意一种参数做为被代理类,实现了动态代理
public Object getInstance(Object target) {
// 设置须要建立子类的类
this.targetObject = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(this);
return enhancer.create();
}
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("开启事物");
Object result = proxy.invoke(targetObject, args);
System.out.println("关闭事物");
// 返回代理对象
return result;
}
public static void main(String[] args) {
CglibProxy cglibProxy = new CglibProxy();
UserDao userDao = (UserDao) cglibProxy.getInstance(new UserDao());
userDao.save();
}
}
复制代码
java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。
而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,经过修改其字节码生成子类来处理。
Spring中。
一、若是目标对象实现了接口,默认状况下会采用JDK的动态代理实现AOP
二、若是目标对象实现了接口,能够强制使用CGLIB实现AOP
三、若是目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换
JDK动态代理只能对实现了接口的类生成代理,而不能针对类 。 CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法 。 由于是继承,因此该类或方法最好不要声明成final ,final能够阻止继承和多态。
<aop:aspectj-autoproxy></aop:aspectj-autoproxy> 开启事物注解权限
@Aspect 指定一个类为切面类
@Pointcut("execution(* com.service.UserService.add(..))") 指定切入点表达式
@Before("pointCut_()") 前置通知: 目标方法以前执行
@After("pointCut_()") 后置通知:目标方法以后执行(始终执行)
@AfterReturning("pointCut_()") 返回后通知: 执行方法结束前执行(异常不执行)
@AfterThrowing("pointCut_()") 异常通知: 出现异常时候执行
@Around("pointCut_()") 环绕通知: 环绕目标方法执行
@Component
@Aspect
public class AopLog {
// 前置通知
@Before("execution(* com.service.UserService.add(..))")
public void begin() {
System.out.println("前置通知");
}
// 后置通知
@After("execution(* com.service.UserService.add(..))")
public void commit() {
System.out.println("后置通知");
}
// 运行通知
@AfterReturning("execution(* com.service.UserService.add(..))")
public void returning() {
System.out.println("运行通知");
}
// 异常通知
@AfterThrowing("execution(* com.service.UserService.add(..))")
public void afterThrowing() {
System.out.println("异常通知");
}
// 环绕通知
@Around("execution(* com.service.UserService.add(..))")
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println("环绕通知开始");
proceedingJoinPoint.proceed();
System.out.println("环绕通知结束");
}
}
复制代码
Xml实现aop编程:
1) 引入jar文件 【aop 相关jar, 4个】
2) 引入aop名称空间
3)aop 配置
* 配置切面类 (重复执行代码造成的类)
* aop配置
拦截哪些方法 / 拦截到方法后应用通知代码
复制代码
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:context="http://www.springframework.org/schema/context" 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/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- dao 实例 -->
<bean id="userService" class="com.service.UserService"></bean>
<!-- 切面类 -->
<bean id="aop" class="com.aop2.AopLog2"></bean>
<!-- Aop配置 -->
<aop:config>
<!-- 定义一个切入点表达式: 拦截哪些方法 -->
<aop:pointcut expression="execution(* com.service.UserService.*(..))" id="pt" />
<!-- 切面 -->
<aop:aspect ref="aop">
<!-- 环绕通知 -->
<aop:around method="around" pointcut-ref="pt" />
<!-- 前置通知: 在目标方法调用前执行 -->
<aop:before method="begin" pointcut-ref="pt" />
<!-- 后置通知: -->
<aop:after method="after" pointcut-ref="pt" />
<!-- 返回后通知 -->
<aop:after-returning method="afterReturning" pointcut-ref="pt" />
<!-- 异常通知 -->
<aop:after-throwing method="afterThrowing" pointcut-ref="pt" />
</aop:aspect>
</aop:config>
</beans>
复制代码
public class AopLog2 {
// 前置通知
public void begin() {
System.out.println("前置通知");
}
//
// 后置通知
public void commit() {
System.out.println("后置通知");
}
// 运行通知
public void returning() {
System.out.println("运行通知");
}
// 异常通知
public void afterThrowing() {
System.out.println("异常通知");
}
// 环绕通知
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println("环绕通知开始");
proceedingJoinPoint.proceed();
System.out.println("环绕通知结束");
}
}
复制代码