编者按: 这篇内容将帮助你总体认识有关Spring框架相关的核心知识。我尽量在这里涵盖Spring框架的一些非常重要的概念。希望这可帮助你更好的总体把握Spring框架,促使你积极探索Spring的有趣的技术栈生态世界。 Spring框架 1、Spring是什么? 简而言之,Spring是企业级Java的开源开发框架。Spring框架的核心功能可用于开发任何java应用程序。Spring框架的核心模块如下:SpringContext(Spring上下文):用于依赖注入(DI);SpringDAO(数据库访问对象):使用DAP模式进行数据库操作;SpringJDBC:用于JDBC和数据源的支持;SpringORM:用于ORM工具支持,如Hibernate;SpringAOP:用于面向方面的编程;SpringWebmodule:用于创建Web应用。 2、Spring框架的优点是什么轻量级:Spring时轻量级应用框架。Spring框架的基本版本在2MB左右;依赖注入控制反转(DIIoC):这有助于通过连接独立的组件对象来实现松散耦合;Spring容器:包含和管理应用程序对象的生命周期;事务管理:Spring支持事务管理,即JDBC操作、文件上传、异常处理等,通过spring注解或bean配置来完成;SpringMVC:SpringMVC可用于创建Web应用程序以及能够返回XML或JSON响应的restfulWeb服务;异常处理:Spring提供了一个方便的API来将特定的技术异常(由JDBC、Hibernate等抛出)转换为一致的未经检查的异常;面向切面编程(AOP):AOP将程序逻辑分解为不同的部分(称为关注点)。它用于通过横切关注点来增加模块化。横切关注点是可以影响整个应用程序的关注点,应尽可能集中在代码中的一个位置,例如事务管理、身份验证、日志记录、安全性等。 3、什么是SpringBean? 任何由SpringIoC容器初始化的普通Java类都称为SpringBean。我们使用spring应用程序上下文来获取SpringBean实例。SpringIoCContainer管理SpringBean范围作用域的生命周期并在bean中注入任何所需的依赖项。 Springbean的不同作用域: 当我们声明时,我们可以指定bean的作用域来通知IoC容器关于bean的创建以及它将存活多长时间。 对于任何Java应用程序,都有两个不同的作用域,称为单例(Singleton)和原型(Prototype) 主要有三种不同的作用域(或范围),即请求(request)、会话(session)和全局会话(globalsession),专门针对基于Spring的JavaWeb应用程序。 Singleton是任何bean的默认作用域。这意味着每个IoC容器将创建单个bean实例。因此,单例bean不是线程安全的。在原型范围内,每次请求bean时都会创建一个新实例。在请求范围内,单个bean被定义到一个HTTP请求内。此范围仅在支持Web的springApplicationContext中有效。在会话范围内,单个bean被定义到一个HTTP会话内。此范围仅在支持Web的springApplicationContext中有效。在globalsession范围内,bean被定义为全局HTTP会话内有效。此范围仅在支持Web的springApplicationContext中有效。 要设置springbean的范围,我们可以在标签中使用scope属性。scope用于基于注释的DI。 4、SpringIoC容器 Spring容器是Spring框架的核心。容器将创建对象,把它们连接在一起,配置它们,并管理它们从创建到销毁的完整生命周期。Spring容器使用依赖注入(DI)来管理组成应用程序的组件。 有两种不同类型的容器: BeanFactory容器:这是Spring容器的核心。org。springframework。beans。factory。BeanFactory是一个接口,充当IoC容器,它实例化、配置和管理许多bean。应用示例如下:Beans。xml:lt;?xmlversion1。0encodingUTF8?beansxmlnshttp:www。springframework。orgschemabeansxmlns:xsihttp:www。w3。org2001XMLSchemainstancexsi:schemaLocationhttp:www。springframework。orgschemabeanshttp:www。springframework。orgschemabeansspringbeans3。0。xsdbeanidhelloWorldclasscom。newday。bean。HelloWorldpropertynamemessagevalueHelloWorld!beanbeanspackagecom。newday。bean;publicclassHelloWorld{privateStringmessage;publicStringgetMessage(){returnmessage;}publicvoidsetMessage(Stringmessage){this。messagemessage;}}MainApp。java:importorg。springframework。beans。factory。BeanFactory;importorg。springframework。beans。factory。xml。XmlBeanFactory;importorg。springframework。core。io。ClassPathResource;importorg。springframework。core。io。Resource;importcom。newday。bean。HelloWorld;publicclassMainApp{publicstaticvoidmain(Stringargs〔〕){ResourceresourcenewClassPathResource(beans。xml);BeanFactorybeanFactorynewXmlBeanFactory(resource);HelloWorldhelloWorld(HelloWorld)beanFactory。getBean(helloWorld);System。out。println(helloWorld。getMessage());}} ApplicationContext容器:org。springframework。context。ApplicationContext接口也充当IoC容器,但ApplicationContext接口建立在BeanFactory接口之上,以提供一些BeanFactory额外的功能,例如与Spring的AOP的简单集成、消息资源处理(对于I18N)、事件传播、Web应用程序的应用层特定上下文(例如WebApplicationContext)。所以使用ApplicationContext比使用BeanFactory更好些。示例代码如下:lt;?xmlversion1。0encodingUTF8?beansxmlnshttp:www。springframework。orgschemabeansxmlns:xsihttp:www。w3。org2001XMLSchemainstancexsi:schemaLocationhttp:www。springframework。orgschemabeanshttp:www。springframework。orgschemabeansspringbeans3。0。xsdbeanidhelloWorldclasscom。newday。bean。HelloWorldpropertynamemessagevalueHelloWorld!beanbeanspackagecom。newday。bean;publicclassHelloWorld{privateStringmessage;publicStringgetMessage(){returnmessage;}publicvoidsetMessage(Stringmessage){this。messagemessage;}}packagecom。newday。applicationcontext;importorg。springframework。context。ApplicationContext;importorg。springframework。context。support。ClassPathXmlApplicationContext;importcom。newday。bean。HelloWorld;publicclassMainApp{publicstaticvoidmain(Stringargs〔〕){ApplicationContextappContextnewClassPathXmlApplicationContext(beans。xml);HelloWorldhelloWorld(HelloWorld)appContext。getBean(helloWorld);System。out。println(helloWorld。getMessage());}}5、Autowired 对于基于注解的依赖注入,使用Autowired注解。标有ComponentServiceRepository等的类可以注入到标有Autowired的属性中 Autowired应用于:字段域:用于基于字段的依赖注入;setter:用于setter依赖注入。与基于字段的依赖注入相同;构造器:基于构造函数的依赖注入。 1)基于构造器和setter的区别依赖项的注入可以是可选的,也可以是强制性的。对于强制注入,我们使用基于构造函数的DI。对于可选的依赖项,我们可以使用基于setter的DI。但是,我们可以使用Required注释标记基于setter的DI。在循环依赖的情况下,基于构造器的DI将无法注入,但基于setter的DI将能够注入如果要注入更多数量的参数,则建议使用基于构造函数的DI。 2)context:annotationconfig和context:componentscan的区别context:annotationconfig用于激活已在应用程序上下文中注册的bean中的注释;context:componentscan也可以执行context:annotationconfig所做的工作,但也可以扫描包以在应用程序上下文中查找和注册bean。 3)Component、Controller、RepositoryService注解的区别 如果一个类用ComponentControllerServiceRepository注解标记,那么SpringDI容器可以在组件扫描机制期间识别该类。但是,对于服务层类使用Service是个好主意,并且Controller应该在springmvcweb控制器中使用。Repository用于将DAO导入DI容器。此外,任何未经检查的异常都将被转换为SpringDataAccessException。 4)ViewResolver与MultipartResolver ViewResolver用于按名称解析视图。该接口由InternalResourceViewResolver实现; MultipartResolver用于处理web应用程序中的文件上传。 5)SpringMVC中的验证 org。springframework。validation。Validator接口支持springMVC验证。验证表单的一些实用方法是ValidationUtils类中的rejectIfEmptyOrWhitespace()和rejectIfEmpty()。示例如下:ComponentpublicclassEmployeeValidatorimplementsValidator{publicbooleansupports(Classclazz){returnEmployeeVO。class。isAssignableFrom(clazz);}publicvoidvalidate(Objecttarget,Errorserrors){ValidationUtils。rejectIfEmptyOrWhitespace(errors,firstName,error。firstName,Firstnameisrequired。);ValidationUtils。rejectIfEmptyOrWhitespace(errors,lastName,error。lastName,Lastnameisrequired。);ValidationUtils。rejectIfEmptyOrWhitespace(errors,email,error。email,Emailisrequired。);}} SpringMVC中验证表单的另一种方法是:对模型bean的属性使用Hibernate验证(例如NotNull、Size等)在控制器的方法签名中使用Valid、BindingResult。BindingResult。hasErrors()方法来验证模型bean。6、SpringMVC拦截器 HandlerInterceptor接口充当springMVC拦截器。它在服务请求之前和之后拦截。如果您实现了HandlerInterceptor接口,则preHandle()、postHandle()和afterCompletion()是要覆盖的方法。但是,为了避免覆盖,您可以使用HandlerInterceptorAdapter类。7、SpringMVC框架中的异常处理 Spring中的异常是使用SimpleMappingExceptionResolver类以声明方式处理的。您要处理的异常列表是针对exceptionMapping属性配置的。如果有异常发生在流程中的任何地方,都将被捕获并显示相应的视图。对于所有其他异常,您可以针对defaultErrorView属性捕获它们。示例配置如下:beanclassorg。springframework。web。servlet。handler。SimpleMappingExceptionResolverpropertynameexceptionMappingspropspropkeycom。newday。exception。XYZExceptionerrorxyzExceptionViewproppropspropertypropertynamedefaultErrorViewvalueerrorgenericExceptionViewbean 8、如何在SpringBean中获取ServletContext和ServletConfig对象? 实现ServletContextAware和ServletConfigAware接口并覆盖以下方法:ControllerRequestMapping(valuemagic)publicclassSimpleControllerimplementsServletContextAware,ServletConfigAware{privateServletContextcontext;privateServletConfigconfig;OverridepublicvoidsetServletConfig(finalServletConfigservletConfig){this。configservletConfig;}OverridepublicvoidsetServletContext(finalServletContextservletContext){this。contextservletContext;}othercode}9、Spring中的事务管理 数据库事务是一组被视为关联工作单元的操作。事务的主要原则是提交所有操作或在失败的情况下回滚所有操作。在交易中提交数据时,我们需要确保交易协议称为ACID(原子性一致性隔离持久性)的属性:原子性(Atomicty):事务中的所有操作都被视为一个工作单元。因此,事务将被提交或回滚。一致性(Consistency):这通过维护数据库的引用完整性来确保事务将数据库从一个有效状态带到另一个有效状态。隔离性(Isolation):每个事务都应该与其他事务隔离,即使它们处理的是同一组数据。持久性(Durability):这确保一旦事务完成,数据将永久存在于数据库中。即使断电或系统崩溃,数据也不会丢失。 Spring支持编程式和声明式两种事务管理。它还支持全局和本地事务。 全局事务vs本地事务:本地事务处理集中式系统中的单个数据源(例如JDBC);全局事务处理分布式系统中的多个数据源(例如JDBC、JMS)。一个全局事务是多个本地事务的集合; 编程式事务与声明式事务: Spring支持两种不同类型的事务管理,称为编程式(Programatic)和声明式(Declarative)事务管理:编程式事务:这意味着,您需要在代码中管理事务。很难维护;声明式事务:这意味着要将事务管理代码与业务逻辑分开。因此需要使用XML或基于注释的配置来管理事务。SpringAOP模块支持声明式事务管理。 脏读、幻读和不可重复读:脏读:当一个事务正在更改记录元组并且第二个事务在原始更改被提交或回滚之前尝试读取此元组记录时,就会发生脏读。这被称为脏读场景,因为第一个事务总是有可能回滚更改,导致第二个事务读取无效值。幻读:幻读发生在同一查询在事务中执行两次,并且第二个结果集包含在第一个结果集中不可见的行。这种情况是由另一个事务在两个查询的执行之间插入新行引起的不可重复读:当在同一个事务中相同的查询产生不同的结果时,就会发生不可重复读取。这因为当另一个事务更新其他事务返回的数据时,就会发生这种情况。 隔离与传播:隔离度(Isolation):此事务与其他事务的工作隔离的程度。例如,这个事务可以看到来自其他事务的未提交的写入吗?传播(Propagation):在传播的情况下,代码将始终在事务范围内运行。创建一个新事务或重用一个(如果可用)。10、HibernateDAOSupport和HibernateTemplate 在旧版本的spring和hibernate集成中,需要HibernateDAOSupport和HibernateTemplate。但是,较新版本的Spring不建议使用这两个类(这里仅做了解)。 通常我们从HibernateDAOSupport扩展我们的DAO类,并且getHibernateTemplate()方法将可用于Hibernate会话中的CRUD操作。由于这不是推荐的方法,所以我们在DAO中注入会话工厂(SessionFactory)。下面的代码片段会给你一些关于HibernateDAOSupport和HibernateTemplate的想法:publicclassEmployeeHibernateDaoextendsHibernateDaoSupportimplementsEmployeeDao{Transactional(readOnlyfalse)publicvoidsaveEmployee(Employeeemp){System。out。println(Createnewemployeeemp);getHibernateTemplate()。save(emp);System。out。println(Employeecreatedemp);}}11、SpringDAO和SpringORM的区别 DAO是一种设计模式,以最大限度地减少应用程序和后端之间的耦合; ORM处理如何将对象映射到对象关系数据库中,从而减少数据库和应用程序之间的耦合。 如果您在没有DAO的情况下使用ORM,那么您的应用程序将变得依赖于ORM,因此很难从一个ORM(例如Hibernate)移动到另一个ORM(例如NoSQL)。 SpringDAO是使用Repository注解实现的。Spring存储库扩展JPARepository并传递JPA实体及其主键。RepositorypublicinterfacePersonRepositoryextendsJPARepositoryPerson,PersonPK{publicListfindByFirstName(StringfirstName);Query(YourSQLquery)publicListfindByAddress(StringfirstName);} 最后,关于Spring框架相关的概念就简要介绍到这里,希望这能给你进入并深入Spring技术栈一个简单入口,而不会被Spring技术生态所惊吓(Spring现在都成软件开发技术的全家桶了,啥都有)日进一步,锲而不舍,终将大成!