一.spring

1.spring体系结构

(1)核心容器(core container):由spring-core,spring-beans,spring-context和spring-expression(SpEL,Spring表达式语言,Spring Expression Language)等模块组成

(2)数据访问(data access):jdbc,orm(jpa,heribate),transactions

(3)web:web,web-mvc,web-socket

(4)其它:aop,AspectJ,测试模块(JUnit)

2.ApplicationContext三个常见实现类

(1).classPathXmlApplicationContext:加载类路径下的配置文件(要求配置文件必须在类路径下,不在的话就加载不了)

(2).FileSystemXmlApplicationContext:加载磁盘任意路径下的配置文件(必须有访问权限)

(3).AnnotationConfigApplicationContext:读取注解创建容器

3.BeanFactory和ApplicationContext区别?

ApplicationContext继承BeanFactory接口,是一种更高级的容器,提供了更多功能(Aop,国际化,消息发送)

区别:BeanFactory在启动的时候不会实例化Bean,从容器拿bean的时候才会实例化。ApplicationContext在启动的时候就把所有Bean实例化

4.配置详解

<?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"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd"> <!--mybatis -->
<context:property-placeholder location="classpath:jdbc.properties" system-properties-mode="NEVER"/>
<bean name="dataSouce" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="url" value="${url}"/>
<property name="username" value="${user}"/>
<property name="password" value="${password}"/>
<property name="driverClassName" value="${driver}"/>
</bean> <bean class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSouce"/>
</bean> <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.kkb.dao"/>
</bean>
<!--mybatis END --> <!--注解扫描 -->
<context:component-scan base-package="com.kkb.service"/> </beans>

(1)bean元素:使用该元素描述需要spring容器管理对象

(2)name属性:给被管理的对象起个名字,获得对象时getBean("name值")

(3)class属性:被管理对象的完整类名

(4)id属性:与name属性一模一样,名称不可重复,不能使用特殊字符

id与name的区别?

①id是唯一的,配置文件中不允许出现两个id相同的。中允许出现两个name相同的,在用getBean()返回实例时,后面一个Bean被返回

②如果没有配置id和name,则用类的全名作为name

(5)scope:singleton,prototype,request,session

绝大多数情况下,使用单例singleton(默认值),但是在与struts整合时候,务必要用prototype多例,因为struts2在每次请求都会创建一个新的Action,若为单例,在多请求情况下,每个请求找找spring拿的都是同一个action

(6)初始化和销毁:init-method,destory-method。spring会在对象创建之后立刻调用 init-method 对应注解为@PreDestory,spring容器在关闭并销毁所有容器中的对象之前调用destory-method对应注解为@PostConstruct

5.spring三种对象的创建方式

(1)空参构造

<!-- 使用无参构造函数创建对象 -->
<bean id="bean3" class="cn.itcast.beans.Bean3"></bean>

(2)静态工厂创建

<!-- 静态工厂创建(调用静态方法创建) -->
<bean name="user" class="cn.itcats.UserFactory" factory-method="createUser"></bean> public class Bean1_factory {
public static Bean1 getBean(){
return new Bean1();
}
}
public class Bean1 {
public void add(){
System.out.println("bean1 ........");
}
}

(3)实例工厂

需要配置两个bean,因为无法通过类名调用非静态方法

<!-- 使用实例工厂创建对象 -->
<bean id="bean2_fcty" class="cn.itcast.beans.Bean2_factory">
<bean id="bean2" factory-bean="bean2_fcty" factory-method="getBean"></bean>
public class Bean2_factory {
public Bean2 getBean(){
return new Bean2();
}
}
public class Bean2 {
public void add(){
System.out.println("bean2 ........");
}
}

6.spring注入方式

(1)构造方法注入

使用对象的话,应改用ref属性

<bean id="user" class="bean.User">
<constructor-arg index="0" name="username" value="zhangsan"></constructor-arg>
<constructor-arg index="1" name="password" value="1234"></constructor-arg>
</bean>

(2)set注入

与构造器注入相比,只是少了index属性

<bean id="userService" class="com.lyu.spring.service.impl.UserService">
<property name="userDao" ref="userDaoMyBatis"></property>
</bean>

(3)注解注入

①引用类型注入:

@Autowired:spring注解,默认是以byType的方式去匹配,如果没有找到,就通过byName的方式去查找

@Resource:java的注解,默认以byName的方式去匹配,如果没有找到就会以byType的方式查找

@Qualifier:(spring注解)指定某个具体名称的bean(配合autowired和resource使用)

②值类型注入

@Value("kaka")

