Spring5学习笔记

spring5框架概述

  1. Spring是轻量级的开源的JavaEE框架

  2. Spring可以解决企业应用开发的复杂性

  3. Spring有两个核心部分:IOC和AOP

    1. IOC:控制反转,把创建对象过程交给 Spring 进行管理

    2. Aop:面向切面,不修改源代码进行功能增强

  4. Spring特点

    1. 方便解耦,简化开发

    2. AOP编程支持

    3. 方便程序测试

    4. 方便和其他框架进行整合

    5. 方便进行事务操作

    6. 降低API开发难度

IOC

IOC需要引入的依赖jar包

commons-logging-1.1.1.jar

spring-beans-5.2.6.RELEASE.jar

spring-context-5.2.6.RELEASE.jar

spring-core-5.2.6.RELEASE.jar

spring-exprission-5.2.6.RELEASE.jar

什么是IOC

  1. 控制反转,把对象创建和对象之间的调用过程,交给Spring进行管理

    1. 使用ICO的目的:为了降低耦合度

IOC底层原理

xml解析、工厂模式、反射

IOC过程(底层原理)

  1. xml配置文件,配置创建的对象
<bean id= "dao" class="com.atguigu.UserDao"></bean>
  1. 由service类和dao类,创建工厂类
class UserFactory{
    public static UserDao getDao(){
        String classValue = class属性值//1.xml解析
        Class clazz = Class.forName(classValue);//2.通过反射创建对象
    }
}

IOC接口(BeanFactory)

  1. IOC思想基于IOC容器完成,容器底层就是对象工厂

  2. Spring 提供 IOC 容器实现两种方式:(两个接口)

    • BeanFactory:IOC 容器基本实现,是 Spring 内部的使用接口,不提供开发人员进行使用

      * 加载配置文件时候不会创建对象,在获取对象(使用)才去创建对象

    • ApplicationContext:BeanFactory 接口的子接口,提供更多更强大的功能,一般由开发人

      员进行使用

      * 加载配置文件时候就会把在配置文件对象进行创建

      ​ ApplicationContext接口的实现类:

IOC操作Bean管理(概念)

Bean管理共有两个操作

  1. 创建对象
  2. 注入属性

Bean管理的两种方式

  1. 基于xml配置文件方式实现
  2. 基于注解方式实现

IOC操作Bean管理(基于xml方式)

基于xml方式创建对象

<!--配置User对象创建-->
<bean id="user" class="com.atguigu.spring5.User"></bean>

​ 默认执行无参构造器创建

基于xml方式属性注入

DI:依赖注入,也就是属性注入

  1. 使用set方法进行注入

    创建类,定义属性和对应的set方法

public class Book{
    private String name;
    private String outhor;
    
    public void setName(String name){
        this.name = name;
    }
    public void setOurthor(String outhor){
        this.outhor = outhor;
    }
}

​ 在xml配置文件中配置创建对象,注入属性

<bean id="book" class="com.atguigu.spring5.Book">
	<property name="name" value="葵花宝典"></property>
    <property name="outhor" value="黄裳"></property>
</bean>
  1. 使用有参构造器注入属性

    创建类,定义属性和构造器

    public class Order{
        private String name;
        private String address;
        
        public Order(String name,String address){
            this.name = name;
            this.address = address;
        }
    }
    

    在xml配置文件中进行配置

    <bean id="order" class="com.atguigu.spring5.Order">
    	<constructor-arg name="name" value="铁剑"></constructor-arg>
        <constructor-arg name="address" value="华山派"></constructor-arg>
    </bean>
    
    1. 使用P名称空间注入(了解)

注入其他类型属性

  1. null值
<property name="address">
	<null/>
</property>
  1. 属性值包含特殊符号
输入《华山派》
1.使用转义符
2.使用<![CDATA[]]>
<property name="address">
	<value><![CDATA[《华山派》]]></value>
</property>

注入属性-外部Bean

往userService中注入userDaoImpl
<bean id="userService" class="com.atguigu.spring5.service.UserService">
	<property name="userDao" ref="userDaoImpl"></property>
</bean>
<bean id="userDaoImpl" class="com.atguigu.spring5.dao.UserDaoImpl"></bean>

注入属性-内部Bean

<bean id="userService" class="com.atguigu.spring5.service.UserService">
    <property name="" value=""></property>
    <property name="" value=""></property>
	
    <property name="userDaoImpl">
        <bean id="userDaoImpl" class="com.atguigu.spring5.dao.UserDaoImpl">
        	<property name="" value="" > </property>
        </bean>
    </property>
    
</bean>

属性注入-级联赋值

  1. 第一种写法
<!--级联赋值--> 貌似和注入外部Bean没有区别
<bean id="emp" class="com.atguigu.spring5.bean.Emp">
 	<!--设置两个普通属性-->
 	<property name="ename" value="lucy"></property>
 	<property name="gender" value="女"></property>
	 <!--级联赋值-->
 	<property name="dept" ref="dept"></property>
</bean> 
<bean id="dept" class="com.atguigu.spring5.bean.Dept">
	 	<property name="dname" value="财务部"></property>
</bean>
  1. 第二种写法
