栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

Spring5学习笔记

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Spring5学习笔记

目录

一、Spring框架概述

二、IOC容器

1、什么是IOC

2、IOC底层

IOC过程:

IOC(接口)

3、IOC容器-Bean管理——基于xml的方式

1)什么是Bean管理

2)IOC操作Bean管理——基于xml方式创建对象

3)IOC操作Bean管理——基于xml方式注入属性

4)IOC操作Bean管理——xml注入其他类型属性

5)注入属性 ——外部bean

6)基于xml方式注入内部bean和级联赋值

7)xml方式注入集合属性

8)IOC操作Bean管理(FactoryBean)

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

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

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

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

4、IOC操作Bean管理——基于注解方式

1)什么是注解

2)Spring针对Bean管理中创建对象提供注解

3)基于注解方式创建Bean对象

4)开启组件扫描的细节配置

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

6)完全注解开发

三、AOP

1、AOP基本概念

2、AOP底层原理

3、AOP(JDK动态代理)

1)使用JDK动态代理,使用Proxy类里面的方法创建代理对象

2)编写JDK动态代理代码

4、AOP术语

5、AOP操作(准备)

6、基于AspectJ注解的AOP操作

7、基于AspectJ xml配置文件的AOP操作

四、 JdbcTemplate

1、JdbcTemplate概念及使用(准备工作)

1)什么是JdbcTemplate:

2)引入相关jar包

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

4)创建JdbcTemplate对象实例,注入DataSource

5)创建service类,创建dao类,在dao注入jdbcTemplate

2、JdbcTemplate操作数据库(添加)

1)对应数据库表对应的实体类

2)编写service和dao

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

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

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

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

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

8、JdbcTemplate实现批量修改操作

 五、事物管理

1、事物概念

2、事物操作(搭建事物操作的环境)

3、事务操作(Spring事务管理介绍)

 4、事务操作(基于注解的声明式事务管理)

1)在spring配置文件中配置事务管理器

2)在spring配置文件中开启事务注解

3)在service类上面(或者service类里面方法上面)添加事务注解

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

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

2)propagation:事务传播行为

3)ioslation:事务隔离级别

4)timeout:超时时间

5)readOnly:是否只读

6)rollbackFor:回滚

7)noRollbackFor:不回滚

6、事务操作(基于xml的声明式事务管理)

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

六、Spring5框架新功能

1)整个Spring5框架的代码基于Java8,运行时兼容JDK9,许多不建议使用的类和方法在代码库中删除

2)Spring5.0框架自带了通用的日志封装

3)Spring5框架核心容器支持@Nullable注解

4)Spring5核心容器支持函数式风格GenericApplicationContext

5)Spring5支持整合JUnit5

(1)整合JUnit4

(2)Spring5整合JUint5

6)Spring5框架新功能(Webflux)(暂时没写)

(1)SpringWebflux介绍

(2)响应式编程

(3)Webflux执行流程和核心API

(4)SpringWebflux(基于注解编程模型)

(5)SpringWebflux(基于函数式编程模型)


一、Spring框架概述

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

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

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

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

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

        4.Spring特点

                1.方便解耦,简化开发

                2.Aop编程支持

                3.方便程序测试

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

                5.方便进行事物操作

                6.降低API开发难度

        5.Spring下载地址

                https://repo.spring.io/ui/native/libs-milestone/org/springframework/spring


二、IOC容器

1、什么是IOC

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

(2)使用IOC目的:为了降低耦合度

2、IOC底层

xml解析、工厂模式、反射

IOC过程:
第一步 xml配置文件,配置创建的对象



第二步 有service类和dao类,创建工厂类

class UserFactory{

        public static UserDao getDao(){

                String classValue = class属性值; //1 xml解析

                Class clazz = Class.forName(classValue); //2 通过反射创建对象

                return (UserDao)clazz.newInstance();

        }

}

IOC(接口)

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

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

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

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

(2)ApplicationContext:BeanFactory接口的子接口,提供更大更强大的功能,一般由开发人员进行使用

        *加载配置文件时候就会创建配置文件中的对象

3、IOC容器-Bean管理——基于xml的方式

1)什么是Bean管理

Bean管理值得是两个操作,Spring创建对象和Spring注入属性,Bean管理操作有两种方式

2)IOC操作Bean管理——基于xml方式创建对象

        基于xml方式创建对象

(1)在Spring配置文件中,使用bean标签,标签里面添加对应属性,就可以实现对象创建