@Resource
@Qualifier("userDaoMyBatis")
private IUserDao userDao;
public UserService(){
@Autowired
@Qualifier("userDaoJdbc")
private IUserDao userDao;

7.复杂类型注入

<!-- 注入复杂类型属性 -->

<bean id="user" class="com.siwuxie095.property.User">

	<!-- 数组 -->
<property name="arr">
<list>
<value>小孙</value>
<value>小李</value>
</list>
</property> <!-- List 集合 -->
<property name="list">
<list>
<value>张三</value>
<value>李四</value>
<value>王五</value>
</list>
</property> <!-- Map 集合 -->
<property name="map">
<map>
<entry key="a" value="老大"></entry>
<entry key="b" value="老二"></entry>
</map>
</property> <!-- Properties 类型 -->
<property name="properties">
<props>
<prop key="username">root</prop>
<prop key="password">8888</prop>
</props>
</property>
</bean>

8.web.xml中spring的配置

<!--spring随项目的创建而创建,随项目的关闭而关闭,避免创建多个ApplicationContext-->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!--指定加载spring配置文件的位置-->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/applicationContext.xml</param-value>
</context-param>

9.spring整合junit

@RunWith(SpringJUnit4ClassRunner.class) //RunWith用于指定junit运行环境,是junit提供给其他框架测试环境接口扩展
@ContextConfiguration("classpath:applicationContext.xml") //配置xml文件
public class Demo { @Autowired
private User user; @Test
public void fun01() {
System.out.println(user);
}
}

10.spring的aop中名词解释

(1)JoinPoint(连接点):目标对象中,所有可以增强的方法,就是spring允许你是通知(Advice)的地方,那可就真多了,基本每个方法的前、后(两者都有也行),或抛出异常是时都可以是连接点,spring只支持方法连接点。

(2)Pointcut(切入点):目标对象中,已经被增强的方法。调用这几个方法之前、之后或者抛出异常时干点什么,那么就用切入点来定义这几个方法。

(3)Advice(通知/增强) :增强方法的代码、想要的功能

(4)Target(目标对象):被代理对象,被通知的对象,被增强的类对象

(5)Weaving(织入):将通知应用到连接点形成切入点的过程

(6)Proxy(代理):将通知织入到目标对象之后形成的代理对象

(7)aspect(切面):切入点+通知

11.aop实现(基于注解)

@Before 前置通知(Before advice) :在某连接点(JoinPoint)——核心代码(类或者方法)之前执行的通知,但这个通知不能阻止连接点前的执行。为啥不能阻止线程进入核心代码呢?因为@Before注解的方法入参不能传ProceedingJoinPoint,而只能传入JoinPoint。要知道从aop走到核心代码就是通过调用ProceedingJionPoint的proceed()方法。而JoinPoint没有这个方法。

这里牵扯区别这两个类:Proceedingjoinpoint 继承了 JoinPoint 。是在JoinPoint的基础上暴露出 proceed 这个方法。proceed很重要,这个是aop代理链执行的方法。暴露出这个方法,就能支持 aop:around 这种切面(而其他的几种切面只需要用到JoinPoint,这跟切面类型有关), 能决定是否走代理链还是走自己拦截的其他逻辑。建议看一下 JdkDynamicAopProxy的invoke方法,了解一下代理链的执行原理。这样你就能明白 proceed方法的重要性。

@After 后通知(After advice) :当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)。

@AfterReturning 返回后通知(After return advice) :在某连接点正常完成后执行的通知,不包括抛出异常的情况。

@Around 环绕通知(Around advice) :包围一个连接点的通知,类似Web中Servlet规范中的Filter的doFilter方法。可以在方法的调用前后完成自定义的行为,也可以选择不执行。这时aop的最重要的,最常用的注解。用这个注解的方法入参传的是ProceedingJionPoint pjp,可以决定当前线程能否进入核心方法中——通过调用pjp.proceed();

@AfterThrowing 抛出异常后通知(After throwing advice) : 在方法抛出异常退出时执行的通知。

@Aspect //@Aspect注解代表该类是个通知类,书写切点表达式@Pointcut("execution(返回值 全类名.方法名(参数))")
public class LogAspect { //抽取公共的切入点表达式
//1、本类引用
//2、其他的切面引用
@Pointcut("execution(* com.gz.MathCalculator.*(..))")
public void pointCut(){}; @Before("pointCut()")
public void logStart(JoinPoint joinPoint){
Object[] args = joinPoint.getArgs();
System.out.println(""+joinPoint.getSignature().getName()+"运行。。。@Before:参数列表是:{"+Arrays.asList(args)+"}");
} @After("com.gz.LogAspect.pointCut()")
public void logEnd(JoinPoint joinPoint){
System.out.println(""+joinPoint.getSignature().getName()+"结束。。。@After");
} @AfterReturning(value="pointCut()",returning="result")
public void logReturn(JoinPoint joinPoint,Object result){
System.out.println(""+joinPoint.getSignature().getName()+"正常返回。。。@AfterReturning:运行结果:{"+result+"}");
} @AfterThrowing(value="pointCut()",throwing="exception")
public void logException(JoinPoint joinPoint,Exception exception){
System.out.println(""+joinPoint.getSignature().getName()+"异常。。。异常信息:{"+exception+"}");
} }

