spring 总结
控制反转(Inversion of Control)
依赖注入(Dependency Injection)
一、控制反转
1.控制:传统的方式是由程序创建的,写死的, spring 是由spring管理创建对象,实现了解耦。
2.反转:由主动创建变为被动接受,正转:程序创建对象。
优点:比如两个实现类 mysql的实现,oracle的实现,如果不用控制反转,就需要程序写死,日后维护不方便,如果用控制反转,只需要修改配置文件,动态传入是mysql,oracle
二、依赖注入:有spring容器来装配对象
set注入,构造方法注入,数组注入,集合注入,配置文件properties注入,命名空间,P,C注入
控制反转和依赖注入主要是为了实现多态,实现程序解耦,可扩展
IOC 配置文件方式
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:util="http://www.springframework.org/schema/util"
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/util http://www.springframework.org/schema/util/spring-util.xsd"> <bean id="user1" class="com.model.User">
<property name="name" value="张三"></property>
<!-- 引用外部对象注入 -->
<property name="car" ref="car"></property>
</bean> <bean id="car" class="com.model.Car"></bean>
<!-- 注入特殊字符 -->
<bean id="user2" class="com.model.User">
<constructor-arg index="0">
<value type="java.lang.String">
<![CDATA[<SAFASFD&&&^^^>]]>
</value>
</constructor-arg> <!-- 在当前属性中注入 外部不能引用 -->
<property name="car">
<bean class="com.model.Car"></bean>
</property>
</bean> <bean class="com.process.MyBeanPostProcess"></bean>
<!-- 引用资源文件用SEPL获取${jdbc_url} -->
<!-- <context:property-placeholder location="classpath:db.properties"></context:property-placeholder> -->
</beans>
2注解方式
<!-- 不用默认的扫描 用自定义的扫描,扫描多个包用,base-package="com.controller,com.service" 可以放多个context:include-filter或context:include-filter-->
<!-- <context:component-scan base-package="com.controller" resource-pattern="" use-default-filters="false">
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
<context:include-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/>
</context:component-scan> --> <!-- 对应的整合spring的配置配置文件不扫描springmvc的注解,为了避免重复扫描创建两个对象 -->
<!-- <context:component-scan base-package="com.controller">
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
<context:exclude-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/>
</context:component-scan> --> <!-- resource-pattern="/service/*.class 只扫描service下的加注解的类 -->
<!-- <context:component-scan base-package="com.user" resource-pattern="/service/*.class"></context:component-scan> -->
<context:component-scan base-package="com.base" ></context:component-scan>
- @Component 是一个泛化的概念,仅仅表示一个组件 (Bean) ,可以作用在任何层次。
- @Service 通常作用在业务层,但是目前该功能与 @Component 相同。
- @Constroller 通常作用在控制层,但是目前该功能与 @Component 相同。
- @Repository 通常作用于持久层
@Service("userService")//不写值 默认注入的名称是首字母小写userAddServiceImpl
public class UserAddServiceImpl implements UserService{ public void add() {
System.out.println("UserAddServiceImpl--------添加");
} }
@Autowired//自动装配 @Resource一样的效果,@Inject
@Qualifier(value="userUpdateServiceImpl") //如果有多个实现可具体指定注入哪个
private UserService userService;
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// @Configuration注解的spring容器加载方式,用AnnotationConfigApplicationContext替换ClassPathXmlApplicationContext
ApplicationContext context = new AnnotationConfigApplicationContext(TestConfiguration.class); // 如果加载spring-context.xml文件:
// ApplicationContext context = new ClassPathXmlApplicationContext("spring-context.xml");
ApplicationContext context = WebApplicationContextUtils.getWebApplicationContext(servletContextEvent.getServletContext());
userService = context.getBean(UserService.class);
ResourcePatternResolver resolver =
new
PathMatchingResourcePatternResolver();
Resource resources[] = resolver.getResources(
"classpath*://com/***/*.xml"
);
AOP:注解
<!-- 基础注解方式的aop -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
package com.aop.annotation; import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component; /**
* 日志切面
* @author Administrator
*
*/ @Component
@Aspect
@Order(1)//值越小优先级越高,越先执行
public class LogAspects { /**
* 声明切入点表达式
* 在其他切面中可@Before("com.aop.annotation.declarePointExecution()")引用
*/
@Pointcut("execution(* com.aop.annotation.*.*(..))")
public void declarePointExecution(){ } @Before("declarePointExecution()")
public void before(JoinPoint joinPoint){
System.out.println(joinPoint.getSignature().getName()+"-----方法调用之前");
} /**
* 后置通知方法抛异常仍然会执行,不能得到方法的返回结果
* @param joinPoint
*/
@After("execution(* com.aop.annotation.*.*(..))")
public void after(JoinPoint joinPoint){
System.out.println(joinPoint.getSignature().getName()+"-----方法调用之后");
} /**
* 返回通知,出异常不会被调用
* @param joinPoint
*/
@AfterReturning(value="execution(* com.aop.annotation.*.*(..))",returning="result")
public void AfterReturning(JoinPoint joinPoint,Object result){
System.out.println(joinPoint.getSignature().getName()+"-----方法调用的返回结果-------"+result);
} /**
* 异常通知,返回异常,可指定出现特定异常如NullPointerException
* @param joinPoint
*/
@AfterThrowing(value="execution(* com.aop.annotation.*.*(..))",throwing="ex")
public void AfterReturning(JoinPoint joinPoint,Exception ex){
System.out.println(joinPoint.getSignature().getName()+"-----方法调用出现异常-------"+ex);
} /**
* 环绕通知,和动态代理相似
* 必须有返回值
* @param joinPoint
*/
@Around(value="execution(* com.aop.annotation.*.*(..))")
public Object Around(ProceedingJoinPoint pjd){
//System.out.println(joinPoint.getSignature().getName()+"-----环绕通知-------");
try {
return pjd.proceed();
} catch (Throwable e) {
e.printStackTrace();
}
return null;
} }
AOP:XML
<aop:config>
<aop:aspect id="jptSmsAopAspect" ref="jptSmsAop" order="5">
<aop:after-returning pointcut="execution(* com.jit.gwcs.services.gw..*.*(..))" method="afterService" />
</aop:aspect>
</aop:config> <bean id="jptSmsAop" class="com.jit.gwcs.services.gw.sms.aop.SmsAop">
<property name="jptSmsService" ref="jptSmsService"></property>
</bean>
<!-- 基础xml方式的aop -->
<bean id="userService" class="com.aop.xml.UserService"></bean>
<bean id="logAspects" class="com.aop.xml.LogAspects"></bean> <aop:config>
<!-- 定义一个切点 -->
<aop:pointcut expression="execution(* com.aop.xml.*.*(..))" id="pointcut"/>
<!-- 配置切面通知 -->
<aop:aspect ref="logAspects" order="2">
<aop:before method="before" pointcut-ref="pointcut"/>
</aop:aspect> </aop:config>
1.before:前置通知(应用:各种校验)
在方法执行前执行,如果通知抛出异常,阻止方法运行
2.afterReturning:后置通知(应用:常规数据处理)
方法正常返回后执行,如果方法中抛出异常,通知无法执行
必须在方法执行后才执行,所以可以获得方法的返回值。
3.around:环绕通知(应用:十分强大,可以做任何事情) 【掌握】
方法执行前后分别执行,可以阻止方法的执行。要求必须手动的执行目标方法。
4.afterThrowing:抛出异常通知(应用:包装异常信息)
方法抛出异常后执行,如果方法没有抛出异常,无法执行
5.after:最终通知(应用:清理现场)
方法执行完毕后执行,无论方法中是否出现异常
事务:基于注解的事务
<!-- 配置事务管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean> <!-- 注解方式配置事物 在DAO中方法上添加@Transactional-->
<!-- 事务的传播行为 propagation 对哪些异常不回滚noRollbackFor-->
<!-- 事务的隔离级别@Transactional(isolation=Isolation.READ_COMMITTED) -->
<!-- <tx:annotation-driven transaction-manager="transactionManager" /> -->
事务:基于XML
<!-- 配置事务管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean> <!-- 拦截器方式配置事物 -->
<tx:advice id="transactionAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="append*" propagation="REQUIRED" />
<tx:method name="insert*" propagation="REQUIRED" />
<tx:method name="save*" propagation="REQUIRED" />
<tx:method name="update*" propagation="REQUIRED" />
<tx:method name="modify*" propagation="REQUIRED" />
<tx:method name="edit*" propagation="REQUIRED" />
<tx:method name="delete*" propagation="REQUIRED" />
<tx:method name="remove*" propagation="REQUIRED" />
<tx:method name="repair" propagation="REQUIRED" />
<tx:method name="delAndRepair" propagation="REQUIRED" /> <tx:method name="get*" propagation="SUPPORTS" />
<tx:method name="find*" propagation="SUPPORTS" />
<tx:method name="load*" propagation="SUPPORTS" />
<tx:method name="search*" propagation="SUPPORTS" />
<tx:method name="datagrid*" propagation="SUPPORTS" /> <tx:method name="*" propagation="SUPPORTS" />
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="transactionPointcut" expression="execution(* com.dogoframework.service.*.*(..))" />
<aop:advisor pointcut-ref="transactionPointcut"
advice-ref="transactionAdvice" />
</aop:config>
WEB初始化spring IOC原理自定义
web.xml
<context-param>
<param-name>context</param-name>
<param-value>spring.xml</param-value>
</context-param> <listener>
<listener-class>com.listener.my.SpringServletContextListener</listener-class>
</listener>
package com.listener; import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener; import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils; /**
* Application Lifecycle Listener implementation class SpringServletContextListener
*
*/
public class SpringServletContextListener implements ServletContextListener { /**
* Default constructor.
*/
public SpringServletContextListener() {
// TODO Auto-generated constructor stub
} public void contextInitialized(ServletContextEvent sce) {
//在应用初始化后初始化IOC容器
ServletContext context = sce.getServletContext();
/*<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring.xml,classpath:spring-mybatis.xml,classpath:spring-druid.xml,classpath:spring-shiro.xml</param-value>
</context-param> *<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
*
*/
String config = context.getInitParameter("contextConfigLocation"); ApplicationContext context2 = new ClassPathXmlApplicationContext(config); //WebApplicationContextUtils.getRequiredWebApplicationContext(context); context.setAttribute("applicationContext", context2);
// TODO Auto-generated method stub } public void contextDestroyed(ServletContextEvent sce) {
// TODO Auto-generated method stub } }
spring提供初始化
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring.xml,classpath:spring-mybatis.xml,classpath:spring-druid.xml,classpath:spring-shiro.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
获取spring IOC容器 spring提供的
WebApplicationContext context3 = WebApplicationContextUtils.getRequiredWebApplicationContext(context);
package com.process; import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor; /**
* Bean后置处理器
* @author Administrator
*
*/
public class MyBeanPostProcess implements BeanPostProcessor{ public Object postProcessBeforeInitialization(Object bean, String beanName)
throws BeansException { System.out.println(beanName+"------------所有BEAN init前被调用"); return bean;
} public Object postProcessAfterInitialization(Object bean, String beanName)
throws BeansException {
System.out.println(beanName+"------------所有BEAN init后被调用");
return bean; } }
servlet获取spring 应用上下文
WebApplicationContext ctx = WebApplicationContextUtils.getRequiredWebApplicationContext(pageContext.getServletContext());
spring获取servlet上下文
ServletContext servletContext = ContextLoader.getCurrentWebApplicationContext().getServletContext();
任意公共方法的执行:
execution(public * *(..))
任何一个以“set”开始的方法的执行:
execution(* set*(..))
AccountService 接口的任意方法的执行:
execution(* com.xyz.service.AccountService.*(..))
定义在service包里的任意方法的执行:
execution(* com.xyz.service.*.*(..))
定义在service包和所有子包里的任意类的任意方法的执行:
execution(* com.xyz.service..*.*(..))
定义在pointcutexp包和所有子包里的JoinPointObjP2类的任意方法的执行:
execution(* com.test.spring.aop.pointcutexp..JoinPointObjP2.*(..))")
在多个表达式之间使用 ||,or表示 或,使用 &&,and表示 与,!表示 非.例如:
<aop:config>
<aop:pointcut
id="pointcut" expression="(execution(* com.ccboy.dao..*.find*(..))) or
(execution(* com.ccboy.dao..*.query*(..)))"/>
<aop:advisor advice-ref="jdbcInterceptor" pointcut-ref="pointcut" />
</aop:config>
二、spring作用域
scope
1、单例
@Lazy 延迟加载,初始化时不创建,获取的时候创建
2、原型
单例和原型创建bean的时期不同,单例是ioc容器初始化是创建对象,原型是获取时创建
3、request
4、session
5、globe session
三、bean创建条件
1、实现Condition接口,重写matches
2、@Import({ImportSelector.class})
3、实现FactoryBean,把FactoryBean注册到容器中
四、bean生命周期
1、创建
2、初始化
3、属性赋值
4、销毁
五、@Component,@Service,@Controller,@Repository 注解bean
@PostConstruct 相当于init
@PreDestroy 相当于destroy
或者实现接口
bean
InitializingBean 初始化
DisposableBean 销毁
除了bean ,给所有bean
实现接口 BeanPostProcessor
@Bean
自动装配放到方法参数前
public void aaa(@Autowired XX xx)
spring 总结的更多相关文章
- 基于spring注解AOP的异常处理
一.前言 项目刚刚开发的时候,并没有做好充足的准备.开发到一定程度的时候才会想到还有一些问题没有解决.就比如今天我要说的一个问题:异常的处理.写程序的时候一般都会通过try...catch...fin ...
- 玩转spring boot——快速开始
开发环境: IED环境:Eclipse JDK版本:1.8 maven版本:3.3.9 一.创建一个spring boot的mcv web应用程序 打开Eclipse,新建Maven项目 选择quic ...
- Spring基于AOP的事务管理
Spring基于AOP的事务管理 事务 事务是一系列动作,这一系列动作综合在一起组成一个完整的工作单元,如果有任何一个动作执行失败,那么事务 ...
- [Spring]IoC容器之进击的注解
先啰嗦两句: 第一次在博客园使用markdown编辑,感觉渲染样式差强人意,还是github的样式比较顺眼. 概述 Spring2.5 引入了注解. 于是,一个问题产生了:使用注解方式注入 JavaB ...
- 学习AOP之透过Spring的Ioc理解Advisor
花了几天时间来学习Spring,突然明白一个问题,就是看书不能让人理解Spring,一方面要结合使用场景,另一方面要阅读源代码,这种方式理解起来事半功倍.那看书有什么用呢?主要还是扩展视野,毕竟书是别 ...
- 学习AOP之深入一点Spring Aop
上一篇<学习AOP之认识一下SpringAOP>中大体的了解了代理.动态代理及SpringAop的知识.因为写的篇幅长了点所以还是再写一篇吧.接下来开始深入一点Spring aop的一些实 ...
- 学习AOP之认识一下Spring AOP
心碎之事 要说知道AOP这个词倒是很久很久以前了,但是直到今天我也不敢说非常的理解它,其中的各种概念即抽象又太拗口. 在几次面试中都被问及AOP,但是真的没有答上来,或者都在面上,这给面试官的感觉就是 ...
- 为什么做java的web开发我们会使用struts2,springMVC和spring这样的框架?
今年我一直在思考web开发里的前后端分离的问题,到了现在也颇有点心得了,随着这个问题的深入,再加以现在公司很多web项目的控制层的技术框架由struts2迁移到springMVC,我突然有了一个新的疑 ...
- Spring之旅(2)
Spring简化Java的下一个理念:基于切面的声明式编程 3.应用切面 依赖注入的目的是让相互协作的组件保持松散耦合:而AOP编程允许你把遍布应用各处的功能分离出来形成可重用的组件. AOP面向切面 ...
- Spring之旅
Java使得以模块化构建复杂应用系统成为可能,它为Applet而来,但为组件化而留. Spring是一个开源的框架,最早由Rod Johnson创建.Spring是为了解决企业级应用开发的复杂性而创建 ...
随机推荐
- 洛谷 P1739 表达式括号匹配
题目链接https://www.luogu.org/problemnew/show/P1739 题目描述 假设一个表达式有英文字母(小写).运算符(+,—,*,/)和左右小(圆)括号构成,以“@”作为 ...
- ANTLR v4 权威参考笔记(目录)
ANTLR v4是一款强大的语法分析器生成器,可以用来读取.处理.执行和转换结构化文本或二进制文件.通过称为文法的形式化语言描述,ANTLR可以为该语言自动生成词法分析器.生成的语法分析器可以自动构建 ...
- 安装sqlserver后 服务启动过几秒就自动停止
今天安装sqlserver2014后 发现启动2014的服务,过几秒就会自动停止 通过查看windows日志发现是系统库路径报错~ google了下 发现 "重新生成系统库"就能解 ...
- 怎么让Windows2012和Windows2008多用户同时远程---经过测试有效
链接地址:https://jingyan.baidu.com/article/cd4c2979f19765756e6e60ec.html 怎么让Windows2012和Windows2008多用户同时 ...
- 【Unity】UGUI聊天消息气泡 随文本内容自适应
游戏中需要用做UGUI做聊天界面.其中聊天气泡ChatItem的UI要求能随着聊天内容文本的长度自适应的. 网上搜了一下聊天气泡的UI,发现都不太符合咱的需求,具体来说是文本宽度不足一行时,文本宽度自 ...
- set names utf8;
对应用程序来说,强制将它们发起的数据库链接设置成UTF8编码有什么办法? 每个链接建立时先执行set names utf8; [mysqld] init-connect=‘set names utf8 ...
- 解决Can 't connect to local MySQL server through socket '/tmp/mysql.sock '(2) ";
解决方案: https://blog.csdn.net/HeatDeath/article/details/79065872 https://blog.csdn.net/hjf161105/artic ...
- 【转】WPF Template模版之DataTemplate与ControlTemplate的关系和应用(二)
1. DataTemplate和ControlTemplate的关系 学习过DataTemplate和ControlTemplate,你应该已经体会到,控件只是数据的行为和载体,是个抽象的概念,至于它 ...
- 【转】WPF Template模版之DataTemplate与ControlTemplate(一)
WPF系统不但支持传统的Winfrom编程的用户界面和用户体验设计,更支持使用专门的设计工具Blend进行专业设计,同时还推出了以模板为核心的新一代设计理念. 1. 模板的内涵 作为表现形式,每个控件 ...
- 通用 正则表达式 C# (.NET)Regex 总结
[参考]C#正则表达式Regex类的用法 语法: 1. new System.Text.RegularExpressions.Regex("\\$\\d{1,2}\\}"). ...