<!--级联赋值--> 
<bean id="emp" class="com.atguigu.spring5.bean.Emp">
 	<!--设置两个普通属性-->
 	<property name="ename" value="lucy"></property>
 	<property name="gender" value="女"></property>
 	<!--级联赋值-->
 	<property name="dept" ref="dept"></property>
 	<property name="dept.dname" value="技术部"></property>这条是新的,和注入外部Bean的方式不同,应该知识需要知道这个
</bean> 
<bean id="dept" class="com.atguigu.spring5.bean.Dept">
 	<property name="dname" value="财务部"></property>
</bean>

属性注入-数组&集合

<bean id="book" class="com.atguigu.spring5.Book">
	数组类型属性注入
    <property name="arr">
    	<array>
        	<value>value1</value>
            <value>value2</value>
        </array>
    </property>
    List类型属性注入
    <property name="list">
        <list>
        	<value>value1</value>
            <value>value2</value>
        </list>
    </property>
    Map类型属性注入
    <property id="maps">
    	<map>
        	<entry key="key1" value="value1"></entry>
            <entry key="key2" value="value2"></entry>
        </map>
    </property>
    Set类型属性注入
    <property id="sets">
        <value>value1</value>
        <value>value2</value>
    </property>
</bean>

在集合里设置对象类型的值

创建多个对象
<bean id="book1" class="com.atguigu.spring5.Book">
	<property name="name" value="葵花宝典"></property>
</bean>
<bean id="book2" class="com.atguigu.spring5.Book">
	<property name="name" value="独孤九剑"></property>
</bean>
把上面两个对象注入到List集合中(这个List集合也是其他Bean的属性,所以外面是property标签)
<property name="books">
	<list>
    	<ref bean="book1"></ref>
        <ref bean="book2"></ref>
    </list>
</property>

把集合注入进Bean里

  1. 在spring配置文件中引入名称空间util
<?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:p="http://www.springframework.org/schema/p"
	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/util http://www.springframework.org/schema/util/spring-util.xsd">
  1. 使用util标签完成list集合注入提取
<!--1 提取 list 集合类型属性注入--> 
<util:list id="names">
 	<value>易筋经</value>
 	<value>九阴真经</value>
 	<value>九阳神功</value>
</util:list>
<!--2 提取 list 集合类型属性注入使用--> 
<bean id="book" class="com.atguigu.spring5.Book">
 	<property name="nameList" ref="names"></property>
</bean>

IOC管理普通Bean和工厂Bean

  1. 普通Bean:在配置文件中定义的类型就是返回类型

  2. 工厂Bean(FactoryBean):配置与返回类型不同

    第一步:创建类,实现接口FactioryBean,成为工厂Bean

    第二布:实现接口里的方法,在实现方法中定义返回Bean的类型

    public class MyBean implements FactoryBean<Book> {
     	//定义返回 bean
     	@Override
     	public Book getObject() throws Exception {
     		Book book = new Book();
     		book.setName("独孤九剑");
     		return book;
     	}
     	@Override
     	public Class<?> getObjectType() {
     		return null;
     	}
     	@Override
     		public boolean isSingleton() {
     		return false;
     	} 
    }
    

    在xml配置文件中配置

    <bean id="myBean" class="com.atguigu.spring5.factorybean.MyBean">
    </bean>
    

    测试

    @Test
    public void test3() {
     	ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
     		Book book = context.getBean("myBean", Book.class);
     		System.out.println(book);
    }
    

    IOC操作Bean管理(bean作用域)

    1. 在Spring里,设置创建Bean实例是单实例还是多实例
    2. 默认是单实例对象
    image-20210703031334845
  3. 如何设置单实例还是多实例

    bean标签里有个属性 scope ,

    第一个值 :singleton,默认值,表示为单实例对象

    第二个值 :prototype 表示为多实例对象

区别:singleton:加载spring配置文件的时候就会创建单实例对象

​ prototype:调用 getBean的时候才会创建多实例对象

IOC操作Bean管理(bean生命周期)

  • 生命周期:从对象创建到对象销毁的过程

    1. 通过构造器创建bean实例(无参构造器)
    2. 为bean的属性设置值和对其他bean的引用(调用set方法)
    3. 调用bean的初始化方法(需要进行配置初始化的方法)
    4. bean可以使用了(对象获取到了)
    5. 当容器关闭时,调用bean的销毁的方法(需要进行配置销毁的方法)
  • Bean生命周期演示

    创建Bean

public class Orders {
 	//无参数构造
 	public Orders() {
 		System.out.println("第一步 执行无参数构造创建 bean 实例");
 	}
 	private String oname;
 	public void setOname(String oname) {
 		this.oname = oname;
 		System.out.println("第二步 调用 set 方法设置属性值");
 	}
 	//创建执行的初始化的方法
 	public void initMethod() {
 		System.out.println("第三步 执行初始化的方法");
 	}
 	//创建执行的销毁的方法
 	public void destroyMethod() {
 		System.out.println("第五步 执行销毁的方法");
 	} 
}

在xml文件中进行配置

<bean id="orders" class="com.atguigu.spring5.bean.Orders" init-method="initMethod" destroy-method="destroyMethod">
 <property name="oname" value="手机"></property>
</bean>

测试