(2)在bean标签中有很多属性,id属性:唯一标识,class属性:类全路径(包类路径)

(3)创建对象时,默认使用无参构造器创建对象

3)IOC操作Bean管理——基于xml方式注入属性

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

(1)第一种注入方式:使用set方式进行注入

// (1)传统方式:创建类,定义属性和对应的set方法
public Class Book{
    //创建属性
    private String bname;
    private String bauthor;
    
    //创建属性对应的set方法
    public void setBname(String bname){
        this.bname = bname;
    }
    public void setBauthor(String bauthor){
        this.bauthor = bauthor;    
    }
}

    
    
    

(2)第二种注入方式:有参数构造进行注入

public class Order {

    private String id;
    private String address;
// 有参构造器
    public Order(String id, String address) {
        this.id = id;
        this.address = address;
    }
}


    
    

(3)p名称空间注入(了解)



    



4)IOC操作Bean管理——xml注入其他类型属性

        注入空值和特殊符号

    

        
        

        
            
        

        
            >]]>
        
    

5)注入属性 ——外部bean

(1)创建两个类 service 和 dao 类

public class UserService {
    private UserDao userDao;

    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }

    public void add(){
        System.out.println("service add...............");
        // 原始方式: 创建UserDao对象
//        UserDao userDao = new UserDaoImpl();
//        userDao.update();
        userDao.update();
    }
}
public class UserDaoImpl implements UserDao{
    @Override
    public void update() {
        System.out.println("dao update...........");
    }
}

(2)在spring配置文件中进行配置

    
    
        
        
    
    

6)基于xml方式注入内部bean和级联赋值

(1)注入属性--内部bean

        *一对多关系:部门和员工

        *一个部门有多个员工,一个员工属于一个部门

        *在实体类之间表示一对多关系,员工表示所属部门,使用对象类型属性进行设置

//部门类
public class Department {
    private String name;

    public void setName(String name) {
        this.name = name;
    }
}
//员工类
public class Employee {
    private String name;
    private String gender;
    private Department department;

    public void setName(String name) {
        this.name = name;
    }

    public void setGender(String gender) {
        this.gender = gender;
    }

    public void setDepartment(Department department) {
        this.department = department;
    }
}

        在xml文件中配置

    
    
        
        
        
        
        
            
                
            
        
    

(2)注入属性--级联赋值

    
    
    
        
        
        
        
        
    
    
        
    
    
    
    
        
        
        
        
        
    
    
        
    

7)xml方式注入集合属性

(1)注入数组类型属性

(2)注入List集合类型属性

(3)注入Map集合类型属性

//创建Student类,其中的属性包含 数组类型、list类型、map类型
public class Student {
    //数组类型属性
    private Integer[] scores;
    //list集合类型属性
    private List courses;
    //map集合类型属性
    private Map report;

    public void setScores(Integer[] scores) {
        this.scores = scores;
    }

    public void setCourses(List courses) {
        this.courses = courses;
    }

    public void setReport(Map report) {
        this.report = report;
    }
}
    
    
        
        
            
                90
                89
                96
            
        
        
        
            
                Java
                C++
                Python
            
        
        
        
            
                
                
                
            
        
    

(4)注入list集合,集合里面每一项都是对象类型

    //list集合中值是对象类型的类型属性
    private List passCourses;

    public void setPassCourses(List passCourses) {
        this.passCourses = passCourses;
    }
        
            
                
                
                
                
                
                
            
        
    
    
    
        
    
    
        
    
    
        
    
    
        
    
    
        
    
    
        
    

(5)使用util标签提取list集合


    
    
        朝花夕拾
        百年孤独
        资本论
    
    
    
        
    

8)IOC操作Bean管理(FactoryBean)

(1)spring有两种类型bean,一种是普通bean,另外一种是工厂bean(FactoryBean)

(2)普通bean:在配置文件中定义bean类型就是返回类型

(3)工厂bean:在配置文件中定义bean类型可以和返回类型不一样

//创建工厂Bean  MyBean 其中的泛型是 Course类
public class MyBean implements FactoryBean {
    
    @Override
    public Course getObject() throws Exception {
        return new Course();//底层用到反射,这里就简单的new一个对象
    }

    @Override
    public Class getObjectType() {
        return null;
    }

    @Override
    public boolean isSingleton() {
        return FactoryBean.super.isSingleton();
    }
}
    
    
    @Test
    public void testFactoryBean(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean3.xml");
        Course myBean = context.getBean("myBean", Course.class);
        System.out.println(myBean);
    }

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

