Spring学习(1)——快速入门
本文内容纲要:
-认识Spring框架
-SpringIoC和DI简介
-SpringAOP简介
认识Spring框架
Spring框架是Java应用最广的框架,它的成功来源于理念,而不是技术本身,它的理念包括IoC(InversionofControl,控制反转)和AOP(AspectOrientedProgramming,面向切面编程)。
什么是Spring:
- Spring是一个轻量级的DI/IoC和AOP容器的开源框架,来源于RodJohnson在其著作**《ExpertoneononeJ2EEdesignanddevelopment》**中阐述的部分理念和原型衍生而来。
- Spring提倡以**“最少侵入”**的方式来管理应用中的代码,这意味着我们可以随时安装或者卸载Spring
- 适用范围:任何Java应用
- Spring的根本使命:简化Java开发
尽管J2EE能够赶上Spring的步伐,但Spring并没有停止前进,Spring继续在其他领域发展,而J2EE则刚刚开始涉及这些领域,或者还没有完全开始在这些领域的创新。移动开发、社交API集成、NoSQL数据库、云计算以及大数据都是Spring正在涉足和创新的领域。Spring的前景依然会很美好。
Spring中常用术语:
- 框架:是能完成一定功能的半成品。
框架能够帮助我们完成的是:项目的整体框架、一些基础功能、规定了类和对象如何创建,如何协作等,当我们开发一个项目时,框架帮助我们完成了一部分功能,我们自己再完成一部分,那这个项目就完成了。 - 非侵入式设计:
从框架的角度可以理解为:无需继承框架提供的任何类
这样我们在更换框架时,之前写过的代码几乎可以继续使用。 - 轻量级和重量级:
轻量级是相对于重量级而言的,轻量级一般就是非入侵性的、所依赖的东西非常少、资源占用非常少、部署简单等等,其实就是比较容易使用,而重量级正好相反。 - JavaBean:
即符合JavaBean规范的Java类 - POJO:即PlainOldJavaObjects,简单老式Java对象
它可以包含业务逻辑或持久化逻辑,但不担当任何特殊角色且不继承或不实现任何其它Java框架的类或接口。
注意:bean的各种名称——虽然Spring用bean或者JavaBean来表示应用组件,但并不意味着Spring组件必须遵循JavaBean规范,一个Spring组件可以是任意形式的POJO。
- 容器:
在日常生活中容器就是一种盛放东西的器具,从程序设计角度看就是装对象的的对象,因为存在放入、拿出等操作,所以容器还要管理对象的生命周期。
Spring的优势
- 低侵入/低耦合(降低组件之间的耦合度,实现软件各层之间的解耦)
- 声明式事务管理(基于切面和惯例)
- 方便集成其他框架(如MyBatis、Hibernate)
- 降低Java开发难度
- Spring框架中包括了J2EE三层的每一层的解决方案(一站式)
Spring能帮我们做什么
①.Spring能帮我们根据配置文件创建及组装对象之间的依赖关系。
②.Spring面向切面编程能帮助我们无耦合的实现日志记录,性能统计,安全控制。
③.Spring能非常简单的帮我们管理数据库事务。
④.Spring还提供了与第三方数据访问框架(如Hibernate、JPA)无缝集成,而且自己也提供了一套JDBC访问模板来方便数据库访问。
⑤.Spring还提供与第三方Web(如Struts1/2、JSF)框架无缝集成,而且自己也提供了一套SpringMVC框架,来方便web层搭建。
⑥.Spring能方便的与JavaEE(如JavaMail、任务调度)整合,与更多技术整合(比如缓存框架)。
Spring的框架结构
- DataAccess/Integration层包含有JDBC、ORM、OXM、JMS和Transaction模块。
- Web层包含了Web、Web-Servlet、WebSocket、Web-Porlet模块。
- AOP模块提供了一个符合AOP联盟标准的面向切面编程的实现。
- **CoreContainer(核心容器):**包含有Beans、Core、Context和SpEL模块。
- Test模块支持使用JUnit和TestNG对Spring组件进行测试。
SpringIoC和DI简介
IoC:InverseofControl(控制反转)
- 读作**“反转控制”,更好理解,不是什么技术,而是一种设计思想**,就是将原本在程序中手动创建对象的控制权,交由Spring框架来管理。
- 正控:若要使用某个对象,需要自己去负责对象的创建
- 反控:若要使用某个对象,只需要从Spring容器中获取需要使用的对象,不关心对象的创建过程,也就是把创建对象的控制权反转给了Spring框架
- **好莱坞法则:**Don’tcallme,I’llcallyou
一个例子
控制反转显然是一个抽象的概念,我们举一个鲜明的例子来说明。
在现实生活中,人们要用到一样东西的时候,第一反应就是去找到这件东西,比如想喝新鲜橙汁,在没有饮品店的日子里,最直观的做法就是:买果汁机、买橙子,然后准备开水。值得注意的是:这些都是你自己**“主动”创造**的过程,也就是说一杯橙汁需要你自己创造。
然而到了今时今日,由于饮品店的盛行,当我们想喝橙汁时,第一想法就转换成了找到饮品店的联系方式,通过电话等渠道描述你的需要、地址、联系方式等,下订单等待,过一会儿就会有人送来橙汁了。
请注意你并没有“主动”去创造橙汁,橙汁是由饮品店创造的,而不是你,然而也完全达到了你的要求,甚至比你创造的要好上那么一些。
编写第一个Spring程序
- 新建一个空的Java项目,命名为【spring】
- 新建一个名为【lib】的目录,并添加进必要的jar包,导入项目
-
在Packge【pojo】下新建一个【Source】类:
packagepojo;
publicclassSource{
privateStringfruit;//类型 privateStringsugar;//糖分描述 privateStringsize;//大小杯
/*setterandgetter*/ } -
在【src】目录下新建一个【applicationContext.xml】文件,通过xml文件配置的方式装配我们的bean
<beanname="source"class="pojo.Source"> <propertyname="fruit"value="橙子"/> <propertyname="sugar"value="多糖"/> <propertyname="size"value="超大杯"/> </bean>
-
在Packge【test】下新建一个【TestSpring】类:
packagetest;
importorg.junit.Test; importorg.springframework.context.ApplicationContext; importorg.springframework.context.support.ClassPathXmlApplicationContext; importpojo.Source;
publicclassTestSpring{
@Test publicvoidtest(){ ApplicationContextcontext=newClassPathXmlApplicationContext( newString[]{"applicationContext.xml"} ); Sourcesource=(Source)context.getBean("source"); System.out.println(source.getFruit()); System.out.println(source.getSugar()); System.out.println(source.getSize()); }
}
-
运行测试代码,可以正常拿到xml配置的bean
- 总结:
- 传统的方式:
通过new关键字主动创建一个对象 - IOC方式:
对象的生命周期由Spring来管理,直接从Spring那里去获取一个对象。IOC是反转控制(InversionOfControl)的缩写,就像控制权从本来在自己手里,交给了Spring。
参考地址:这里
DI:DependencyInjection(依赖注入)
- 指Spring创建对象的过程中,将对象依赖属性(简单值,集合,对象)通过配置设值给该对象
继续上面的例子
-
在Packge【pojo】下新建一个【JuiceMaker】类:
packagepojo;
publicclassJuiceMaker{
//唯一关联了一个Source对象 privateSourcesource=null; /*setterandgetter*/ publicStringmakeJuice(){ Stringjuice="xxx用户点了一杯"+source.getFruit()+source.getSugar()+source.getSize(); returnjuice; }
}
-
在xml文件中配置JuiceMaker对象:
-
注意:这里要使用ref来注入另一个对象
<beanname="source"class="pojo.Source"> <propertyname="fruit"value="橙子"/> <propertyname="sugar"value="多糖"/> <propertyname="size"value="超大杯"/> </bean> <beanname="juickMaker"class="pojo.JuiceMaker"> <propertyname="source"ref="source"/> </bean>
-
在【TestSpring】中添加如下代码:
packagetest;
importorg.junit.Test; importorg.springframework.context.ApplicationContext; importorg.springframework.context.support.ClassPathXmlApplicationContext; importpojo.JuiceMaker; importpojo.Source;
publicclassTestSpring{
@Test publicvoidtest(){ ApplicationContextcontext=newClassPathXmlApplicationContext( newString[]{"applicationContext.xml"} ); Sourcesource=(Source)context.getBean("source"); System.out.println(source.getFruit()); System.out.println(source.getSugar()); System.out.println(source.getSize()); JuiceMakerjuiceMaker=(JuiceMaker)context.getBean("juickMaker"); System.out.println(juiceMaker.makeJuice()); }
}
-
运行测试代码:
**总结:**IoC和DI其实是同一个概念的不同角度描述,DI相对IoC而言,明确描述了“被注入对象依赖IoC容器配置依赖对象”
IoC如何实现的
最后我们简单说说IoC是如何实现的。想象一下如果我们自己来实现这个依赖注入的功能,我们怎么来做?无外乎:
- 读取标注或者配置文件,看看JuiceMaker依赖的是哪个Source,拿到类名
- 使用反射的API,基于类名实例化对应的对象实例
- 将对象实例,通过构造函数或者setter,传递给JuiceMaker
我们发现其实自己来实现也不是很难,Spring实际也就是这么做的。这么看的话其实IoC就是一个工厂模式的升级版!当然要做一个成熟的IoC框架,还是非常多细致的工作要做,Spring不仅提供了一个已经成为业界标准的JavaIoC框架,还提供了更多强大的功能,所以大家就别去造轮子啦!希望了解IoC更多实现细节不妨通过学习Spring的源码来加深理解!
引用地址:这里
SpringAOP简介
如果说IoC是Spring的核心,那么面向切面编程就是Spring最为重要的功能之一了,在数据库事务中切面编程被广泛使用。
AOP即AspectOrientedProgram面向切面编程
首先,在面向切面编程的思想里面,把功能分为核心业务功能,和周边功能。
- 所谓的核心业务,比如登陆,增加数据,删除数据都叫核心业务
- 所谓的周边功能,比如性能统计,日志,事务管理等等
周边功能在Spring的面向切面编程AOP思想里,即被定义为切面
在面向切面编程AOP的思想里面,核心业务功能和切面功能分别独立进行开发,然后把切面功能和核心业务功能"编织"在一起,这就叫AOP
AOP的目的
AOP能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任(例如事务处理、日志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可拓展性和可维护性。
AOP当中的概念:
- 切入点(Pointcut)
在哪些类,哪些方法上切入(where) - 通知(Advice)
在方法执行的什么实际(**when:**方法前/方法后/方法前后)做什么(**what:**增强的功能) - 切面(Aspect)
切面=切入点+通知,通俗点就是:在什么时机,什么地方,做什么增强! - 织入(Weaving)
把切面加入到对象,并创建出代理对象的过程。(由Spring来完成)
AOP编程
-
在Packge【service】下创建【ProductService】类:
packageservice;
publicclassProductService{ publicvoiddoSomeService(){ System.out.println("doSomeService"); } }
-
在xml文件中装配该bean:
-
在【TestSpring】中编写测试代码,运行:
-
在Packge【aspect】下准备日志切面【LoggerAspect】类:
packageaspect;
importorg.aspectj.lang.ProceedingJoinPoint;
publicclassLoggerAspect{
publicObjectlog(ProceedingJoinPointjoinPoint)throwsThrowable{ System.out.println("startlog:"+joinPoint.getSignature().getName()); Objectobject=joinPoint.proceed(); System.out.println("endlog:"+joinPoint.getSignature().getName()); returnobject; }
}
-
在xml文件中声明业务对象和日志切面:
<beanname="productService"class="service.ProductService"/> <beanid="loggerAspect"class="aspect.LoggerAspect"/> <!--配置AOP--> <aop:config> <!--where:在哪些地方(包.类.方法)做增加--> <aop:pointcutid="loggerCutpoint" expression="execution(*service.ProductService.*(..))"/> <!--what:做什么增强--> <aop:aspectid="logAspect"ref="loggerAspect"> <!--when:在什么时机(方法前/后/前后)--> <aop:aroundpointcut-ref="loggerCutpoint"method="log"/> </aop:aspect> </aop:config>
-
再次运行TestSpring中的测试代码,代码并没有改变,但是在业务方法运行之前和运行之后,都分别输出了日志信息:
欢迎转载,转载请注明出处!
简书ID:@我没有三颗心脏
github:wmyskxz
欢迎关注公众微信号:wmyskxz_javaweb
分享自己的JavaWeb学习之路以及各种Java学习资料
本文内容总结:认识Spring框架,SpringIoC和DI简介,SpringAOP简介,
原文链接:https://www.cnblogs.com/wmyskxz/p/8820371.html