@Test
 public void test1() {
	// ApplicationContext context =
	// new ClassPathXmlApplicationContext("bean1.xml");
 	//	ApplicationContext中没有close方法
     ClassPathXmlApplicationContext context =
 			new ClassPathXmlApplicationContext("bean1.xml");
 		Orders orders = context.getBean("orders", Orders.class);
 		System.out.println("第四步 获取创建 bean 实例对象");
 		System.out.println(orders);
 	//手动让 bean 实例销毁
 	context.close();
 }

测试输出结果:

第一步 执行无参数构造创建 bean 实例
第二步 调用 set 方法设置属性值
第三步 执行初始化的方法
第四步 获取创建 bean 实例对象
com.atguigu.spring5.bean.Orders@112@3243
第五步 执行销毁的方法
  • bean 的后置处理器,****bean 生命周期有七步

(1)通过构造器创建 bean 实例(无参数构造)

(2)为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)

(3)把 bean 实例传递 bean 后置处理器的方法 postProcessBeforeInitialization

(4)调用 bean 的初始化的方法(需要进行配置初始化的方法)

(5)把bean实例传递bean后置处理器的方法 postProcessAfterInitialization

(6)bean 可以使用了(对象获取到了)

(7)当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)

  • 演示添加后置处理器效果

    创建类,实现接口BeanPostProcessor,创建后置处理器

    public class MyBeanPost implements BeanPostProcessor {
     	@Override
     	public Object postProcessBeforeInitialization(Object bean, String beanName) 
    throws BeansException {
     		System.out.println("在初始化之前执行的方法");
     		return bean;
     	}
     	@Override
     	public Object postProcessAfterInitialization(Object bean, String beanName) 
    throws BeansException {
     		System.out.println("在初始化之后执行的方法");
     		return bean;
     	} 
    }
    

    配置后置处理器

    <bean id="myBeanPost" class="com.atguigu.spring5.bean.MyBeanPost"></bean>
    

    执行效果:

    第一步 执行无参数构造创建 bean 实例
    第二步 调用 set 方法设置属性值
    在初始化之前执行的方法
    第三步 执行初始化的方法
    在初始化之后执行的方法
    第四步 获取创建 bean 实例对象
    com.atguigu.spring5.bean.Orders@112@3243
    第五步 执行销毁的方法
    

IOC操作Bean管理(xml自动装配)

  • 什么是自动装配:根据指定装配规则(属性名称或属性类型),spring自动将匹配的属性值进行注入
  • 演示自动装配过程
    1. 根据属性名称自动注入
<!--实现自动装配
 	bean 标签属性 autowire,配置自动装配
 	autowire 属性常用两个值:
 	byName 根据属性名称注入 ,注入值 bean 的 id 值和类属性名称一样
 	byType 根据属性类型注入
-->
<bean id="emp" class="com.atguigu.spring5.autowire.Emp" autowire="byName">
 	<!--<property name="dept" ref="dept"></property>-->
</bean> 
<bean id="dept" class="com.atguigu.spring5.autowire.Dept"></bean>

​ 2.根据属性类型自动注入

<bean id="emp" class="com.atguigu.spring5.autowire.Emp" autowire="byType">
 	<!--<property name="dept" ref="dept"></property>-->
</bean> 
<bean id="dept" class="com.atguigu.spring5.autowire.Dept"></bean>

IOC操作Bean管理(外部属性文件)

1.直接配置数据库信息

​ (1)配置德鲁伊数据库连接池连接池

​ (2)引入德鲁伊数据库连接池依赖jar包:druid-1.1.9.jar

  • 配置数据库连接池的两种方式

方式一:直接配置数据库连接池

<!--直接配置连接池--> 
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
 	<property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
 	<property name="url" value="jdbc:mysql://localhost:3306/userDb"></property>
 	<property name="username" value="root"></property>
 	<property name="password" value="123456"></property>
</bean>

方式二:引入外部属性文件配置数据库连接池

(1)创建外部properties格式文件,写入数据库信息

prop.driverClass=com.mysql.jdbc.Driver
prop.url=jdbc:mysql://localhost:3306/userDb
prop.username=root
prop.password=123456

​ (2)把外部properties属性文件引入到spring配置文件中

​ 引入context名称空间

<beans xmlns="http://www.springframework.org/schema/beans" 
 	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
 	xmlns:p="http://www.springframework.org/schema/p" 
 	xmlns:util="http://www.springframework.org/schema/util" 
 	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/util 
http://www.springframework.org/schema/util/spring-util.xsd 
 
                        http://www.springframework.org/schema/context 
http://www.springframework.org/schema/context/spring-context.xsd">

在 spring 配置文件使用标签引入外部属性文件

<!--引入外部属性文件--> 
<context:property-placeholder location="classpath:jdbc.properties"/>
<!--配置连接池--> 
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
 	<property name="driverClassName" value="${prop.driverClass}"></property>
 	<property name="url" value="${prop.url}"></property>
 	<property name="username" value="${prop.userName}"></property>
 	<property name="password" value="${prop.password}"></property>
</bean>

IOC操作Bean管理(基于注解方式)

什么是注解

  • 什么是注解

    (1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值,属性名称=属性值...)

    (2)使用注解:注解作用在类上面,方法上面,属性上面

    (3)使用注解目的:简化xml配置

  • Spring针对Bean管理中创建对象提供的注解

    (1)@Component

    (2)@Service

    (3)@Controlle

    (4)@Repository

    四个注解功能一样,都可以用来创建Bean实例,但是有各自习惯用的地方