(1)@Before 前置通知(Before advice) :在某连接点(JoinPoint)之前执行的通知

(2)@After 后通知(After advice) :当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)

(3)@AfterReturning 返回后通知(After return advice) :在某连接点正常完成后执行的通知,不包括抛出异常的情况

(4)@AfterThrowing 抛出异常后通知(After throwing advice) : 在方法抛出异常退出时执行的通知

(5)@Around 环绕通知(Around advice):包围一个连接点的通知,类似Web中Servlet规范中的Filter的doFilter方法。可以在方法的调用前后完成自定义的行为,也可以选择不执行

12.spring中事务

spring中事务可以分为编程式事务控制和声明式事务控制

编程式事务:自己手动控制事务,就叫做编程式事务控制。 【细粒度的事务控制: 可以对指定的方法、指定的方法的某几行添加事务控制】比较灵活,但开发起来比较繁琐: 每次都要开启、提交、回滚

声明式事务:Spring提供了对事务的管理, 这个就叫声明式事务管理【粗粒度的事务控制: 只能给整个方法应用事务,不可以对方法的某几行应用事务】 (因为aop拦截的是方法)

13.事务传播行为

事务传播行为(propagation behavior)指的就是当一个事务方法被另一个事务方法调用时,这个事务方法应该如何进行。

例如:methodA事务方法调用methodB事务方法时,methodB是继续在调用者methodA的事务中运行呢,还是为自己开启一个新事务运行,这就是由methodB的事务传播行为决定的。

(1)propagation_required:如果存在一个事务,则支持当前事务。如果没有事务则开启一个新的事务。



(2)propagation_supports:如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行

@Transactional(propagation = Propagation.REQUIRED)
public void methodA() {
methodB();
// do something
} // 事务属性为SUPPORTS
@Transactional(propagation = Propagation.SUPPORTS)
public void methodB() {
// do something
}
单纯的调用methodB时,methodB方法是非事务的执行的。当调用methdA时,methodB则加入了methodA的事务中,事务地执行

(3)propagation_mandatory:如果已经存在一个事务,支持当前事务。如果没有一个活动的事务,则抛出异常

(4)propagation_requires_new:它会开启一个新的事务。如果一个事务已经存在,则先将这个存在的事务挂起(需要使用 JtaTransactionManager作为事务管理器)



(5)propagation_not_supported:总是非事务地执行,并挂起任何存在的事务。使用PROPAGATION_NOT_SUPPORTED(也需要使用JtaTransactionManager作为事务管理器)

(6)propagation_never:总是非事务地执行,如果存在一个活动事务,则抛出异常。

(7)propagation_nested: 如果一个活动的事务存在,则运行在一个嵌套的事务中。 如果没有活动事务, 则按propagation_required属性执行(嵌套事务一个非常重要的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚)

14.propagation_nested 与propagation_requires_new的区别?

它们非常类似,都像一个嵌套事务,如果不存在一个活动的事务,都会开启一个新的事务。

(1)使用 PROPAGATION_REQUIRES_NEW时,内层事务与外层事务就像两个独立的事务一样,一旦内层事务进行了提交后,外层事务不能对其进行回滚。两个事务互不影响。两个事务不是一个真正的嵌套事务。同时它需要JTA事务管理器的支持。

(2)使用PROPAGATION_NESTED时,外层事务的回滚可以引起内层事务的回滚。而内层事务的异常并不会导致外层事务的回滚,它是一个真正的嵌套事务。DataSourceTransactionManager支持

15.aop管理事务(注解)

(1)开启注解

<tx:annotation-driven/>

(2)方法上增加@Transactional注解方法就可以被事务管理起来

@Transactional注解的属性:

readOnly : 是否仅仅只读。默认读写都可以

timeout : 事务超时时间,默认没有超时时间

isolation: 事务的隔离级别 默认:TransactionDefinition.ISOLATION_DEFAULT(大部分数据库的默认隔离级别是可重复读)

propagation :事务的传播属性 默认:TransactionDefinition.PROPAGATION_REQUIRED

