热烈欢迎我们高度关注我的社会公众号【李兰聊构架】,Java后端非主流控制技术栈的基本原理、源代码预测、构架和各式各样网络高mammalian、高效能、高需用的软件控制系统。
Spring 是一类轻量合作开发构架,意在提升合作开发者的合作开发工作效率和控制系统的可移植性。Spring 官方网站:http://spring.io/。
他们通常说 Spring 构架指的都是 Spring Framework,它是许多组件的子集,采用那些组件能很方便快捷地帮助他们展开合作开发。那些组件是:核心理念罐子、统计数据出访/软件控制系统,、Web、AOP(面向全国锐角程式设计)、辅助工具、最新消息和试验组件。比如说:Core Container 中的 Core 组件是Spring 大部份组件的核心理念,Beans 组件和 Context 组件是与此同时实现IOC和倚赖转化成的此基础,AOP组件用以与此同时实现面向全国锐角程式设计。
Spring 官方网站列举的 Spring 的 6 个特点:
核心理念控制技术 :倚赖转化成(DI),AOP,该事件(events),天然资源,i18n,校正,统计数据存取,隐式,SpEL。
试验 :演示第一类,TestContext构架,Spring MVC 试验,WebTestClient。
统计数据出访 :外交事务,DAO全力支持,JDBC,ORM,番台XML。
Web全力支持 : Spring MVC和Spring WebFlux Web构架。
软件控制系统 :远距处置,JMS,JCA,JMX,邮件,任务,调度,内存。
词汇 :Kotlin,Groovy,静态词汇。
右图相关联的是 Spring4.x 版。现阶段新一代的5.x版中 Web 组件的 Portlet 组件早已被弃置掉,与此同时减少了用作触发器积极响应式处置的 WebFlux 组件。
Spring Core: 此基础,能说 Spring 其他大部份的功能都需要倚赖于该类库。主要提供 IOC 倚赖转化成功能。
Spring Aspects : 该组件为与AspectJ的软件控制系统提供全力支持。
Spring AOP :提供了面向全国方面的程式设计与此同时实现。
Spring JDBC : Java统计数据库连接。
Spring JMS :Java最新消息服务。
Spring ORM : 用作全力支持Hibernate等ORM辅助工具。
Spring Web : 为创建Web应用程序提供全力支持。
Spring Test : 提供了对 JUnit 和 TestNG 试验的全力支持。
IoC
IoC(Inverse of Control:控制反转)是一类设计思想,就是 将原本在程序中手动创建第一类的控制权,交由Spring构架来管理。 IoC 在其他词汇中也有应用,并非 Spirng 特有。 IoC 罐子是 Spring 用以与此同时实现 IoC 的载体, IoC 罐子实际上就是个Map(key,value),Map 中存放的是各式各样第一类。
将第一类之间的相互倚赖关系交给 IOC 罐子来管理,并由 IOC 罐子完成第一类的转化成。这样能很大程度上简化应用的合作开发,把应用从复杂的倚赖关系中解放出来。 IOC 罐子就像是一个工厂一样,当他们需要创建一个第一类的时候,只需要配置好配置文件/注解即可,完全不用考虑第一类是如何被创建出来的。 在实际项目中一个 Service 类可能有几百甚至上千个类作为它的底层,假如他们需要实例化这个 Service,你可能要每次都要搞清这个 Service 大部份底层类的构造函数,这可能会把人逼疯。如果利用 IOC 的话,你只需要配置好,然后在需要的地方引用就行了,这大大减少了项目的可移植性且降低了合作开发难度。
Spring 时代他们通常通过 XML 文件来配置 Bean,后来合作开发者觉得 XML 文件来配置不太好,于是 SpringBoot 注解配置就慢慢开始流行起来。
推荐阅读:http://www.zhihu.com/question/23277575/answer/169698662
Spring IOC的初始化过程:
IOC源代码阅读
http://javadoop.com/post/spring-ioc
AOP
AOP(Aspect-Oriented Programming:面向全国锐角程式设计)能够将那些与业务无关,却为业务组件所共同调用的逻辑或责任(例如外交事务处置、日志管理、权限控制等)封装起来,便于减少控制系统的重复代码,降低组件间的耦合度,并有利于未来的可拓展性和可移植性。
Spring AOP就是基于静态代理的,如果要代理的第一类,与此同时实现了某个接口,那么Spring AOP会采用JDK Proxy,去创建代理第一类,而对于没有与此同时实现接口的第一类,就无法采用 JDK Proxy 去展开代理了,这时候Spring AOP会采用Cglib ,这时候Spring AOP会采用 Cglib 生成一个被代理第一类的子类来作为代理,如右图所示:
当然你也能采用 AspectJ ,Spring AOP 早已软件控制系统了AspectJ ,AspectJ 应该算的上是 Java 生态控制系统中最完整的 AOP 构架了。
采用 AOP 之后他们能把一些通用功能抽象出来,在需要用到的地方直接采用即可,这样大大简化了代码量。他们需要减少新功能时也方便快捷,这样也提升了控制系统扩展性。日志功能、外交事务管理等等场景都用到了 AOP 。
Spring AOP 属于运行时增强,而 AspectJ 是编译时增强。 Spring AOP 基于代理(Proxying),而 AspectJ 基于字节码操作(Bytecode Manipulation)。
Spring AOP 早已软件控制系统了 AspectJ ,AspectJ 应该算的上是 Java 生态控制系统中最完整的 AOP 构架了。AspectJ 相比于 Spring AOP 功能更加强大,但是 Spring AOP 相对来说更简单,
如果他们的锐角比较少,那么两者性能差异不大。但是,当锐角太多的话,最好选择 AspectJ ,它比Spring AOP 快许多。
singleton : 唯一 bean 实例,Spring 中的 bean 默认都是单例的。
prototype : 每次请求都会创建一个新的 bean 实例。
request : 每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTP request内有效。
session : 每一次HTTP请求都会产生一个新的 bean,该bean仅在当前 HTTP session 内有效。
global-session: 全局session作用域,仅仅在基于portlet的web应用中才有意义,Spring5早已没有了。Portlet是能够生成语义代码(例如:HTML)片段的小型Java Web插件。它们基于portlet罐子,能像servlet一样处置HTTP请求。但是,与 servlet 不同,每个 portlet 都有不同的会话。
大部分时候他们并没有在控制系统中采用多线程,所以很少有人会高度关注这个问题。单例 bean 存在线程问题,主要是因为当多个线程操作同一个第一类的时候,对这个第一类的非静态成员变量的写操作会存在线程安全问题。
常用的有两种解决办法:
1、在Bean第一类中尽量避免定义可变的成员变量(不太现实)。
2、在类中定义一个ThreadLocal成员变量,将需要的可变成员变量保存在 ThreadLocal 中(推荐的一类方式)。
这部分网上有许多文章都讲到了,下面的内容整理自:http://yemengying.com/2016/07/14/spring-bean-life-cycle/ ,除了这篇文章,再推荐一篇很不错的文章 :http://www.cnblogs.com/zrtqsk/p/3735273.html 。
Bean 罐子找到配置文件中 Spring Bean 的定义。
Bean 罐子利用 Java Reflection API 创建一个Bean的实例。
如果涉及到一些属性值 利用 set()方法设置一些属性值。
如果 Bean 与此同时实现了 BeanNameAware 接口,调用 setBeanName()方法,传入Bean的名字。
如果 Bean 与此同时实现了 BeanClassLoaderAware 接口,调用 setBeanClassLoader()方法,传入 ClassLoader第一类的实例。
如果Bean与此同时实现了 BeanFactoryAware 接口,调用 setBeanClassLoader()方法,传入 ClassLoade r第一类的实例。
与上面的类似,如果与此同时实现了其他 *.Aware接口,就调用相应的方法。
如果有和加载这个 Bean 的 Spring 罐子相关的 BeanPostProcessor 第一类,执行postProcessBeforeInitialization() 方法
如果Bean与此同时实现了InitializingBean接口,执行afterPropertiesSet()方法。
如果 Bean 在配置文件中的定义包含 init-method 属性,执行指定的方法。
如果有和加载这个 Bean的 Spring 罐子相关的 BeanPostProcessor 第一类,执行postProcessAfterInitialization() 方法
当要销毁 Bean 的时候,如果 Bean 与此同时实现了 DisposableBean 接口,执行 destroy() 方法。
当要销毁 Bean 的时候,如果 Bean 在配置文件中的定义包含 destroy-method 属性,执行指定的方法。
图示:
Spring Bean 生命周期
与之比较类似的中文版:
谈到这个问题,他们不得不提提之前 Model1 和 Model2 这两个没有 Spring MVC 的时代。
Model1 时代 : 许多学 Java 后端比较晚的朋友可能并没有接触过 Model1 模式下的 JavaWeb 应用合作开发。在 Model1 模式下,整个 Web 应用几乎全部用 JSP 页面组成,只用少量的 JavaBean 来处置统计数据库连接、出访等操作。这个模式下 JSP 即是控制层又是表现层。显而易见,这种模式存在许多问题。比如说①将控制逻辑和表现逻辑混杂在一起,导致代码重用率极低;②前端和后端相互倚赖,难以展开试验并且合作开发工作效率极低;
Model2 时代 :学过 Servlet 并做过相关 Demo 的朋友应该了解Java Bean(Model)+ JSP(View,)+Servlet(Controller) 这种合作开发模式,这就是早期的 JavaWeb MVC 合作开发模式。Model:控制系统涉及的统计数据,也就是 dao 和 bean。View:展示模型中的统计数据,只是用以展示。Controller:处置用户请求都发送给 ,返回统计数据给 JSP 并展示给用户。
Model2 模式下还存在许多问题,Model2的抽象和封装程度还远远不够,采用Model2展开合作开发时不可避免地会重复造轮子,这就大大降低了程序的可移植性和复用性。于是许多JavaWeb合作开发相关的 MVC 构架营运而生比如说Struts2,但是 Struts2 比较笨重。随着 Spring 轻量合作开发构架的流行,Spring 生态圈出现了 Spring MVC 构架, Spring MVC 是当前最优秀的 MVC 构架。相比于 Struts2 , Spring MVC 采用更加简单和方便快捷,合作开发工作效率更高,并且 Spring MVC 运行速度更快。
MVC 是一类设计模式,Spring MVC 是一款很优秀的 MVC 构架。Spring MVC 能帮助他们展开更简洁的Web层的合作开发,并且它天生与 Spring 构架软件控制系统。Spring MVC 下他们通常把后端项目分为 Service层(处置业务)、Dao层(统计数据库操作)、Entity层(实体类)、Controller层(控制层,返回统计数据给前台页面)。
Spring MVC 的简单基本原理图如下:
基本原理如右图所示:
上图的一个笔误的小问题:Spring MVC 的入口函数也就是前端控制器 DispatcherServlet 的作用是接收请求,积极响应结果。
流程说明(重要):
1、客户端(浏览器)发送请求,直接请求到 DispatcherServlet。
2、DispatcherServlet 根据请求信息调用 HandlerMapping,解析请求相关联的 Handler。
3、解析到相关联的 Handler(也就是他们平常说的 Controller 控制器)后,开始由 HandlerAdapter 适配器处置。
4、HandlerAdapter 会根据 Handler来调用真正的处置器开处置请求,并处置相应的业务逻辑。
处置器处置完业务后,会返回一个 ModelAndView 第一类,Model 是返回的统计数据第一类,5、View 是个逻辑上的 View。
6、ViewResolver 会根据逻辑 View 查找实际的 View。
7、DispaterServlet 把返回的 Model 传给 View(视图渲染)。
8、把 View 返回给请求者(浏览器)
工厂设计模式 : Spring采用工厂模式通过 BeanFactory、ApplicationContext 创建 bean 第一类。
代理设计模式 : Spring AOP 功能的与此同时实现。
单例设计模式 : Spring 中的 Bean 默认都是单例的。
模板方法模式 : Spring 中 jdbcTemplate、hibernateTemplate 等以 Template 结尾的对统计数据库操作的类,它们就采用到了模板模式。
包装器设计模式 : 他们的项目需要连接多个统计数据库,而且不同的客户在每次出访中根据需要会去出访不同的统计数据库。这种模式让他们能根据客户的需求能够静态切换不同的统计数据源。
观察者模式: Spring 该事件驱动模型就是观察者模式很经典之作的一个应用。
适配器模式 :Spring AOP 的增强或通知(Advice)采用到了适配器模式、spring MVC 中也是用到了适配器模式适配Controller。
…
1、作用第一类不同: @Component 注解作用作类,而@Bean注解作用作方法。
2、@Component通常是通过类路径扫描来自动侦测和自动装配到Spring罐子中(他们能采用 @ComponentScan 注解定义要扫描的路径从中找出标识了需要装配的类自动装配到 Spring 的 bean 罐子中)。@Bean 注解通常是他们在标有该注解的方法中定义产生这个 bean,@Bean告诉了Spring这是某个类的示例,当我需要用它的时候还给我。
3、@Bean 注解比 Component 注解的自定义性更强,而且许多地方他们只能通过 @Bean 注解来注册bean。比如说当他们引用第三方库中的类需要装配到 Spring罐子时,则只能通过 @Bean来与此同时实现。
@Bean注解采用示例:
上面的代码相当于下面的 xml 配置
下面这个例子是通过 @Component 无法与此同时实现的。
他们通常采用 @Autowired 注解自动装配 bean,要想把类标识成需用作 @Autowired注解自动装配的 bean 的类,采用以下注解可与此同时实现:
@Component :通用的注解,可标注任意类为 Spring 组件。如果一个Bean不知道属于拿个层,能采用@Component 注解标注。
@Repository : 相关联持久层即 Dao 层,主要用作统计数据库相关操作。
@Service : 相关联服务层,主要涉及一些复杂的逻辑,需要用到 Dao层。
@Controller : 相关联 Spring MVC 控制层,主要用户接受用户请求并调用 Service 层返回统计数据给前端页面。
1、程式设计式外交事务,在代码中硬编码。(不推荐采用)
2、声明式外交事务,在配置文件中配置(推荐采用)
声明式外交事务又分为两种:
1、基于XML的声明式外交事务
2、基于注解的声明式外交事务
TransactionDefinition 接口中定义了五个表示隔离级别的常量:
TransactionDefinition.ISOLATION_DEFAULT: 采用后端统计数据库默认的隔离级别,Mysql 默认采用的 REPEATABLE_READ隔离级别 Oracle 默认采用的 READ_COMMITTED隔离级别.
TransactionDefinition.ISOLATION_READ_UNCOMMITTED: 最低的隔离级别,允许读取尚未提交的统计数据变更,可能会导致脏读、幻读或不可重复读
TransactionDefinition.ISOLATION_READ_COMMITTED: 允许读取mammalian外交事务早已提交的统计数据,能阻止脏读,但是幻读或不可重复读仍有可能发生
TransactionDefinition.ISOLATION_REPEATABLE_READ: 对同一字段的多次读取结果都是一致的,除非统计数据是被本身外交事务自己所修改,能阻止脏读和不可重复读,但幻读仍有可能发生。
TransactionDefinition.ISOLATION_SERIALIZABLE: 最高的隔离级别,完全服从ACID的隔离级别。大部份的外交事务依次逐个执行,这样外交事务之间就完全不可能产生干扰,也就是说,该级别能防止脏读、不可重复读和幻读。但是这将严重影响程序的性能。通常情况下也不会用到该级别。
全力支持当前外交事务的情况:
TransactionDefinition.PROPAGATION_REQUIRED: 如果当前存在外交事务,则加入该外交事务;如果当前没有外交事务,则创建一个新的外交事务。
TransactionDefinition.PROPAGATION_SUPPORTS: 如果当前存在外交事务,则加入该外交事务;如果当前没有外交事务,则以非外交事务的方式继续运行。
TransactionDefinition.PROPAGATION_MANDATORY: 如果当前存在外交事务,则加入该外交事务;如果当前没有外交事务,则抛出异常。(mandatory:强制性)
不全力支持当前外交事务的情况:
TransactionDefinition.PROPAGATION_REQUIRES_NEW: 创建一个新的外交事务,如果当前存在外交事务,则把当前外交事务挂起。
TransactionDefinition.PROPAGATION_NOT_SUPPORTED: 以非外交事务方式运行,如果当前存在外交事务,则把当前外交事务挂起。
TransactionDefinition.PROPAGATION_NEVER: 以非外交事务方式运行,如果当前存在外交事务,则抛出异常。
其他情况:
TransactionDefinition.PROPAGATION_NESTED: 如果当前存在外交事务,则创建一个外交事务作为当前外交事务的嵌套外交事务来运行;如果当前没有外交事务,则该取值等价于TransactionDefinition.PROPAGATION_REQUIRED。
发表评论