基于注解方式创建对象

第一步 引入依赖jar包

spring-aop-5.2.6.RELEASE.jar

第二步 开启组建扫描

<!--开启组件扫描
 1 如果扫描多个包,多个包使用逗号隔开
 2 扫描包上层目录
-->
<context:component-scan base-package="com.atguigu"></context:component-scan>

第三步 创建类,在类上面添加创建对象注解

//注解里面的value值可以省略不写,默认值为类名称,首字母小写(UserService-userService)
@Component(value = "userService") //<bean id="userService" class=".."/>
public class UserService {
 	public void add() {
 	System.out.println("service add.......");
 } 
}
  • 开启组件扫描的细节配置
<!--示例 1
 use-default-filters="false" 表示现在不使用默认 filter,自己配置 filter
 context:include-filter ,设置扫描哪些内容
-->
<context:component-scan base-package="com.atguigu" use-default-filters="false">
	<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
<!--示例 2
 下面配置扫描包所有内容
 context:exclude-filter: 设置哪些内容不进行扫描
-->
<context:component-scan base-package="com.atguigu">
 	<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

基于注解方式实现属性注入

(1)@AutoWired: 根据属性类型进行自动装配

​ 第一步: 把service和dao对象创建,在service和dao类添加创建对象注解

​ 第二部: 在service中注入dao对象,在service类添加dao类型属性,在属性上面使用注解

@Service
public class UserService {
 	//定义 dao 类型属性
 	//不需要添加 set 方法
 	//添加注入属性注解
 	@Autowired 
 	private UserDao userDao;
 	public void add() {
 		System.out.println("service add.......");
 		userDao.add();
 	} 
}

​ (2) @Qualifier: 根据名称进行注入

​ 这个Qualifier注解的使用,和上面@AutoWired一起使用

@Autowired //根据类型进行注入
@Qualifier(value = "userDaoImpl1") //根据名称进行注入
private UserDao userDao;

​ (3) @Resource: 可以根据类型注入,也可以根据名称注入

//@Resource //根据类型进行注入
@Resource(name = "userDaoImpl1") //根据名称进行注入
private UserDao userDao;

(4)@Value: 注入普通类型属性

@Value(value = "abc")
private String name;

完全注解开发

  1. 创建配置类,替代xml配置文件
@Configuration //作为配置类,替代 xml 配置文件
@ComponentScan(basePackages = {"com.atguigu"})//开启组件扫描
public class SpringConfig {
}
  1. 编写测试类
@Test
public void testAnnotation() {
 	//加载配置类
 	ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);
 	UserService userService = context.getBean("userService", UserService.class);
 	System.out.println(userService);
 	userService.add();
}

AOP

什么是AOP

  1. 面向切面编程(方面),利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各个部分之间的耦合度减低,提高程序的可重用性,同时提高了开发的效率。
  2. 通俗描述:不通过修改源代码的方式,在主干功能里添加新功能

使用登录例子说明AOP

image

AOP(底层原理)

AOP(JDK动态代理)

java.lang.reflect

Class Proxy

java.lang.Object

​ java.lang.reflect.Proxy

(1)调用newProxyInctance 方法

static Object newProxyInstance(ClassLoader,类<?>[] interfaces,invocationHandler h)

返回指定接口的代理类的实例,该接口将方法调用分派给指定的调用处理程序

方法有三个参数:

第一个参数:类加载器

第二个参数:增强方法所在的类,这个类实现的接口,支持多个接口

第三个参数:实现这个接口InvocationHandler,创建代理对象,写增强的部分

  1. 编写JDK动态代理代码

(1)创建接口,定义方法

public interface UserDao{
  	public int add(int a,intb);
  	public String update(String id);
}

(2)创建接口实现类,实现方法

public class UserDaoImpl implements UserDao{
    @Override
    public int add(int a,int b){
        return a + b;
    }
    @Override
    public String update(String id){
        return id;
    }
}

(3)使用Proxy类创建接口代理对象

public class JDKProxy{
    public static void main(String[] args){
        //创建接口实现类代理对象
        class[] interfaces = {UserDao.class};
        UserDaoImpl userDao = new UserDaoImpl();
        UserDao =          (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassHandler(),interfaces,newUserDaoProxy(userDao))
        int result = dao.add(1,2);
        System.out.println("result:" + result);
    }
}
//创建代理对象代码
class UserDaoProxy implements InvocationHandler{
    // 把创建的是谁的代理对象,把谁传递过来
    //有参数构造传递
    private Object obj;
    public UserDaoProxy(Object obj){
        this.obj = obj;
    }
    //增强的逻辑
    @Override
    public Object invoke(Object proxy,Method method,Object[] args)throws Throwable{
        //方法之前
        System.out.println("方法之前执行..." + method.getName() + ":传递的参数..." + Arrays.toString(args));
        //被增强的方法执行
        Object res = method.invoke(obj,args);
       //方法之后
        System.out.println("方法之后执行..." + obj);
        return res;
    }
}

AOP术语

  1. 连接点:类里面那些方法可以被增强,这些方法成为连接点

  2. 切入点:实际被真正增强的方法,成为切入点

  3. 通知(增强):

    (1)实际增强的逻辑部分成为通知(增强)

    (2)通知有多种类型

     	* 前置通知
     	* 后置通知
     	* 环绕通知
     	* 异常通知
     	* 最终通知
    
  4. 切面:是动作,把通知应用到切入点的过程

