介绍
Spring是一个分层的(一站式) 轻量级开源框架
Spring的核心是控制反转(IoC)和面向切面(AOP)
为什么说分层一站式呢?
javaEE分三层开发 WEB层,业务层,持久层。在ssh整合框架中s == Struts2, s == spring,h == Hibernate ,spring 的一站式开发就是不用struts2 和hibernate,在spring中有SpringMvc可以替代Struts2,springJDBC可以替代Hibernate。等于一个spring框架可以快速开发JavaEE应用。关于轻量级就不太多说了,spring整个框架打包出来也才1M多的内存大小。spring运行中的消耗也不大。那肯定是轻量级的框架。
什么是IOC的功能?
概念
IoC – Inverse of Control,控制反转,将对象的创建权反转给Spring!!
使用IOC可以解决的程序耦合性高的问题!!
控制反转
假设我需要做一个功能,在这个功能当中我需要调用servic层,然后再调用dao层,去取数据。在传统的javaEE开发中我就直接去new一个service 然后再new一个dao。但是在spring框架中,我们吧new service和new dao的权利交个spring框架,假设我需要使用我就直接去spring框架中寻找。等于说我的资源创建的权利交给了spring框架,这就叫做控制反转。
解耦
刚刚我们说资源创建交给了sring,我们需要什么就找spring。这过程就像是工厂模式。但是在spring框架中它需要创建哪些对象,它需要一个配置文件。这个配置文件告诉spring,需要创建哪些资源。
例如:假设我需要去数据库查询数据显示页面
程序启动,spring框架去找配置文件创建资源,把资源放置再一个容器中,开始运行,前端请求数据,在spring中找controller层,再找service层,再找dao层要数据,最后数据原路返回controller,再显示到页面上。其中service被spring注入到controlller层,dao层被spring注入到service层。这个过程分工明确。每一层各司其职。传统的一个开发,在servlet中直接new然后去查数据,然后数据返回到界面上。万一操作一多所有的判断,查询不同的表,这个servlet的代码变得十分的臃肿。不说开发慢,你开发完了看代码也费劲。 所以说控制反转可以用来解耦
什么是面向切面(AOP)?
概念
- 在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程
- AOP是一种编程范式,隶属于软工范畴,指导开发者如何组织程序结构
- AOP最早由AOP联盟的组织提出的,制定了一套规范.Spring将AOP思想引入到框架中,必须遵守AOP联盟的规范
- 通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术
- AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型
- 利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率
其实AOP可以在不修改源代码的前提下,对程序进行增强!!
Spring框架的AOP的底层实现
Srping框架的AOP技术底层也是采用的代理技术,代理的方式提供了两种
- 基于JDK的动态代理
必须是面向接口的,只有实现了具体接口的类才能生成代理对象
- 基于CGLIB动态代理
对于没有实现了接口的类,也可以产生代理,产生这个类的子类的方式
- Spring的传统AOP中根据类是否实现接口,来采用不同的代理方式
- 如果实现类接口,使用JDK动态代理完成AOP
- 如果没有实现接口,采用CGLIB动态代理完成AOP
JDK的动态代理
注意:需要实现类接口
例子:假设我有两个工作,工作1,工作2.
//写一个接口
public interface Working {
void wokingOne();
void WorkingTwo();
}
//接口实现类
public class WorkingImpl implements Working {
@Override
public void wokingOne() {
System.out.println("做任务1");
}
@Override
public void WorkingTwo() {
System.out.println("做任务2");
}
}
好的,现在我要先做任务1,然后再做任务2我们的写法为:
public static void main(String[] args) {
Working working = new WorkingImpl();
working.wokingOne();//做任务1
working.WorkingTwo();//做任务2
}
好的精彩的地方来了,我再做任务2之前我要先休息10分钟,但是不能修改源代码。怎么班呢?这时候就用到我们的JDK动态代理了。代码如下:
先写一个代理的工具类。再做任务2前我们休息十分钟
public class MyProxyUtils {
public static Working getProxy(final Working working) {
// 使用Proxy类生成代理对象
Working proxy = (Working) Proxy.newProxyInstance(working.getClass().getClassLoader(),
working.getClass().getInterfaces(), new InvocationHandler() {
// 代理对象方法一直线,invoke方法就会执行一次
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//再做工作2之前我先休息10分钟
if ("WorkingTwo".equals(method.getName())) {
System.out.println("休息10分钟");
}
//工作继续进行下去
return method.invoke(working, args);
}
});
// 返回代理对象
return proxy;
}
}
public static void main(String[] args) {
Working working = new WorkingImpl();
Working proxy = MyProxyUtils.getProxy(working);
proxy.wokingOne();
proxy.WorkingTwo();
}
运行的结果可想而知:
做任务1
休息10分钟
做任务2
CGLIB动态代理
注意:没有实现类接口
CGLIB也是一个java项目,所以要使用它就要引入CGLIB的开发的jar包,因为在Spring框架核心包(core)中已经引入了CGLIB的开发包了。所以直接引入Spring核心开发包即可
好我们同样使用上面的例子来做事例吧
//工作类
public class Working {
public void wokingOne() {
System.out.println("做任务1");
}
public void WorkingTwo() {
System.out.println("做任务2");
}
}
new一个对象调用任务1 任务2得到的结果这个就没必要说了。我们重点看下怎么使用CGLIB来在不改变源码的情况下,做任务之前休息十分钟。
public static Working getProxy(){
// 创建CGLIB核心的类
Enhancer enhancer = new Enhancer();
// 设置父类
enhancer.setSuperclass(Working.class);
// 设置回调函数
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy methodProxy) throws Throwable {
if("WorkingTwo".equals(method.getName())){
// 休息10分钟
System.out.println("休息10分钟...");
}
return methodProxy.invokeSuper(obj, args);
}
});
// 生成代理对象
Working proxy = (Working) enhancer.create();
return proxy;
}
public static void main(String[] args) {
Working working = new WorkingImpl();
Working proxy = MyCglibUtils.getProxy(working);
proxy.wokingOne();
proxy.WorkingTwo();
}
最后的运行结果也是相同的:
做任务1
休息10分钟
做任务2
分析
我上面简单分析了一下spring AOP的底层实现的两种方式:JDK动态代理,CGLIB。在Spring框架中它会自动选择使用哪一种方式。如果有接口实现类,那就使用jdk动态代理,没有接口就使用CGLIB。有了这个功能那么我们修改东西就方便多了。假设我再做任务一前我需要记录下日志。我就直接写一个切面类,直接去记录日志。都不用修改本身的源码。
结尾
作者刚刚学习spring中,记录一些知识点,梳理了一下自己的理解,如有错误希望各位大神提出。