【总结】spring基础的更多相关文章

  1. Spring基础知识

    Spring基础知识 利用spring完成松耦合 接口 public interface IOutputGenerator { public void generateOutput(); } 实现类 ...

  2. spring基础整理

    spring基础教程:https://www.tutorialspoint.com/spring/spring_overview.htm 注入实例 <bean id="" c ...

  3. Spring 基础知识

    Spring架构简单描述 原文:https://www.shiyanlou.com/courses/document/212 Spring 概述 1. Spring 是什么 Spring是一个开源的轻 ...

  4. Spring基础配置

    从毕业到现在我一直从事Android开发,但是对JavaEE一直念念不忘,毕业校招的时候,一个礼拜拿了三个offer,岗位分别是Android.JavaEE和JavaSE,后来觉得Android比较简 ...

  5. Spring基础系列--AOP织入逻辑跟踪

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9619910.html 其实在之前的源码解读里面,关于织入的部分并没有说清楚,那些前置.后 ...

  6. 第65节:Java后端的学习之Spring基础

    Java后端的学习之Spring基础 如果要学习spring,那么什么是框架,spring又是什么呢?学习spring中的ioc和bean,以及aop,IOC,Bean,AOP,(配置,注解,api) ...

  7. Spring基础系列-AOP源码分析

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9560803.html 一.概述 Spring的两大特性:IOC和AOP. AOP是面向切 ...

  8. Spring基础系列-Spring事务不生效的问题与循环依赖问题

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9476550.html 一.提出问题 不知道你是否遇到过这样的情况,在ssm框架中开发we ...

  9. Spring Boot实战(1) Spring基础

    1. Spring基础配置 Spring框架本身有四大原则: 1) 使用POJO进行轻量级和最小侵入式开发 2) 通过依赖注入和基于接口编程实现松耦合 3) 通过AOP和默认习惯进行声明式编程 4) ...

  10. spring基础学习01

    spring基础 Spring是一个开放源代码的设计层面框架,他解决的是业务逻辑层和其他各层的松耦合问题,因此它将面向接口的编程思想贯穿整个系统应用 IOC控制反转 把创建对象和维护对象之间的关系权利 ...

随机推荐

  1. 用JTable 实现日历

    效果图: 主要思想:日历最核心的功能就是能显示某年某月对应的日期和星期几.因此只要实现传入具体的年份和月份,得到一组存放了日期的数组a[ ]即可.其中数组的大小设置成42,要考虑的问题是当月的第一天对 ...

  2. SpringCache整合Redis

    之前一篇文章 SpringBoot整合Redis 已经介绍了在SpringBoot中使用redisTemplate手动 操作redis数据库的方法了.其实这个时候我们就已经可以拿redis来做项目了, ...

  3. Visual C# 制作DLL文件

    一.制作.dll1.首先创建一个新类库工程文件  文件->新建->项目->Visual C#->类库.填入工程文件名称,并且选择文件要存放的目录. 2.工程文件 将Class1 ...

  4. VBScript 教程

    VBScript 教程 VB 不区分大小写 变量 普通变量 关键词声明 Dim.Public.Private 赋值动态创建 name = "hello" Option Explic ...

  5. Mac 每次都要执行source ~/.bash_profile 后,配置的环境变量才生效

    问题: 自己在 ~/.bash_profile 中配置环境变量, 可是每次重启终端后配置的不生效.需要重新执行 : $source ~/.bash_profile后,才会生效. 原因: 自己是在bas ...

  6. MeteoInfoLab脚本示例:SeaWiFS HDF Grid数据

    SeaWiFS HDF Grid数据读取,特别是涉及到了文件的众多属性数据的读取,数据取对数后绘图.脚本程序: #Add data file f = addfile('D:/Temp/hdf/S199 ...

  7. 快速掌握ES6语法

    常量变量 let and const 先说说常量和变量的概念吧, 常量是说那种进行一次赋值后不会更改的值,比如说游戏账户的 ID, 变量是说赋值后有更改的需求的,比如游戏名,游戏密码. 在之前的 Ja ...

  8. 数组的高级应用含ES6 for of 用法

    // 在ES5中常用的10种数组遍历方法: // 1. 原始的for循环语句 // 2. Array.prototype.forEach数组对象内置方法 // 3. Array.prototype.m ...

  9. spring boot:接口站增加api版本号后的安全增强(spring boot 2.3.3)

    一,接口站增加api版本号后需要做安全保障? 1,如果有接口需要登录后才能访问的, 需要用spring security增加授权 2,接口站需要增加api版本号的检验,必须是系统中定义的版本号才能访问 ...

  10. spring boot:spring security整合jwt实现登录和权限验证(spring boot 2.3.3)

    一,为什么使用jwt? 1,什么是jwt? Json Web Token, 它是JSON风格的轻量级的授权和身份认证规范, 可以实现无状态.分布式的Web应用授权 2,jwt的官网: https:// ...