(1)在Spring里面,设置创建bean实例是单实例还是多实例

(2)在Spring里面,默认情况下,bean是单实例对象

    @Test
    public void testCollection2(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean2.xml");
        Book book1 = context.getBean("book", Book.class);
        Book book2 = context.getBean("book", Book.class);
        System.out.println(book1);
        System.out.println(book2);
        System.out.println(book1==book2);
    }

(3)如何设置单实例还是多实例

a)在spring配置文件bean标签里面有属性(scope)用于设置单实例还是多实例

b)scope属性值

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

        第二个值 prototype,表示是多实例对象

    
    
        
    
    @Test
    public void testCollection2(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean2.xml");
        Book book1 = context.getBean("book", Book.class);
        Book book2 = context.getBean("book", Book.class);
        System.out.println(book1);
        System.out.println(book2);
        System.out.println(book1==book2);
    }

c)singleton和prototype区别

        第一 singleton单实例,prototype多实例

        第二 设置scope值是singleton时,加载spring配置文件时候就会创建单实例对象

                设置scope值是prototype时,不是在加载spring配置文件时候创建对象,在调用getBean方法时候创建多实例对象

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

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

(2)bean的生命周期

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

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

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

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

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

(3)演示bean生命周期

public class Order {
    private String itemName;

    //创建空参构造器
    public Order() {
        System.out.println("第一步 调用空参构造器方法");
    }

    //创建set方法
    public void setItemName(String itemName) {
        this.itemName = itemName;
        System.out.println("第二步 调用set方法");
    }

    //创建初始化方法
    public void initMethod(){
        System.out.println("第三步 调用默认初始化方法");
    }

    //创建销毁方法
    public void destroyMethod(){
        System.out.println("第五步 调用默认销毁方法");
    }