AOP操作(准备工作)

  1. Spring框架一般都是基于AspectJ实现AOP操作

    (1)AspectJ不是Spring组成部分,独立AOP框架,一般把AspectJ和Spring框架一起使用,进行AOP操作

  2. 基于AspectJ实现AOP操作

    (1)基于xml配置文件方式实现

    (2)基于注解方式实现(常用)

  3. 在项目里面引入AOP相关依赖

    • com.springsource.net.sf.cglib-2.2.0.jar
    • com.springsource.org.aoplliance-1.0.0.jar
    • com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
    • commons-logging-1.1.1.jar
    • druid-1.1.9.jar
    • spring-aop-5.2.6.RELEASE.jar
    • spring-aspects-5.2.6.RELEASE.jar
    • spring-beans-5.2.6.RELEASE.jar
    • spring-context-5.2.6.RELEASE.jar
    • spring-core-5.2.6.RELEASE.jar
    • spring-expression-5.2.6.RELEASE.jar
  4. 切入点表达式

    (1)切入点表达式作用:知道对哪个类里面的哪个方法进行增强

    (2)语法结构:execution( 权限修饰符 返回类型 类全路径 方法名称(参数列表) )
    ​ 举例 1:对 com.atguigu.dao.BookDao 类里面的 add 进行增强

execution(* com.atguigu.dao.BookDao.add(..))

​ 举例2:对com.atguigu.dao.BookDao类里面的所有方法进行增强

execution(* com.atguigu.dao.BookDao.*(..))

​ 举例3:对com.atguigu.dao包里面所有类,类里面所有方法进行增强

execution(* com.atguigu.dao.*.*(..))

AOP操作(AspectJ注解)

  1. 创建类,在类里面定义方法