    @Override
    public String toString() {
        return "Order{" +
                "itemName='" + itemName + ''' +
                '}';
    }
}
    
        
    
    @Test
    public void testOrder(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml");
        Order order = context.getBean("order", Order.class);
        System.out.println("第四步 获得bean对象实例,使用bean对象实例");
        System.out.println(order);
        ((ClassPathXmlApplicationContext)context).close();
    }

(4)bean的后置处理器,bean生命周期共有七步

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

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

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

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

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

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

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

(5)演示添加后置处理器后,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;
    }
}

        在xml配置文件中配置bean后置处理器

    
    

        重新run Order的测试方法 

    @Test
    public void testOrder(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml");
        Order order = context.getBean("order", Order.class);
        System.out.println("第四步 获得bean对象实例,使用bean对象实例");
        System.out.println(order);
        ((ClassPathXmlApplicationContext)context).close();
    }

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

(1)什么是自动装配

        根据指定装配规则(属性名称或属性类型),spring自动将匹配的属性值进行注入

(2)演示自动装配过程

public class Employee {
    private String eName;
    private Department department;

    public void seteName(String eName) {
        this.eName = eName;
    }

    public void setDepartment(Department department) {
        this.department = department;
    }
}
    
    
        
    
    
    
        
    

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

 (1)直接配置数据库信息

    
    
        
        
        
        
    

(2)引入外部属性文件配置数据库连接池

a)创建外部属性文件,properties格式文件,写数据库信息

prop.driverClass=com.mysql.cj.jdbc.Driver
prop.url=jdbc:mysql://localhost:3306/test?serverTimezone=GMT
prop.username=root
prop.password=lyg1998

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

* 引入context名称空间


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

    
    
    
    
        
        
        
        
    

4、IOC操作Bean管理——基于注解方式

1)什么是注解

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

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

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

2)Spring针对Bean管理中创建对象提供注解

(1)@Component

(2)@Service

(3)@Controller

(4)@Repository

* 上面的四个注解功能是一样的,都可以用来创建bean实例

3)基于注解方式创建Bean对象

a)引入依赖 spring-aop-5.3.13.jar

b)开启组件扫描

    
    

c)创建类,在类上面添加创建对象注解

@Service(value="userService") //等同于在xml中写 
public class UserService {
    public void add(){
        System.out.println("service add.........");
    }
}

4)开启组件扫描的细节配置

        要看懂下面两个示例

    
    
        
    
    
    
        
    

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

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

//第一步,把service和dao对象创建,在service和dao类添加创建对象注解
@Repository
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("dao add......");
    }
}
//第二步,在service注入dao对象,在service类添加dao类型属性,在属性上面使用注解
@Service(value="userService") //等同于在xml中写 
public class UserService {
    
    //定义dao类型属性
    //不需要添加set方法
    //添加注入属性注解
    @Autowired //根据类型进行注入
    private UserDao userDao;

    public void add(){
        System.out.println("service add.........");
    }

}

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

@Qualifier注解的使用时,要和@Autowired一起使用

@Repository(value="userDaoImpl1") 
//把这个对象的实例取名为 userDaoImpl1 ,以便属性注入时,找到这个实例
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("dao add......");
    }
}
    @Autowired //根据类型进行注入
    @Qualifier(value="userDaoImpl1") //根据名称进行注入
    private UserDao userDao;

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

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

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

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

6)完全注解开发

(1)创建配置类,替代xml配置文件

@Configuration
@ComponentScan(basePackages = {"com.atguigu"})
public class SpringConfig {
}

(2)编写测试类

    //使用完全注解方式
    @Test
    public void testService1(){
        //加载配置类
        ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);
        UserService userService = context.getBean("userService", UserService.class);
        System.out.println(userService);
        userService.add();
    }

三、AOP

1、AOP基本概念

1)面向切面编程,利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

2)通俗描述,不通过修改源代码的方式,在主干功能里面添加新功能

3)如下使用登录的例子,作为说明

2、AOP底层原理

AOP底层使用动态代理,动态代理有两种情况:

第一种有接口情况,使用JDK动态代理,创建接口实现类的代理对象,增强类的方法

第二种没有接口情况,使用CGLIB动态代理,创建子类的代理对象,增强类的方法

3、AOP(JDK动态代理)

1)使用JDK动态代理,使用Proxy类里面的方法创建代理对象

java.lang.reflect.Proxy,调用 newInstance() 方法,方法有三个参数:

    public static Object newProxyInstance(ClassLoader loader,
                                          Class[] interfaces,
                                          InvocationHandler h)

ClassLoader loader        类加载器

Class[] interfaces        增强方法所在的类,这个类实现的接口,支持多个接口

InvocationHandler h        实现这个接口InvocationHandler,创建代理对象,写增强的部分

2)编写JDK动态代理代码
//(1)创建接口,定义方法
public interface UserDao {
    int add(int a,int b);
    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};
        
        UserDao userDao = (UserDao) Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(new UserDaoImpl()));
        System.out.println(userDao.add(1, 2));
    }
}

//创建代理对象代码
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() + "之前新增了一些功能");
        //执行原方法
        Object res = method.invoke(obj, args);
        //执行原方法之后,添加的事情(新增的功能)
        System.out.println(method.getName() + "之后新增了一些功能");

        return res;
    }

}

更加规范的写法,为多个被代理对象,创一个统一的代理类

静态代理与动态代理的区别

//动态代理类只能代理接口(不支持抽象类),代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类  

public class LogHandler implements InvocationHandler {

    // 目标对象  
    private Object targetObject;
    //绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。              
    public Object newProxyInstance(Object targetObject){
        this.targetObject=targetObject;
        //该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例    
        //第一个参数指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器  
        //第二个参数要实现和目标对象一样的接口,所以只需要拿到目标对象的实现接口  
        //第三个参数表明这些被拦截的方法在被拦截时需要执行哪个InvocationHandler的invoke方法  
        //根据传入的目标返回一个代理对象  
        return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
                targetObject.getClass().getInterfaces(),this);
    }
    @Override
    //关联的这个实现类的方法被调用时将被执行  
    
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        System.out.println("start-->>");
        for(int i=0;i>");

            //调用目标方法  
            ret=method.invoke(targetObject, args);
            
            System.out.println("success-->>");
        }catch(Exception e){
            e.printStackTrace();
            System.out.println("error-->>");
            throw e;
        }
        return ret;
    }

}  

4、AOP术语

1)连接点:类里面哪些方法可以被增强,这些方法称为连接点

2)切入点:实际被真正增强的方法称为切入点

3)通知(增强):实际增强的逻辑部分称为通知,且分为五种类型:前置通知、后置通知、环绕通知、异常通知、最终通知

4)切面:把通知应用到切入点过程

5、AOP操作(准备)

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

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

2)基于AspectJ实现AOP操作

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

(2)基于注解方式实现(使用更多)

3)引入依赖(相关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.*.*(..))

6、基于AspectJ注解的AOP操作
//(1) 创建被代理类
public class User {
    public void add(){
        System.out.println("add.......");
    }
}
// (2)创建增强类,里面写增强的方法,不同的方法代表不同的通知类型
public class UserProxy {
    public void before(){//前置通知
        System.out.println("before......");
    }
}
//(3)通过注解,创建 被代理类 和 增强类的对象实例
@Component
public class User {
    public void add(){
        System.out.println("add.......");
    }
}
@Component
@Aspect //生成代理对象
public class UserProxy {
    public void before(){//前置通知
        System.out.println("before......");
    }
}
//(4)开启注解扫描,有两种方式 一种是写一个注解配置类(如下),另外一种是在xml文件里配置
@Configuration
@ComponentScan(basePackages = {"com.atguigu.spring5.aopanno"})
@EnableAspectJAutoProxy
public class SpringConfig {
}



    
    

    
    

// (6)配置不同类型的通知
@Component
@Aspect //生成代理对象
public class UserProxy {

    //抽取相同的切入点
    @Pointcut(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
    public void pointDemo() {

    }

    //前置通知
    //@Before 表示前置通知,value值后面写 切入点表达式, 指明要给哪个包下的哪个类的哪个方法进行增强
    @Before(value = "pointDemo()")
    public void before() {
        System.out.println("before......");
    }

    //后置通知(返回通知)
    @AfterReturning(value = "pointDemo()")
    public void afterReturning() {
        System.out.println("afterReturning......");
    }

    //最终通知
    @After(value = "pointDemo()")
    public void after() {
        System.out.println("after......");
    }

    //异常通知
    @AfterThrowing(value = "pointDemo()")
    public void afterThrowing() {
        System.out.println("afterThrowing......");
    }

    //环绕通知
    @Around(value = "pointDemo()")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("before around......");
        //被增强执行的方法
        proceedingJoinPoint.proceed();
        System.out.println("after around......");
    }
}
// 测试
    //通过注解形式完成aop操作,通过xml配置文件配置spring
    @Test
    public void testAOPAnno1(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        User user = context.getBean("user", User.class);
        user.add();
    }

有多个增强类对同一个方法进行增强,设置增强类优先级

//在增强类上面添加注解 @Order(数字类型值),数字类型值越小优先级越高
@Component
@Aspect
@Order(1)
public class PersonProxy {
    @Before(value="execution(* com.atguigu.spring5.aopanno.User.add(..))")
    public void before(){
        System.out.println("Person before......");
    }
}
@Component
@Aspect //生成代理对象
@Order(2)
public class UserProxy {
    xxxx
}

7、基于AspectJ xml配置文件的AOP操作

1)创建两个类,增强类和被增强类,创建方法

public class Book {
    public void buy(){
        System.out.println("buy......");
    }
}
public class BookProxy {
    public void before(){
        System.out.println("before......");
    }
}

2)在Spring配置文件中创建两个类对象

    
    
    

3)在Spring配置文件中配置切入点

    
    
        
        
        
        
            
            
        
    

四、 JdbcTemplate

1、JdbcTemplate概念及使用(准备工作)

1)什么是JdbcTemplate:

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

2)引入相关jar包

        mysql驱动、druid数据库连接池、spring-jdbc、spring-orm、spring-tx

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

4)创建JdbcTemplate对象实例,注入DataSource
    
    
        
        
    

5)创建service类,创建dao类,在dao注入jdbcTemplate
    
    
@Repository
public class UserDaoImpl implements UserDao {

    //注入JdbcTemplate
    @Autowired
    private JdbcTemplate jdbcTemplate;
}
@Service
public class UserService {

    //注入dao
    @Autowired
    private UserDao userDao;
}

2、JdbcTemplate操作数据库(添加)

1)对应数据库表对应的实体类
public class User {
    private Integer userId;
    private String username;
    private String password;
    private String userStatus;

    public void setUsername(String username) {
        this.username = username;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public void setUserStatus(String userStatus) {
        this.userStatus = userStatus;
    }

    public void setUserId(Integer userId) {
        this.userId = userId;
    }

    public Integer getUserId() {
        return userId;
    }

    public String getUsername() {
        return username;
    }

    public String getPassword() {
        return password;
    }

    public String getUserStatus() {
        return userStatus;
    }
}

2)编写service和dao

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

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

@Service
public class UserService {

    //注入dao
    @Autowired
    private UserDao userDao;

    //添加用户
    public void addUser(User user){
        userDao.add(user);
    }
}
@Repository
public class UserDaoImpl implements UserDao {

    //注入JdbcTemplate
    @Autowired
    private JdbcTemplate jdbcTemplate;

    @Override
    public void add(User user) {
        String sql = "insert into `t_user`(`username`,`password`,`userStatus`) values(?,?,?)";
        Object[] args = {user.getUsername(),user.getPassword(),user.getUserStatus()};
        int update = jdbcTemplate.update(sql, args);
        System.out.println(update);
    }
}

(3)测试

public class TestUser {
    @Test
    public void testJdbcTemplate(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService = context.getBean("userService", UserService.class);
        User user = new User();
        user.setUsername("Tom");
        user.setPassword("tomtom123");
        user.setUserStatus("On_Vacation");
        userService.addUser(user);
    }
}

 3、JdbcTemplate操作数据库(修改和删除)
    @Override
    public void delete(String username) {
        String sql = "delete from `t_user` where `username` = ?";
        jdbcTemplate.update(sql, username);
    }

    @Override
    public void update(User user) {
        String sql = "update `t_user` set `password` = ? where `username` = ?";
        Object[] args = {user.getPassword(),user.getUsername()};
        jdbcTemplate.update(sql,args);
    }

4、JdbcTemplate操作数据库(查询返回某个值)
    @Override
    public int queryUserCount() {
        String sql = "select count(*) from `t_user`";
        return jdbcTemplate.queryForObject(sql,Integer.class);
    }

5、JdbcTemplate操作数据库(查询返回对象)
    @Override
    public User queryByUsername(String username) {
        String sql = "select `userId` userId,`username` username,`password` password,`userStatus` userStatus from `t_user` where `username` = ?";
        
        return jdbcTemplate.queryForObject(sql,new BeanPropertyRowMapper(User.class),username);
    }

6、JdbcTemplate操作数据库(查询返回集合)
    @Override
    public List queryByUserStatus(String userStatus) {
        String sql = "select `userId` userId,`username` username,`password` password,`userStatus` userStatus from `t_user` where `userStatus` = ?";
        
        return jdbcTemplate.query(sql,new BeanPropertyRowMapper(User.class),userStatus);

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

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

  // UserService里面  
  //批量添加
    public void batchAddUser(List users){
        List args = new ArrayList<>();
        for (User user : users) {
            args.add(new Object[]{user.getUsername(),user.getPassword(),user.getUserStatus()});
        }
        userDao.batchAdd(args);
    }
// UserDaoImpl里面
    @Override
    public void batchAdd(List args) {
        String sql = "insert into `t_user`(`username`,`password`,`userStatus`) values(?,?,?)";
        jdbcTemplate.batchUpdate(sql,args);
    }
  //测试
 //批量添加
    @Test
    public void testJdbcTemplate5(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService = context.getBean("userService", UserService.class);
        List users = new ArrayList<>();
        users.add(new User(null,"A","123456","BUSY"));
        users.add(new User(null,"B","123456","BUSY"));
        users.add(new User(null,"C","123456","On_Vacation"));
        users.add(new User(null,"D","123456","On_Vacation"));
        users.add(new User(null,"E","123456","On_Vacation"));
        userService.batchAddUser(users);
    }

8、JdbcTemplate实现批量修改操作
// UserService里面   
 //批量修改
    public void batchUpdateUser(List users){
        List args = new ArrayList<>();
        for (User user : users) {
            args.add(new Object[]{user.getPassword(),user.getUserStatus(),user.getUsername()});
        }
        userDao.batchUpdate(args);
    }



//UserDaoImpl里面
    @Override
    public void batchUpdate(List args) {
        String sql = "update `t_user` set `password` = ?, `userStatus` = ? where `username` = ?";
        jdbcTemplate.batchUpdate(sql,args);
    }



//测试类里面

    //批量修改
    @Test
    public void testJdbcTemplate6(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService = context.getBean("userService", UserService.class);
        List users = new ArrayList<>();
        users.add(new User(null,"A","111111","On_Vacation"));
        users.add(new User(null,"B","111111","On_Vacation"));
        users.add(new User(null,"C","111111","BUSY"));
        users.add(new User(null,"D","111111","BUSY"));
        users.add(new User(null,"E","111111","BUSY"));
        userService.batchUpdateUser(users);
    }


 五、事物管理

1、事物概念

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

2)典型例子:A给B转账100:  A  -100 ,B  +100

3)事物的四个特性(ACID): 原子性、一致性、隔离性、持久性

2、事物操作(搭建事物操作的环境)

1)创建数据库表,添加记录

2) 创建service,搭建dao,完成对象创建和注入关系

        service注入dao,在dao注入JdbcTemplate,在JdbcTemplate注入DataSource

    
    
    
    
        
        
        
        
    

    
    
        
        
    

    
    
@Repository
public class AccountDaoImpl implements AccountDao{
    @Autowired
    private JdbcTemplate jdbcTemplate;
}




@Service
public class AccountService {
    @Autowired
    private AccountDao accountDao;
}

3)在dao创建两个方法:多钱和少钱的方法,在service创建方法(转账的方法)

@Repository
public class AccountDaoImpl implements AccountDao{
    @Autowired
    private JdbcTemplate jdbcTemplate;

    @Override
    public void reduceBalance(String username,int money) {
        String sql = "update t_account set balance = balance - ? where username = ?";
        jdbcTemplate.update(sql,money,username);
    }

    @Override
    public void addBalance(String username,int money) {
        String sql = "update t_account set balance = balance + ? where username = ?";
        jdbcTemplate.update(sql,money,username);
    }
}



@Service
public class AccountService {
    @Autowired
    private AccountDao accountDao;

    public void transferAccount(String transferor,String transferee,int money){
        //转出
        accountDao.reduceBalance(transferor,money);
        //转入
        accountDao.addBalance(transferee,money);
    }
}

4)测试,如果上面的代码没有错误,则会正常完成转账操作

    @Test
    public void testTransfer(){
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        AccountService accountService = context.getBean("accountService", AccountService.class);
        accountService.transferAccount("Tom","Jerry",100);
    }


5)在 AccountService的  transfer中添加异常,则会导致事物出现部分执行的情况

    public void transferAccount(String transferor,String transferee,int money){
        //转出
        accountDao.reduceBalance(transferor,money);

        //手动制造异常
        int i = 12 / 0;

        //转入
        accountDao.addBalance(transferee,money);
    }

6)如何解决

    public void transferAccount(String transferor,String transferee,int money){
        try {
            //第一步 开启事务
        
            //第二步 进行业务操作
            //转出
            accountDao.reduceBalance(transferor,money);
            //手动制造异常
            int i = 12 / 0;
            //转入
            accountDao.addBalance(transferee,money);
            //第三步 没有发生异常,提交事务
        } catch (Exception e) {
            //第四步 出现异常,事务回滚
            e.printStackTrace();
        }
    }

3、事务操作(Spring事务管理介绍)

1)事务添加到JavaEE三层结构里面Service层(业务逻辑层)

2)在Spring进行事务管理操作有两种方式:编程式事务管理和声明式事务管理(使用)

3)声明式事务管理:

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

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

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

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

 4、事务操作(基于注解的声明式事务管理)

1)在spring配置文件中配置事务管理器
    
    
        
        
    

2)在spring配置文件中开启事务注解

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

(2)开启事务注解

    
    

3)在service类上面(或者service类里面方法上面)添加事务注解
@Service
@Transactional
public class AccountService {
    xxx
}

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

(2)添加到类上面,这个类里面所有方法都添加事务

(3)添加到方法上面,为这个方法添加事务

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

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

2)propagation:事务传播行为

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

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

spring中的事务7种传播行为

3)ioslation:事务隔离级别

spring种的事务隔离级别问题

(1)事务有特性称为隔离性,多事务操作之间不会产生影响,不考虑隔离性会产生很多问题

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

        脏读:一个未提交事务读取到另一个未提交事务的数据

        不可重复读:一个未提交事务读取到另一个提交事务修改的数据(修改前后的数据不一样,在同一个事务中读取同一个数据出现两次数据不一样)

        幻读:一个未提交事务读取到另一个提交事务添加的数据

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

@Transactional(propagation= Propagation.REQUIRED,isolation= Isolation.READ_COMMITTED,
timeout=10,readonly=true)
public class AccountService {
    xxx
}

4)timeout:超时时间

        事务需要在一定的时间内进行提交,如果不提交进行回滚,默认值是-1,设置时间以秒为单位进行计算

5)readOnly:是否只读

        readOnly默认值为false,表示可以查询,也可以添加修改删除操作,如果设置为true,表示只能查询。

6)rollbackFor:回滚

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

7)noRollbackFor:不回滚

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

6、事务操作(基于xml的声明式事务管理)

在spring配置文件中进行配置,步骤如下

(1)配置事务管理器

(2)配置通知

(3)配置切入点和切面




    
    

    
    
    
    
        
        
        
        
    

    
    
        
        
    

    
    
        
        
    

    
    
        
        
            
            
        
    

    
    
        
        
        
        
    

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

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

@Configuration  //配置类
@ComponentScan(basePackages = {"com.atguigu.spring5"})  //开启组件扫描
@EnableTransactionManagement    //开启事务
public class TxConfig {
    //创建数据库连接池
    @Bean
    public DruidDataSource getDruidDataSource(){
        DruidDataSource druidDataSource = null;
        try {
            Properties properties = new Properties();
            //读取jdbc.properties属性配置文件
            InputStream is = TxConfig.class.getClassLoader().getResourceAsStream("jdbc.properties");
            //从流中加载数据
            properties.load(is);
            //创建数据库连接池
            druidDataSource = (DruidDataSource) DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return druidDataSource;
    }

    //创建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 dataSourceTransactionManager = new DataSourceTransactionManager();
        dataSourceTransactionManager.setDataSource(dataSource);
        return dataSourceTransactionManager;
    }
}

六、Spring5框架新功能

1)整个Spring5框架的代码基于Java8,运行时兼容JDK9,许多不建议使用的类和方法在代码库中删除

2)Spring5.0框架自带了通用的日志封装

(1)Spring5已经移除Log4jConfigListener,官方建议使用Log4j2

(2)Spring5框架整合Log4j2

        a)引入jar包

        b)创建log4j2.xml配置文件





    
    
        
        
            
            
        
    
    
    
    
        
            
        
    

3)Spring5框架核心容器支持@Nullable注解

(1)@Nullable注解可以使用在方法、属性、参数上面,表示方法返回值、属性值、参数值可以为空

(2)注解用在方法上面,方法返回值可以为空

    @Nullable
    String getId();

(3)注解用在方法参数前面,方法参数可以为空

    public ClassPathXmlApplicationContext(String[] configLocations, @Nullable ApplicationContext parent) throws BeansException {
        this(configLocations, true, parent);
    }

(4)注解用在属性上面,属性可以为空

    @Nullable
    private Resource[] configResources;

4)Spring5核心容器支持函数式风格GenericApplicationContext
    //函数式风格创建对象,交给spring进行管理
    @Test
    public void testGenericApplicationContext(){
        //1 创建GenericApplicationContext对象
        GenericApplicationContext context = new GenericApplicationContext();
        //2 调用context的方法注册对象
        context.refresh();//清空
        context.registerBean("user",User.class,() -> new User());
        //3 获取在spring注册的对象
        User user = (User) context.getBean("user");
        System.out.println(user);
    }

5)Spring5支持整合JUnit5

(1)整合JUnit4

a)引入Spring中针对测试的相关依赖        spring-test-5.3.13

b)创建测试类,使用注解方式完成

@RunWith(SpringJUnit4ClassRunner.class) //单元测试框架
@ContextConfiguration("classpath:bean1.xml") //加载配置文件
public class JTest4 {

    @Autowired
    private AccountService accountService;

    @Test
    public void test1(){
        accountService.transferAccount("Tom","Jerry",100);
    }
}

(2)Spring5整合JUint5

a)引入JUint5的jar包

 b)创建测试类,使用注解方式完成

@ExtendWith(SpringExtension.class) //单元测试框架
@ContextConfiguration("classpath:bean1.xml") //加载配置文件
public class JTest5 {

    @Autowired
    private AccountService accountService;
    
    @Test
    public void test1(){
        accountService.transferAccount("Tom","Jerry",100);
    }
}

c)使用复合注解,简化代码

@SpringJUnitConfig(locations = {"classpath:bean1.xml"})
public class JTest5 {

    @Autowired
    private AccountService accountService;

    @Test
    public void test1(){
        accountService.transferAccount("Tom","Jerry",100);
    }
}

6)Spring5框架新功能(Webflux)(暂时没写)

(1)SpringWebflux介绍

        a)SpringWebflux是Spring5添加的新的模块,用于web开发,功能和SpringMVC类似,Webflux是使用了当前一种比较流行的响应式编程而出现的框架。

(2)响应式编程

(3)Webflux执行流程和核心API

(4)SpringWebflux(基于注解编程模型)

(5)SpringWebflux(基于函数式编程模型)

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/643789.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号