public class User{
    public void add(){
        System.out.println("add...");
    }
}
  1. 创建增强类(编写增强逻辑)

    (1)在增强类里面,创建方法,让不同方法代表不同通知类型

    //增强的类
    public class UserProxy{
        public void before(){//前置通知
            System.out.println("before...");
        }
    }
    
  2. 进行通知的配置

    (1)在spring配置文件中,开启注解扫描

    <?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:aop="http://www.springframework.org/schema/aop" 
     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/aop 
    http://www.springframework.org/schema/aop/spring-aop.xsd">
     <!-- 开启注解扫描 -->
     <context:component-scan base-package="com.atguigu.spring5.aopanno"></context:component-scan>
    

    (2)使用注解创建User和UserProxye对象

    //被增强的类
    @Component
    public class User{
    //增强的类
    @Component
    public class UserProxy{
    

    (3)在增强的类上面添加注解@Aspect

    //增强的类
    @Component
    @Aspect //生成代理对象
    public class UserProxy{
    

    (4)在spring配置文件中开启生成代理对象

    <!-- 开启 Aspect 生成代理对象--> 
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
    
  3. 配置不同类型的通知

    (1)在增强的类里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置

    //增强的类
    @Component
    @Aspect //生成代理对象
    public class UserProxy{
        //前置通知
        //@Before注解表示作为前置通知
        @Before(value = "extution(* com.atguigu.sprint5.aopanno.User.add(..))")
        public void before(){
            System.out.println("before...");
        }
        //后置通知(返回通知)
        @AfterReturning(value = "extution(* com.atguigu.sprint5.aopanno.User.add(..))")
        public void afterReturning(){
            System.out.println("afterReturning...");
        }
        //最终通知
        @After(value = "extution(* com.atguigu.sprint5.aopanno.User.add(..))")
        public void after(){
            System.out.println("after...");
        }
        //异常通知
        @AfterThrowing(value = "extution(* com.atguigu.sprint5.aopanno.User.add(..))")
        public void throwing(){
            System.out.println("throwing...");
        }
        //环绕通知
        @Around(value = "extution(* com.atguigu.sprint5.aopanno.User.add(..))")
        public void around((ProceedingJoinPoint proceedingJoinPoint)throws Throwable {
            System.out.println("环绕之前...");
            //被增强的方法
            proceedingJoinPoint.proceed();
            System.out.println("环绕之前...");
        }
    }
    
  4. 相同切入点抽取

    //相同切入点抽取
    @Pointcut(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
    public void pointdemo() {
    }
    //前置通知
    //@Before 注解表示作为前置通知
    @Before(value = "pointdemo()")
    public void before() {
     	System.out.println("before.........");
    }
    
  5. 有多个增强类对同一个方法进行增强,设置增强类优先级

    (1)在增强类上面添加注解@Order(数字类型值),数字类型值越小,优先级越高

    @Component
    @Aspect
    @Order(1)
    public class PersonProxy{...}
    
  6. 使用完全注解开发

    (1)创建配置类,不需要创建xml配置文件

    @Configuration
    @ComponentScan(basePackages = {"com.atguigu"})
    @EnableAspectJAutoProxy(proxyTargetClass = true)
    public class ConfigAop{
        
    }
    

    AOP操作(AspectJ配置文件)

    1. 创建两个类,增强类和被增强类,创建方法
    2. 在spring配置文件中创建两个类对象
    <!--创建对象--> 
    <bean id="book" class="com.atguigu.spring5.aopxml.Book"></bean> 
    <bean id="bookProxy" class="com.atguigu.spring5.aopxml.BookProxy"></bean>
    
    1. 在spring配置文件中配置切入点
    <!--配置 aop 增强--> 
    <aop:config>
     	<!--切入点-->
     	<aop:pointcut id="p" expression="execution(* com.atguigu.spring5.aopxml.Book.buy(..))"/>
     	<!--配置切面-->
     	<aop:aspect ref="bookProxy">
     		<!--增强作用在具体的方法上-->
     		<aop:before method="before" pointcut-ref="p"/>
     	</aop:aspect>
    </aop:config>
    

JdbcTemplate

JdbcTemplate(概念和准备)

  1. 什么是JdbcTemplate

    Spring框架对JDBC进行封装,使用JdbcTemplate方便实现对数据库操作

  2. 准备工作:

    (1)引入相关jar包

    • druid-1.1.9.jar

    • mysql-connection-java-5.1.7-bin.jar

    • spring-jdbc-5.2.6.RELEASE.jar

    • spring-orm-5.2.6.RELEASE.jar

    • spring-tx-5.2.5.RELEASE.jar

    (2)在spring配置文件中配置数据库连接池

    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" destroy-method="close">
     	<property name="url" value="jdbc:mysql:///user_db" />
     	<property name="username" value="root" />
    	<property name="password" value="root" />
    	<property name="driverClassName" value="com.mysql.jdbc.Driver" />
    </bean>
    

    (3)配置JdbcTemplate对象,注入DataSource

    <!-- JdbcTemplate 对象 --> 
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
     	<!--注入 dataSource-->
     	<property name="dataSource" ref="dataSource"></property>
    </bean>
    

    (4)创建service类,创建dao类,在dao中注入JdbcTemplate对象

    *配置文件

    <!-- 组件扫描 --> 
    <context:component-scan base-package="com.atguigu"></context:component-scan>
    

    Service类:

    @Service
    public class BookService{
        //注入dao
        private BookDao bookDao;
    }
    

    Dao类:

    @Repository
    public class BookDaoImpl implements BookDao{
        //注入JdbcTemplate
        private JdbcTemplate jdbcTemplate;
    }
    

JdbcTemplate操作数据库(添加)

  1. 对应数据库创建实体类

    public class Book{
        private String BookId;
        private String BookName;
        private String BookStatus;
        //构造器
        ...
        //set方法
        ...
    }
    
  2. 编写service和dao

    (1)在dao进行数据库添加操作

    (2)调用JdbcTemplate对象里面的update方法实现添加操作

    update(String sql,Object... args)//参数1:sql语句,参数2:可变参数,设置sql语句值(填充占位符)
    

    dao类

    @Repository
    public class BookDaoImpl implements BookDao{
        //注入JdbcTemplate
        @AutoWried
        private JdbcTemplate jdbcTemplate;
        //添加的方法
        @Override
        public void add(Book book){
            //1 创建 sql 语句
     		String sql = "insert into t_book values(?,?,?)";
            //2 调用方法实现
    		Object[] args = {book.getUserId(), book.getUsername(), book.getUstatus()};
     		int update = jdbcTemplate.update(sql,args);
            System.out.println(update);
        }
    }
    

    service类:

    ...
    
  3. 测试类

    @Test
    public void testJdbcTemplate(){
        ApplicationContext context = 
            new ClassPathXmlApplicationContext("bean1.xml");
        BookService bookService = context.getBean("bookService",BookService.class);
        Book book = new Book("1","java","a");
        bookService.addBook(book);
    }
    

JdbcTemplate操作数据库(修改和删除)

  1. 修改
@Override
public void updateBook(Book book){
    String sql = "update t_book set username=?,ustatus=? where user_id=?";
 	Object[] args = {book.getBookId(), book.getBookName(),book.getBookStatus()};
 	int update = jdbcTemplate.update(sql, args);
 	System.out.println(update);
}
  1. 删除
@Override
public void deleteBook(String id){
    String sql = "delete from t_book where user_id=?";
 	int update = jdbcTemplate.update(sql, id);
 	System.out.println(update);
}

JdbcTemplate操作数据库(查询返回某个值)

  1. 查询表里有多少条记录,返回某个值
  2. 使用JdbcTemplate实现查询返回某个值代码
queryForObject(String sql,Class<T> requiredType)//参数1:sql语句,参数2:返回值类型Class
//查询表记录数
@Override
public int selectCount(){
    String sql = "select count(*) from t_book";
 	Integer count = jdbcTemplate.queryForObject(sql, Integer.class);
	return count;
}

JdbcTemplate操作数据库(查询返回对象)

  1. 场景:查询图书详情
  2. JdbcTemplate实现查询返回对象
//参数1:sql语句,参数2:RowMapper是接口,针对返回不同类型数据,使用这个接口里面实现类完成数据封装,参数3:sql语句值
queryForObject(String sql,RowMapper<T> rowMapper,Object...args)
//查询返回对象
@Override
public Book queryBook(String id){
    String sql = "select * from t_book where user_id=?";
    Book book = jdbcTemplate.queryForObject(sql,new BeanPropertyRowMapper<Book>(Book.class),id);
    return book;
}

JdbcTemplate操作数据库(查询返回集合)

  1. 场景:查询图书列表分页
  2. 调用JdbcTemplate方法实现查询返回集合
//参数1:sql语句,参数2:RowMapper是接口,针对返回不同类型数据,使用这个接口里面实现类完成数据封装,参数3:sql语句值
query(String sql,RowMapper<T> rowMapper,Object... args)
//查询返回集合
@Override
public List<Book> queryBookList(){
    String sql = "select * from t_book";
 	List<Book> bookList = jdbcTemplate.query(sql, new BeanPropertyRowMapper<Book>(Book.class));
 	return bookList;
}

JdbcTemplate操作数据库(批量操作)

批量操作:操作表里面的多条记录

  1. JdbcTemplate实现批量添加操作
//参数1:sql语句,参数2:List集合,添加多条数据记录
batchUpdate(String sql,List<Object[]> batchArgs)
//批量添加
@Override
public void batchAddBook(List<Object[]> batchArgs){
    String sql = "insert into t_book values(?,?,?)";
 	int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs);
 	System.out.println(Arrays.toString(ints));
}

批量添加测试:

//批量添加测试
List<Object[]> batchArgs = new ArrayList<>();
Object o1 = {"1","独孤九剑","绝版"};
Object o2 = {"2","葵花宝典","典藏"};
Object o3 = {"3","辟邪剑谱","发售中"};
batchArgs.add(o1);
batchArgs.add(o2);
batchArgs.add(o3);
//调用批量添加
bookService.batchAddBook(batchArgs);
  1. JdbcTemplate 实现批量修改操作
//批量修改
@Override
public void batchUpdateBook(List<Object[]> batchArgs){
    String sql = "update t_book set username=?,ustatus=? where user_id=?";
 	int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs);
 	System.out.println(Arrays.toString(ints));
}

批量修改测试:

List<Object[]> batchArgs = new ArrayList<>();
Object[] o1 = {"独孤九剑增强版","发售中","1"};
Object[] o2 = {"葵花宝典增强版","发售中","2"};
Object[] o3 = {"辟邪剑谱增强版","发售中","3"};
batchArgs.add(o1);
batchArgs.add(o2);
batchArgs.add(o3);
//调用方法实现批量修改
bookService.batchUpdate(batchArgs);
  1. JdbcTemplate实现批量删除操作
//批量删除
@Override
public void batchDeleteBook(List<Object[]> batchArgs) {
 	String sql = "delete from t_book where user_id=?";
 	int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs);
 	System.out.println(Arrays.toString(ints));
}

批量删除测试:

//批量删除
List<Object[]> batchArgs = new ArrayList<>();
	Object[] o1 = {"1"};
	Object[] o2 = {"2"};
	batchArgs.add(o1);
	batchArgs.add(o2);
	//调用方法实现批量删除
	bookService.batchDelete(batchArgs);

事务操作

事务概念

  1. 什么是事务

    (1)事务是数据库操作的最基本单元,逻辑上一组操作,要么都成功,如果有一个失败,所有操作都失败

  2. 事务四个特性(ACID)

    (1)原子性

    (2)一致性

    (3)隔离性

    (4)持久性

事务操作(Spring事务管理)

  1. 事务添加到JavaEE三层结构中的Service层(业务逻辑层)

  2. 在Spring进行事务管理操作

    • 有两种方式 :编程式事务管理和声明式事务管理(常用)
  3. 声明式事务管理

    (1)基于注解方式(常用)

    (2)基于xml配置文件方式

  4. 在Spring进行声明式事务管理,底层使用AOP原理

  5. Spring事务管理API

    (1) 提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类

事务操作(注解声明式事务管理)

  1. 在spring配置文件配置事务管理器
<!--创建事务管理器--> 
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
 	<!--注入数据源-->
 	<property name="dataSource" ref="dataSource"></property>
</bean>
  1. 在spring配置文件,开启事务注解

    (1)在spring配置文件引入空间名称tx

    <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:aop="http://www.springframework.org/schema/aop" 
     		xmlns:tx="http://www.springframework.org/schema/tx" 
     		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/aop 
    http://www.springframework.org/schema/aop/spring-aop.xsd
                         	http://www.springframework.org/schema/tx 
    http://www.springframework.org/schema/tx/spring-tx.xsd">
    

    (2)开启注解事务

    <!--开启事务注解--> 
    <tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>
    
  2. 在service类上面(或service类里面的方法上面)添加事务注解

    (1)@Transaction,这个注解添加到类上面,也可以添加在方法上面

    (2)如果把这个注解添加类上面,这个类里面所有的方法都添加事务

    (3)如果把这个注解添加方法上面,为这个方法添加事务

    @Service
    @Transaction
    public class UserService{...}
    

事务操作(声明式事务管理参数配置)

  1. 在service类上面添加注解@Transaction,在这个注解里面可以配置事务相关参数

    image

  2. propagation:事务传播行为

    (1)多事务方法直接进行调用,这个过程中,事务是如何进行管理的

    ​ 事务传播行为

    ​ 事务方法:对数据库表数据进行变化的操作

    public void update(){
        
    }
    @Transactional
    public void add(){
        //调用update()方法
        update();
    }
    

    REQUIRED:如果add方法本身有事务,调用update方法之后,update使用点给钱add方法里面的事务

    ​ 如果add方法本身没有事务,调用update方法之后,创建新事务

    REQUIERD_NEW: 使用add方法调用update方法,无论add方法是否有事务,都创建新的事务

    事务的传播行为可以由传播属性指定,Spring框架定义了7中传播行为

    传播属性 描述
    REQUIRED 如果由事务在运行,当前的方法就在这个事务内运行,否则,就启动一个新的事务,并在自己的事务内运行
    REQUIRED_NEW 当前的方法必须启动新事务,并在它自己的事务内运行,如果由事务正在运行,应该将他挂起
    SUPPORTS 如果有事务在运行,当前的方法就在这个事务内运行,否则它可以不运行在事务中
    NOT_SUPPORTE 当前的方法不应该运行在事务中,如果有运行的事务,将它挂起
    MANDATORY 当前的方法必须运行在事务内部,如果没有正在运行的事务,就抛出异常
    NEVER 当前的方法不应该运行在事务中,如果有运行的事务,就抛出异常
    NESTED 如果有事务在运行,当前的方法就应该在这个事务的嵌套事务内运行,否则,就启动一个新的事务,并在他自己的事务内运行
    @Service
    @Truansaction(propagation = Propagation.REQUIERD)
    public class UserService{...}
    
    1. ioslation: 事务隔离级别

      (1)事务的隔离性:多事务之间的操作不会产生影响。不考虑隔离性会产生很多问题

      (2)有三个读问题:脏读、不可重复读、幻读(虚读)

      • 脏读:一个未提交事务读取到另一个未提交事务的数据
      • 不可重复读:一个未提交事务读取到另一提交事务修改的数据
      • 幻读:一个未提交事务读取到另一提交事务添加的数据

      (3)解决:通过设置事务隔离级别,解决读问题

      脏读 不可重复读 幻读
      READ UNCOMMITTED(读未提交)
      READ COMMITTED(读已提交)
      REPEATABLE READ(可重复读)
      SERIALIZABLE(串行化)
      @Service
      @Transaction(propagation = Propagation.REQUIRED,isolation = Isolation.REPEATABLE_READ)
      public class UserService{...}
      
    2. timeout: 超时时间

      (1)事务需要在一定时间内进行提交,如果不提交进行回滚

      (2)默认值是 -1,设置时间以秒为单位进行计算

    3. readOnly: 是否只读

      (1)读:查询操作,写:添加修改删除操作

      (2)readOnly默认值是false,表示可以查询,也可以进行增删改操作

      (3)设置readOnly值为true,则只能查询

    4. rollbackFor: 回滚

      (1)设置出现哪些异常进行事务回滚

    5. noRollbackFor: 不回滚

      (1)设置出现哪些异常不进行事务回滚

事务操作(xml声明式事务管理)

spring 配置文件中进行配置

第一步 配置事务管理器

第二步 配置通知

第三步 配置切入点和切面

<!--1 创建事务管理器--> 
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
 	<!--注入数据源-->
 	<property name="dataSource" ref="dataSource"></property>
</bean>
<!--2 配置通知--> 
<tx:advice id="txadvice">
 	<!--配置事务参数-->
 	<tx:attributes>
 	<!--指定哪种规则的方法上面添加事务-->
 	<tx:method name="accountMoney" propagation="REQUIRED"/>
 	<!--<tx:method name="account*"/>-->
 	</tx:attributes>
</tx:advice>
<!--3 配置切入点和切面--> <aop:config>
 	<!--配置切入点-->
 	<aop:pointcut id="pt" expression="execution(* com.atguigu.spring5.service.UserService.*(..))"/>
 	<!--配置切面-->
 	<aop:advisor advice-ref="txadvice" pointcut-ref="pt"/>
</aop:config>

事务操作(完全注解声明式事务管理)

创建配置类,使用配置类替代 xml 配置文件

@Configuration //配置类
@ComponentScan(basePackages = "com.atguigu") //组件扫描
@EnableTransactionManagement //开启事务
public class TxConfig {
 	//创建数据库连接池
 	@Bean
 	public DruidDataSource getDruidDataSource() {
 		DruidDataSource dataSource = new DruidDataSource();
 		dataSource.setDriverClassName("com.mysql.jdbc.Driver");
 		dataSource.setUrl("jdbc:mysql:///user_db");
 		dataSource.setUsername("root");
 		dataSource.setPassword("root");
 		return dataSource;
 	}
 	//创建 JdbcTemplate 对象
 	@Bean
 	public JdbcTemplate getJdbcTemplate(DataSource dataSource) {
 		//到 ioc 容器中根据类型找到 dataSource
 		JdbcTemplate jdbcTemplate = new JdbcTemplate();
 		//注入 dataSource
 		jdbcTemplate.setDataSource(dataSource);
 		return jdbcTemplate;
 	}
 	//创建事务管理器
 	@Bean
 	public DataSourceTransactionManager 
				getDataSourceTransactionManager(DataSource dataSource) {
 		DataSourceTransactionManager transactionManager = new 
			DataSourceTransactionManager();
 		transactionManager.setDataSource(dataSource);
 		return transactionManager;
 	} 
}

Spring5框架新功能

以后补齐。。。

posted @ 2021-07-03 05:34  浮云墨迹  阅读(116)  评论(0)    收藏  举报