SSM+Maven高级+MybatisPlus万字笔记

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: SSM+Maven高级+MybatisPlus万字笔记

SSM框架学习

前言:花了一周学习了SSM(Spring+SpringMVC+Mybatis)以及Maven高级部分加MybatisPlus并且做了这个笔记,自己看黑马视频做的总结,不得不说学习还得多做笔记,做完笔记后还得在回头回忆一遍,遇到没记住的部分可以回来翻阅,很方便。仅上传作纪念,之前看JavaWeb也有记笔记,到时候有时间再放上来,仅供参考,有做的不好的地方还请多多指教!

一:Spring Framework系统框架

二:Spring核心概念

1.IoC、Bean、DI
1.1:IoC&Bean
  • IoC(Inversion of Control)控制反转
  • 使用对象时,由主动new产生对象转换为由外部提供对象,此过程中对象创建控制权转移到外部,此思想称为控制反转。
  • Spring提供了一个容器,称为IoC容器,用来充当IoC思想中的"外部"
  • IoC容器负责对象的创建、初始化等一系列工作,被创建的对象在IoC容器中被称为Bean
1.2:DI
  • DI(Dependency Injection)依赖注入
  • 在容器中建立Bean与Bean之间的依赖关系的整个过程,称为依赖注入。
1.3:最终效果
  • 充分解耦
  • 使用IoC容器管理Bean
  • 在IoC容器内将有依赖关系的Bean进行关系绑定(ID)
  • 使用对象时不仅可以直接从IoC容器中获取,并且获取到的Bean已经绑定了所有的依赖关系。
2.快速入门
2.1:导入相关坐标

<!--Spring框架-->

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-context</artifactId>
    <version>5.2.21.RELEASE</version>
</dependency>
2.2:定义Spring管理的类(接口)
public interface BookDao {
    public void save();
}
public class BookDaoImpl implements BookDao {
    public void save() {
        System.out.println("book dao save ...");
    }
}
public interface BookService {
    public void save();
}
public class BookServiceImpl implements BookService {
    //5.删除业务层中使用new的方式创建的dao对象
    private BookDao bookDao;
    public void save() {
        System.out.println("book service save ...");
        bookDao.save();
    }
    //6.提供对应的set方法(容器在执行)
    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
}
2.3:创建Spring配置文件,配置对应类作为Spring管理的Bean

applicationContext.xml

<!--1.导入spring坐标:spring-context-->
    <!--2.配置bean-->
    <!--id属性表示bean的名字
    class属性表示给bean定义类型
  name属性表示给bean取别名,可以当做id使用-->
    <bean id="bookDao" name="dao bookDaoImpl" class="com.my.dao.impl.BookDaoImpl"/>
    <bean id="bookService" class="com.my.service.impl.BookServiceImpl">
        <!--7.配置server与Dao的关系-->
        <!--name属性表示配置哪一个具体的属性,属性名称
        ref属性表示参照哪一个bean-->
        <property name="bookDao" ref="bookDao"/>
    </bean>
2.4:初始化IoC容器(Spring核心容器/Spring容器),通过容器获取Bean
public static void main(String[] args) {
    //3.加载配置文件得到上下文对象,也就是容器对象
    ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
    //4.获取资源
    // BookDao bookDao = (BookDao) ctx.getBean("bookDao");
    // bookDao.save();
    BookService bookService = (BookService) ctx.getBean("bookService");
    bookService.save();
}
3:bean的作用范围
  • scope属性用于改变bean的作用范围,默认情况下scope属性为singleton,此时为单例模式,即创建出来的bean都是同一个,当scope属性为prototype时候,即为非单例模式。
  • 为什么bean默认情况下为单例模式?可以用一个bean完成多个功能,提高复用性。
  • 适合交给容器进行管理的bean:
  • 表现层对象(servlet)
  • 业务层对象(service)
  • 数据层对象(Dao)
  • 工具类对象
  • 不适合交给容器进行管理的bean
  • 封装实体的域对象
4:实例化bean的四种方式
4.1:构造方法(常用)
  • 需要提供默认构造方法(无参构造方法)
  • 如果无参构造方法不存在,将抛出异常BeanCreationException
4.2:静态工厂(了解)
  • 静态工厂
public class OrderDaoFactory{
    public static OrderDao getOrderDao(){
        return new OrderDaoImpl();
    }
}
  • 配置
<bean
    id="orderDao"
    class="com.my.factory.OrderDaoFactory"
    factory-method="getOrderDao"
/>
4.3:实例工厂(了解)

4.4:FactoryBean(实用)
  • BookDaoFactoryBean.java
public class BookDaoFactoryBean implements FactoryBean<BookDao> {
    @Override
    public BookDao getObject() throws Exception {
        return new BookDaoImpl();
    }
    @Override
    public Class<?> getObjectType() {
        return BookDao.class;
    }
}
  • 配置信息
<!--FactoryBean方式-->
<bean id="bookDaoFac" class="com.my.factory.BookDaoFactoryBean"/>
4.5:Bean生命周期
  • 初始化容器
  1. 创建对象(内存分配)
  2. 执行构造方法
  3. 执行属性注入(set操作)
  4. 执行bean初始化方法
  • 使用Bean
  1. 执行业务操作
  • 关闭/销毁容器
  1. 执行bean销毁方法
  • bean生命周期控制
  • 配置
  • init-method
  • destroy-method
  • 接口(了解)
  • InitializingBean
  • DisposableBean
  • 关闭容器
  • ConfigurableApplicationContext
  • 手工关闭容器
    ConfigurableApplicationContext接口close()操作
  • 注册关闭钩子,在虚拟机退出前先关闭容器再退出虚拟机
    ConfigurableApplicationContext接口registerShutdownHook()操作

5:向类传递数据
5.1:依赖注入方式
  1. setter注入
  • 简单类型
  • 引用类型
  1. 构造器注入
  • 简单类型
  • 引用类型
5.2:setter注入
  1. 引用类型
  • 在bean中定义引用数据类型属性并提供可访问的set方法
private UserDao userDao;
public void setUserDao(UserDao userDao) {
    this.userDao = userDao;
}
  • 配置中使用property标签ref属性注入引用类型对象
<bean id="userDao" class="com.my.dao.impl.UserDaoImpl"/>
<bean id="bookService" class="com.my.service.impl.BookServiceImpl">
    <!--name属性表示配置哪一个具体的属性
    ref属性表示参照哪一个bean-->
    <!--注入引用数据类型-->
    <property name="userDao" ref="userDao"/>
</bean>
  1. 简单类型
  • 在bean中定义引用类型属性并提供可访问的set方法
    BookDaoImpl
public class BookDaoImpl implements BookDao {
    private int conNum;
    private String dbName;
    public void setConNum(int conNum) {
        this.conNum = conNum;
    }
    public void setDbName(String dbName) {
        this.dbName = dbName;
    }
    public void save() {
        System.out.println("book dao save ..."+conNum+" "+dbName);
    }
}
  • 配置中使用property标签value属性注入简单类型数据
<bean id="bookDao" class="com.my.dao.impl.BookDaoImpl">
    <!--注入简单数据类型-->
    <property name="conNum" value="10"/>
    <property name="dbName" value="mysql"/>
</bean>
5.3:构造器注入
  1. 引用类型
  • 在bean中定义引用数据类型属性并提供可访问的构造方法
private BookDao bookDao;
private UserDao userDao;
public BookServiceImpl(BookDao bookDao, UserDao userDao) {
    this.bookDao = bookDao;
    this.userDao = userDao;
}
  • 配置中使用constructor-arg标签ref属性注入引用类型对象
<bean id="bookService" class="com.my.service.impl.BookServiceImpl">
    <!--注入引用数据类型-->
    <constructor-arg name="bookDao" ref="bookDao"/>
    <constructor-arg name="userDao" ref="userDao"/>
</bean>
  1. 简单类型
  • 在bean中定义简单数据类型属性并提供可访问的构造方法
private int conNum;
private String dbName;
public BookDaoImpl(int conNum, String dbName) {
    this.conNum = conNum;
    this.dbName = dbName;
}
  • 配置中使用constructor-arg标签value属性注入简单类型对象
<bean id="bookDao" class="com.my.dao.impl.BookDaoImpl">
    <constructor-arg name="conNum" value="10"/>
    <constructor-arg name="dbName" value="mysql"/>
</bean>
  • 配置中使用constructor-arg标签type属性设置按照形参类型注入
  • 配置中使用constructor-arg标签index属性设置按照形参位置注入
5.4:依赖注入方式选择

5.5:自动类型的装配
  1. 使用bean标签autowire属性设置自动装配的类型(byType、byName(耦合度高,不推荐))
<bean id="bookDao" class="com.my.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.my.service.impl.BookServiceImpl" autowire="byType"/>
<bean id="bookService" class="com.my.service.impl.BookServiceImpl" autowire="byName"/>
  1. 依赖自动装配特征
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-y8PTeIUR-1653122017072)(SSM笔记.assets/image-20220515195109170.png)]
5.6:注入集合
public class BookDaoImpl implements BookDao {
    private int[] array;
    private List<String> list;
    private Set<String> set;
    private Map<String, String> map;
    private Properties properties;
    public void setArray(int[] array) {
        this.array = array;
    }
    public void setList(List<String> list) {
        this.list = list;
    }
    public void setSet(Set<String> set) {
        this.set = set;
    }
    public void setMap(Map<String, String> map) {
        this.map = map;
    }
    public void setProperties(Properties properties) {
        this.properties = properties;
    }
    public void save() {
        System.out.println(Arrays.toString(array));
        System.out.println(list);
        System.out.println(set);
        System.out.println(map);
        System.out.println(properties);
    }
}

``

<bean id="bookDao" class="com.my.dao.impl.BookDaoImpl">
    <property name="array">
        <array>
            <value>100</value>
            <value>200</value>
            <value>300</value>
        </array>
    </property>
    <property name="list">
        <list>
            <value>country</value>
            <value>aaa</value>
            <value>bbb</value>
        </list>
    </property>
    <property name="set">
        <set>
            <value>100</value>
            <value>aaa</value>
            <!--自动去重-->
            <value>aaa</value>
            <value>bbb</value>
        </set>
    </property>
    <property name="map">
        <map>
            <entry key="country" value="china"/>
            <entry key="province" value="guangdong"/>
            <entry key="city" value="shenzhen"/>
        </map>
    </property>
    <property name="properties">
        <props>
            <prop key="country">china</prop>
            <prop key="province">guangdong</prop>
            <prop key="city">guangzhou</prop>
        </props>
    </property>
</bean>
6:数据源对象管理
6.1:基本配置

6.1.1导入Druid坐标

<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid</artifactId>
    <version>1.1.16</version>
</dependency>

6.1.2配置数据源对象作为spring管理的bean

<bean  class="com.alibaba.druid.pool.DruidDataSource">
    <property name="driverClassName" value="com.mysql.cj.jdbc.Driver"/>
    <property name="url" value="jdbc:mysql://localhost:3306/spring_db"/>
    <property name="username" value="root"/>
    <property name="password" value="440983"/>
</bean>
6.2:加载properties文件

6.2.1.开启context名称空间

<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"  <!--更改1-->
       xsi:schemaLocation="
            http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans.xsd
            http://www.springframework.org/schema/context  <!--更改2-->
            http://www.springframework.org/schema/context/spring-context.xsd"> <!--更改3-->

6.2.2.使用context命名空间,加载指定properties文件

<context:property-placeholder location="jdbc.properties"/>

6.2.3.使用${}读取加载的属性值

<bean id="dataSource"  class="com.alibaba.druid.pool.DruidDataSource">
    <property name="driverClassName" value="${jdbc.driver}"/>
    <property name="url" value="${jdbc.url}"/>
    <property name="username" value="${jdbc.username}"/>
    <property name="password" value="${jdbc.password}"/>
</bean>

6.2.4.其他设置

6.2.5.加载配置文件

6.2.6.容器相关

6.2.7获取bean

6.3:注解开发bean

6.3.1使用@Componet定义bean

@Component("bookDao")
public class BookDaoImpl implements BookDao {
}
@Component
public class BookServiceImpl implements BookService {
}

6.3.2核心配置文件中通过组件扫描加载bean

<!--配置核心文件扫描bean-->
<context:component-scan base-package="com.my"/>

6.3.3 Spring提供@Component注解的三个衍生注解(与Component功能一样)

@Controller  //用于表现层bean定义
@Service  //用于业务层定义bean
@Repository  //用于数据层bean定义
7:纯注解开发:
7.1:基本配置

7.1.1:Spring3.0开启了纯注解开发模式,使用Java类替代配置文件,开启了Spring快速开发赛道

7.1.2:Java类代替Spring核心配置文件

com.my.config.SpringConfig.java

@Configuration   //用于设定当前类为配置类
// @ComponentScan("com.my")  
@ComponentScan({"com.my.service","com.my.dao"})  //用于设定扫描路径,只能添加一次,多个数据用数组格式
public class SpringConfig {
}

7.1.3:读取Spring核心配置文件初始化容器对象切换为读取Java配置类初始化容器对象

public static void main(String[] args) {
    ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
    //获取资源 按bean名称获取
    BookDao bookDao = (BookDao) ctx.getBean("bookDao");
    System.out.println(bookDao);
    // 按bean类型获取
    BookService bookService = ctx.getBean(BookService.class);
    System.out.println(bookService);
}

7.1.4:小结

7.2:作用范围及生命周期

7.3:依赖注入

7.3.1:注入引用类型

7.3.2:简单数据类型

@Value("${}")
private String name;

7.3.3:小结

7.4:第三方Bean的注入

7.5:XML配置VS注解配置

8:Spring整合Mybatis
8.1:jdbcConfig
public class jdbcConfig {
    @Value("${jdbc.driver}")
    private String driver;
    @Value("${jdbc.url}")
    private String url;
    @Value("${jdbc.username}")
    private String username;
    @Value("${jdbc.password}")
    private String password;
    @Bean
    public DataSource dataSource(){
        DruidDataSource ds = new DruidDataSource();
        ds.setDriverClassName(driver);
        ds.setUrl(url);
        ds.setUsername(username);
        ds.setPassword(password);
        return ds;
    }
8.2:mybatisConfig
public class mybatisConfig {
    //替换xml基本配置信息
    @Bean
    public SqlSessionFactoryBean sqlSessionFactory(DataSource dataSource){
        SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
        sqlSessionFactoryBean.setTypeAliasesPackage("com.my.pojo");
        sqlSessionFactoryBean.setDataSource(dataSource);
        return sqlSessionFactoryBean;
    }
    //替换
    /*<mappers>
        <package name="com.my.dao"/>
    </mappers>*/
    @Bean
    public MapperScannerConfigurer mapperScannerConfigurer(){
        MapperScannerConfigurer msc = new MapperScannerConfigurer();
        msc.setBasePackage("com.my.dao");
        return msc;
    }
}
8.3:SpringConfig
@Configuration
@ComponentScan("com.my")
@PropertySource("classpath:jdbc.properties")
@Import({jdbcConfig.class,mybatisConfig.class})
public class SpringConfig {
}
9:Spring整合Junit
9.1:导入坐标
<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.13</version>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-test</artifactId>
    <version>5.2.20.RELEASE</version>
</dependency>
9.2:创建junit测试类
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = SpringConfig.class)
public class AccountServiceTest {
    @Autowired
    private AccountService accountService;
    @Test
    public void testFindById(){
        Account ac = accountService.findById(1);
        System.out.println(ac);
    }
    @Test
    public void testFindAll(){
        for (Account account : accountService.findAll()) {
            System.out.println(account);
        }
    }
}
10:AOP(面向切面编程)
10.1:基本概念
  • AOP(Aspect Oriented Programming)面向切面编程,一种编程规范,指导开发者如何组织程序结构
  • OOP:面向对象编程
  • 作用:在不惊动原始设计的基础上为其进行功能增强
  • Spring概念:无侵入式/无入侵式
  • 连接点(JoinPoint):程序执行过程中的任意位置,粒度为执行方法、抛出异常、设置变量等
  • 在SpringAOP中,理解为方法的执行
  • 切入点(Pointcut):匹配连接点的式子
  • 在SpringAOP中,一个切入点可以描述一个具体方法,也可以匹配多个方法
  • 一个具体方法:com.my.dao包下的BookDao接口中的无形参无返回值的save()方法
  • 匹配多个方法:所有的save方法,所有以get开头的方法,所有以Dao结尾的接口中的任意方法,所有带有一个参数的方法。
  • 通知(Advice):在切入点处执行的操作,也就是共性功能
  • 在SpringAOP中,功能最终以方法的形式呈现
  • 通知类:定义通知的类
  • 切面(Aspect):描述通知与切入点的对应关系
10.2:入门案例

10.2.1:导入aop相关坐标

<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.9.4</version>
</dependency>

10.2.2:实现dao接口与实现类

@Repository
public class BookDaoImpl implements BookDao {
    public void save() {
        System.out.println(System.currentTimeMillis());
        System.out.println("book dao save ...");
    }
    @Override
    public void update() {
        System.out.println("book dao update...");
    }
}

10.2.3:定义通知类,制作通知

10.2.4:定义切入点

10.2.5:绑定切入点与通知关系,并指定通知添加到原始连接点的具体执行位置

10.2.6:定义通知类受Spring容器管理,并定义当前类为切面类

@Component   //6.1 定义通知类受Spring容器管理
@Aspect  //6.2 定义当前类为切面类
public class MyAdvice {
    //4.定义切入点
    @Pointcut("execution(void com.my.dao.BookDao.update())")
    public void pt(){}
    //3.定义通知类
    @Before("pt()")  //5.绑定切入点与通知关系,并指定通知添加到原始连接点的具体执行位置
    public void before(){
        System.out.println(System.currentTimeMillis());
    }
}

10.2.7:开启Spring对Aop注解驱动支持

@Configuration
@ComponentScan("com.my")
@EnableAspectJAutoProxy  //7.开启Spring对Aop注解驱动支持
public class SpringConfig {
}

10.2.8:其他代码

public class App {
    public static void main(String[] args) {
        ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
        BookDao bookDao = ctx.getBean(BookDao.class);
        bookDao.update();
    }
}
10.3:AOP执行流程

10.4:AOP切入点表达式

10.5:AOP通知

10.5.1:前置通知 @Before

10.5.2:后置通知 @After

10.5.3:环绕通知(重点)

●参数ProceedingJoinPoint参数必须放在第一位。

10.5.4:返回后通知(了解) @AfterReturning 原始切入点方法正常执行完毕后运行

10.5.5:抛出异常通知(了解) @AfterThrowing 抛出异常后执行,不抛异常不执行

10.6:执行效率案例
@Component
@Aspect
public class ProjectAdvice {
    @Pointcut("execution(* com.my.service.*Service.*(..))")
    private void servicePt(){}
    @Around("ProjectAdvice.servicePt()")
    public void runSpeed(ProceedingJoinPoint pjp) throws Throwable {
        //获取执行签名信息
        Signature signature = pjp.getSignature();
        //通过签名获取执行类型(接口名)
        String className = signature.getDeclaringTypeName();
        //通过签名获取执行操作名称(方法名)
        String methodName = signature.getName();
        long start = System.currentTimeMillis();
        for (int i = 0; i < 10000; i++) {
            Object ret = pjp.proceed();
        }
        long end = System.currentTimeMillis();
        System.out.println("执行"+className+"."+methodName+"万次耗时"+(end-start)+"ms");
    }
}
10.7:AOP通知获取数据

10.7.1:获取参数

10.7.2:获取返回值

11:Spring事务
11.1:开启事务步骤

11.2:事务角色

11.3:事务配置&传播行为

11.3.1:事务配置

11.3.2:事务传播

11.3.3:案例

  1. 在jdbcConfig中添加事务管理器
//设置事务管理器
@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource){
    DataSourceTransactionManager ptm = new DataSourceTransactionManager();
    ptm.setDataSource(dataSource);
    return ptm;
}
  1. 在SpringConfig中开启事务支持
@Configuration
@ComponentScan("com.my")
@PropertySource("classpath:jdbc.properties")
@Import({jdbcConfig.class,mybatisConfig.class})
@EnableAspectJAutoProxy   //开启Spring对Aop注解驱动的支持
@EnableTransactionManagement //开启事务支持
public class SpringConfig {
}
  1. Dao层操作
//数据层操作
public interface AccountDao {
    @Update("update tbl_account set money = money + #{money} where name = #{name}")
    void inMoney(@Param("name")String name,@Param("money")Double money);
    @Update("update tbl_account set money = money - #{money} where name = #{name}")
    void outMoney(@Param("name") String name,@Param("money") Double money);
}
public interface LogDao {
    @Insert("insert into tbl_log(info,createDate) values(#{info},now())")
    void log(String info);
}
  1. Service层
public interface AccountService {
    @Transactional
    public void transfer(String out,String in,Double money);
}
public interface LogService {
    //propagation设置事务属性:传播行为设置为当前操作需要新事务
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    void log(String out,String in,Double money);
}
@Service
public class AccountServiceImpl implements AccountService {
    @Autowired
    private AccountDao accountDao;
    @Autowired
    private LogService logService;
    @Override
    public void transfer(String out, String in, Double money) {
        try{
            accountDao.outMoney(out,money);
            int a = 1/0;
            accountDao.inMoney(in,money);
        }finally {
            logService.log(out,in,money);
        }
    }
}
@Service
public class LogServiceImpl implements LogService {
    @Autowired
    private LogDao logDao;
    @Override
    public void log(String out, String in, Double money) {
        logDao.log(out+"向"+in+"转账"+money+"钱");
    }
}
  1. Test
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = SpringConfig.class)
public class AccountServiceTest {
    @Autowired
    AccountService accountService;
    @Test
    public void transferTest(){
        accountService.transfer("Tom","Jerry",100D);
    }
}

三:SpringMVC

1:概述
  • SpringMVC是一种基于Java实现MVC模型的轻量级Web框架
  • 优点
  • 使用简单,开发便捷(相比于Servlet)
  • 灵活性强
2:快速入门

2.1:导入坐标

<dependencies>
    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>javax.servlet-api</artifactId>
        <version>3.1.0</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>5.2.10.RELEASE</version>
    </dependency>
</dependencies>

2.2:初始化SpringMVC环境

@Configuration
@ComponentScan("com.my")   //设定SpringMVC加载对应bean
public class SpringMVCConfig {
}

2.3:创建SpringMVC控制器类(等同于Servlet功能)

@Controller
public class UserController {
    @RequestMapping("/save")  //设置当前控制器方法请求访问路径
    @ResponseBody   //设置当前控制器方法响应内容为当前返回值,无需解析
    public String save(){
        System.out.println("user save...");
        return "{'info':'springmvc'}";
    }
}

2.4:初始化Servlet容器,加载SpringMVC环境,并设置SpringMVC请求拦截的路径

//AbstractDispatcherServletInitializer类是SpringMVC提供的快速初始化Web3.0容器的抽象类
public class ServletContainersInitConfig extends AbstractDispatcherServletInitializer {
    //初始化容器并注册返回容器到Tomcat中,以便可以访问bean
    @Override
    protected WebApplicationContext createServletApplicationContext() {
        AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
        ctx.register(SpringMVCConfig.class);
        return ctx;
    }
    //设定SpringMVC对应的请求映射路径,设置为/表示拦截所有请求,任意请求都将转入到SpringMVC进行处理
    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
    //如果创建Servlet容器时需要加载非SpringMVC对应的bean,使用当前方法进行,使用方式同createServletApplicationContext()
    @Override
    protected WebApplicationContext createRootApplicationContext() {
        return null;
    }
}
3:Controller加载控制与业务Bean加载控制

4:URL传递参数
  • 正常情况下名称对名称
  • 普通参数 save(String name,int age)
  • 封装对象 save(User user)
  • 数组 save(String[] array)
    参数设置 array 10、array 20…
  • 名称对不上用@RequestParam
  • 集合 save(@RequestParam List list)
    参数设置 list 10、list 20…
5:JSON数据传递参数

5.1:添加JSON数据转换坐标

<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.12.6.1</version>
</dependency>

5.2:在body中设置JSON数据并发送

5.3:开启自动转换JSON数据的支持

@Configuration
@ComponentScan("com.my")
@EnableWebMvc    //开启自动转换JSON数据的支持,开启SpringMVC多项辅助功能
public class SpringMVCConfig {
}

5.4:设置接收JSON数据

@Controller
public class UserController {
    @ResponseBody  //将请求中请求体所包含的数据传递给请求参数,此注解一个方法只能使用一次
    @RequestMapping("/listParamForJson")
    public String listParamForJson(@RequestBody List<String> likes){
        System.out.println("list common(json)参数传递==>" + likes);
        return "{'info':'springmvc'}";
    }

6:日期类型参数传递
  • dataParam(Date date, //2020/2/12
    @DateTimeFormat(pattern=“yyyy-MM-dd”) Date date1, //2020-5-17
    @DateTimeFormat(pattern=“yyyy/MM/dd HH:mm:ss”) Date date2 //2020/02/20 20:20:20
    )
7:响应
//响应页面/跳转页面
@RequestMapping("/toJumpPage")
public String toJumpPage(){
    System.out.println("跳转页面");
    return "page.jsp";
}
//响应纯文本数据
@RequestMapping("/toText")
@ResponseBody
public String toText(){
    System.out.println("返回纯文本数据");
    return "response text";
}
//响应POJO对象
@RequestMapping("/toJsonPOJO")
@ResponseBody
public User toJsonPOJO(){
    System.out.println("返回json对象数据");
    User user = new User();
    user.setName("Tom");
    user.setAge(20);
    return user;
}
//响应POJO集合对象
@RequestMapping("/toJsonList")
@ResponseBody
public List<User> toJsonList(){
    System.out.println("返回JSON集合数据");
    User user1 = new User();
    user1.setName("Tom");
    user1.setAge(15);
    User user2 = new User();
    user2.setName("Jerry");
    user2.setAge(18);
    User user3 = new User();
    user3.setName("Jack");
    user3.setAge(20);
    List<User> users = new ArrayList<>();
    users.add(user1);
    users.add(user2);
    users.add(user3);
    return users;
}
8:REST

8.1:概述

8.2:REST风格案例

// REST/POST
@RequestMapping(value = "/users", method = RequestMethod.POST)
@ResponseBody
public String save() {
    System.out.println("user save...");
    return "{'module':'user save'}";
}
@RequestMapping(value = "/users/{id}",method = RequestMethod.DELETE)
@ResponseBody
public String delete(@PathVariable Integer id) {
    System.out.println("user delete..."+id);
    return "{'module':'user delete'}";
}
@RequestMapping(value = "/users",method = RequestMethod.PUT)
@ResponseBody
public String update(@RequestBody User user){
    System.out.println("user update..."+user);
    return "{'module':'user update'}";
}
//查询
@RequestMapping(value = "/users/{id}",method = RequestMethod.GET)
@ResponseBody
public String getById(@PathVariable Integer id){
    System.out.println("user getById..."+id);
    return "{'module':'user getById'}";
}

8.3:简化

/**
 * 简化REST
 */
@RestController("/book")
public class BookController {
    // REST/POST
    @PostMapping
    public String save() {
        System.out.println("book save...");
        return "{'module':'book save'}";
    }
    @DeleteMapping("/{id}")
    public String delete(@PathVariable Integer id) {
        System.out.println("book delete..."+id);
        return "{'module':'book delete'}";
    }
    @PutMapping
    public String update(@RequestBody Book book){
        System.out.println("book update..."+book);
        return "{'module':'book update'}";
    }
    //查询
    @GetMapping("/{id}")
    public String getById(@PathVariable Integer id){
        System.out.println("book getById..."+id);
        return "{'module':'book getById'}";
    }
    @GetMapping
    public String getAll(){
        System.out.println("book getAll...");
        return "{'module':'book getAll'}";
    }
}

8.4:设置对静态资源的放行

public class SpringMvcSupport extends WebMvcConfigurationSupport {
    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        //当访问/pages/???时候,走pages目录下的内容,防止被SpringMVC拦截
        registry.addResourceHandler("/pages/**").addResourceLocations("/pages/");
        registry.addResourceHandler("/js/**").addResourceLocations("/js/");
        registry.addResourceHandler("/css/**").addResourceLocations("/css/");
        registry.addResourceHandler("/plugins/**").addResourceLocations("/plugins/");
    }
}
9:表现层返回结果规范化处理

9.1:在controller定义Result类

public class Result {
    private Object data;   //返回数据
    private Integer code;  //返回结果代码
    private String msg;   //消息

9.2:定义返回代码

com.my.controller.Code.java

public class Code {
    public static final Integer SAVE_OK = 20011;
    public static final Integer DELETE_OK = 20021;
    public static final Integer UPDATE_OK = 20031;
    public static final Integer GET_OK = 20041;
    public static final Integer SAVE_ERR = 20010;
    public static final Integer DELETE_ERR = 20020;
    public static final Integer UPDATE_ERR = 20030;
    public static final Integer GET_ERR = 20040;
}

9.3:修改表现层返回类型

@GetMapping
public Result getAll() {
    List<Book> bookList = bookService.getAll();
    Integer code = (bookList == null ? Code.GET_ERR:Code.GET_OK);
    String msg = (bookList == null? "数据查询失败!":"");
    return new Result(code,bookList,msg);
}
10:异常处理

10.1:集中处理

com.my.controller.ProjectExceptionAdvice.java

@RestControllerAdvice   //将类声明为异常处理类
public class ProjectExceptionAdvice {
    @ExceptionHandler(Exception.class)  //定义处理哪一种异常
    public Result doException(Exception ex){
        return new Result(11111,null,"捕捉到异常!");
    }
}

10.2:异常分类

  • 业务异常(BusinessException)
  • 规范的用户行为产生的异常
  • 不规范的用户行为操作产生的异常
  • 系统异常(SystemException)
  • 项目运行过程中可预计且无法避免的异常
  • 其他异常(Exception)
  • 编程人员未预期到的异常

10.3:异常分类处理

10.3.1:定义Result返回体

com.my.controller.Result.java

public class Result {
    private Object data;   //返回数据
    private Integer code;  //返回结果代码
    private String msg;   //消息

10.3.2:定义异常编码

package com.my.controller;
public class Code {
    public static final Integer SAVE_OK = 20011;
    public static final Integer DELETE_OK = 20021;
    public static final Integer UPDATE_OK = 20031;
    public static final Integer GET_OK = 20041;
    public static final Integer SAVE_ERR = 20010;
    public static final Integer DELETE_ERR = 20020;
    public static final Integer UPDATE_ERR = 20030;
    public static final Integer GET_ERR = 20040;
    public static final Integer SYSTEM_ERR = 50001;
    public static final Integer SYSTEM_TIMEOUT_ERR = 50002;
    public static final Integer SYSTEM_UNKNOWN_ERR = 59999;
    public static final Integer BUSINESS_ERR = 60002;
}

10.3.3:定义异常类

package com.my.exception;
public class BusinessException extends RuntimeException{
    private Integer code;
    public Integer getCode() {
        return code;
    }
    public void setCode(Integer code) {
        this.code = code;
    }
    public BusinessException(Integer code, String message) {
        super(message);
        this.code = code;
    }
    public BusinessException(Integer code, String message, Throwable cause) {
        super(message, cause);
        this.code = code;
    }
}
package com.my.exception;
public class SystemException extends RuntimeException{
    private Integer code;
    public Integer getCode() {
        return code;
    }
    public void setCode(Integer code) {
        this.code = code;
    }
    public SystemException(Integer code, String message) {
        super(message);
        this.code = code;
    }
    public SystemException(Integer code, String message, Throwable cause) {
        super(message, cause);
        this.code = code;
    }
}

10.3.4:定义异常处理类

package com.my.controller;
import com.my.exception.BusinessException;
import com.my.exception.SystemException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
/**
 * 处理异常
 */
@RestControllerAdvice   //将类声明为异常处理类
public class ProjectExceptionAdvice {
    @ExceptionHandler(SystemException.class)  //定义处理哪一种异常
    public Result doSystemException(SystemException ex){
        //记录日志
        //发送消息给运维
        //发送邮件给开发人员,ex对象发送给开发人员
        return new Result(ex.getCode(),null,ex.getMessage());
    }
    @ExceptionHandler(BusinessException.class)  //定义处理哪一种异常
    public Result doBusinessException(BusinessException ex){
        return new Result(ex.getCode(),null,ex.getMessage());
    }
    @ExceptionHandler(Exception.class)  //定义处理哪一种异常
    public Result doException(Exception ex){
        //记录日志
        //发送消息给运维
        //发送邮件给开发人员,ex对象发送给开发人员
        return new Result(Code.SYSTEM_UNKNOWN_ERR,null,"系统繁忙,请稍后再试!");
    }
}

10.3.5:捕捉异常

com.my.controller.BookController.java

@GetMapping(value = ("/getById"))
public Result getById(@RequestParam("id") Integer id) {
    if(id == 0){
        throw new BusinessException(Code.BUSINESS_ERR,"请规范您的操作!");
    }
    //将可能出现的异常进行包装,转换成自定义异常
    try{
        int i = 1/0;
    }catch (Exception e){
        throw new SystemException(Code.SYSTEM_TIMEOUT_ERR,"服务器访问超时,请稍后再试!",e);
    }
    Book book = bookService.getById(id);
    Integer code = (book == null? Code.GET_ERR:Code.GET_OK);
    String msg = (book == null? "数据查询失败!":"");
    return new Result(code,book,msg);
}
11:拦截器

11.1:声明拦截器的bean,并实现HandlerInterceptor接口(注意:扫描加载bean)

package com.my.controller.interceptor;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@Component
public class ProjectInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("preHandle...");
        return true;
    }
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("postHandle...");
    }
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("afterCompletion...");
    }
}

11.2:定义配置类,继承WebMvcConfigurationSupport,实现addInterceptor方法(注意:扫描加载配置)

package com.my.config;

@Configuration
public class SpringMvcSupport extends WebMvcConfigurationSupport {
    //设置过滤静态访问资源
    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("/pages/**").addResourceLocations("/pages/");
        registry.addResourceHandler("/css/**").addResourceLocations("/css/");
        registry.addResourceHandler("/js/**").addResourceLocations("/js/");
        registry.addResourceHandler("/plugins/**").addResourceLocations("/plugins/");
    }
    //自动装配拦截器
    @Autowired
    private ProjectInterceptor projectInterceptor;
    //添加拦截器
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(projectInterceptor).addPathPatterns("/books","/books/*");  //设置具体拦截路径,可设置多个
    }
}

11.3:添加扫描路径

@Configuration
@ComponentScan({"com.my.controller","com.my.config"})
@EnableWebMvc
public class SpringMvcConfig {
}

11.4:简化(注意:侵入式较强)

com.my.config.SpringMvcConfig.java

@Configuration
@ComponentScan({"com.my.controller"})
@EnableWebMvc
public class SpringMvcConfig implements WebMvcConfigurer {
    //自动装配拦截器
    @Autowired
    private ProjectInterceptor projectInterceptor;
    //添加拦截器
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(projectInterceptor).addPathPatterns("/books","/books/*");  //设置具体拦截路径
    }
}

11.5:执行流程

11.6:拦截器参数

11.7:拦截器链

四:Maven高级

1:分模块开发

1.1:创建Maven模块

1.2:书写模块代码

1.3:通过Maven指令安装模块到本地仓库(install指令)

2:可选依赖于排除依赖

2.1:可选依赖

true

指对外隐藏当前所依赖的资源----不透明

2.2:排除依赖

指主动断开依赖的资源,被排除的资源无需指定版本-------不需要

3:多环境配置与应用

3.1:多环境

3.2:跳过测试

4:yml文件读取

五:SpringBoot&MybatisPlus

1:快速入门

1.1:导入坐标

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.0</version>
    </parent>
    <groupId>com.my</groupId>
    <artifactId>MybatisPlus-demo1</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>MybatisPlus-demo1</name>
    <description>MybatisPlus-demo1</description>
    <properties>
        <java.version>11</java.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
            <scope>provided</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.1.16</version>
        </dependency>
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.5.0</version>
        </dependency>
    </dependencies>
    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <excludes>
                        <exclude>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                        </exclude>
                    </excludes>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

1.2:创建配置文件(注意空格)

application.yml

spring:
  datasource:
    type: com.alibaba.druid.pool.DruidDataSource
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/ssm_db
    username: root
    password: 440983

1.3:创建Book及BookDao

@Data    //添加Lombok注解
public class Book {
    private int id;
    private String type;
    private String name;
    private String description;
}
@Mapper
public interface BookDao extends BaseMapper<Book> {
}
  • 说明:@Data注解为当前实体类在编译期设置对应的get/set方法,toString方法、hashCode方法、equals方法等
  • 要注意数据库中表的名称要为book

1.4:测试

@SpringBootTest
class ApplicationTests {
    @Autowired
    private BookDao bookDao;
    @Test
    void getByIdTest() {
        Book book = bookDao.selectById(1);
        System.out.println(book);
    }
    @Test
    void updateTest(){
        Book book = new Book();
        book.setId(1);
        book.setDescription("操作系统从入门到入坟");
        int i = bookDao.updateById(book);
        System.out.println(i);
    }
    <!--
        bookDao.deleteBatchIds(List);   <!--多数据按id删除-->
        bookDao.selectBatchIds(List);   <!--多数据按id查询-->
    -->
}

2:分页查询

2.1:设置分页拦截器作为Spring管理的bean

@Configuration
public class MpConfig {
    @Bean
    public MybatisPlusInterceptor pageInterceptor(){
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        interceptor.addInnerInterceptor(new PaginationInnerInterceptor());
        return interceptor;
    }
}

2.2:执行分页查询

public void pageTest(){
    IPage<Book> page = new Page<Book>(1, 3);
    bookDao.selectPage(page,null);
    System.out.println("当前页码:"+page.getCurrent());
    System.out.println("每页数据总量:"+page.getSize());
    System.out.println("总页数:"+page.getPages());
    System.out.println("数据总量:"+page.getTotal());
    System.out.println("当前页数据:"+page.getRecords());
}
3:条件查询
//条件查询
@Test
public void getAllTest(){
    //方式一:按条件查询
    QueryWrapper<Book> qw = new QueryWrapper<Book>();
    qw.lt("id",6);    //小于
    for (Book book : bookDao.selectList(qw)) {
        System.out.println(book);
    }
    //方式二:lambda格式
    QueryWrapper<Book> qw = new QueryWrapper<Book>();
    qw.lambda().lt(Book::getId,6);
    for (Book book : bookDao.selectList(qw)) {
        System.out.println(book);
    }
    //方式三(推荐)
    LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
    lqw.lt(Book::getId,6);
    for (Book book : bookDao.selectList(lqw)) {
        System.out.println(book);
    }
    //多条件查询
    LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
    //4到10之间
    // lqw.lt(Book::getId,10).gt(Book::getId,4);
    //小于4或者大于10
    lqw.lt(Book::getId,4).or().gt(Book::getId,10);
    for (Book book : bookDao.selectList(lqw)) {
        System.out.println(book);
    }
    /*----------------------------------------NULL判定---------------------------------*/
    LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
    BookQuery bookQuery = new BookQuery();
    bookQuery.setId(10);
    bookQuery.setId2(5);
    //lt 小于,le 小于等于
    //gt 大于,ge 大于等于
    //链式编程
    /* 
    lqw.lt(null != bookQuery.getId(),Book::getId,bookQuery.getId())
    .gt(null != bookQuery.getId2(),Book::getId,bookQuery.getId2());
    */
    lqw.lt(null != bookQuery.getId(),Book::getId,bookQuery.getId());
    lqw.gt(null != bookQuery.getId2(),Book::getId,bookQuery.getId2());
    for (Book book : bookDao.selectList(lqw)) {
        System.out.println(book);
    }
    /*----------------------------查询投影---------------------------------*/
    //查询结果包含模型中部分属性
    LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
    //查询id、书名、描述
    lqw.select(Book::getId,Book::getName,Book::getDescription);
    List<Book> bookList = bookDao.selectList(lqw);
    bookList.forEach(System.out::println);
    //查询结果包含模型中未定义的属性
    QueryWrapper<Book> qw = new QueryWrapper<>();
    /* qw.select("id","name","description");
    for (Book book : bookDao.selectList(qw)) {
        System.out.println(book);
    } */
    qw.select("count(*) as nums,type");
    qw.groupBy("type");
    List<Map<String, Object>> maps = bookDao.selectMaps(qw);
    System.out.println(maps);
    /*-------------------------其他查询---------------------------------*/
    lqw.between(Book::getId,10,20);    //查询id在10到20之间,包含10和20
    lqw.like("计算机");      //like %计算机%
    lqw.likeLeft("计算机");    //like %计算机
    lqw.likeRigt("计算机");    //like 计算机%
}
4:映射匹配兼容性
@Data    //添加Lombok注解
@TableName("tbl_book")   //设置当前类与数据库表之间的对应关系
public class Book {
    private Integer id;
    @TableField(value = "type",select = false)   //value设置数据库表段名称,select:设置是否参与查询
    private String Type;
    private String name;
    private String description;
    @TableField(exist = false)   //设置属性在数据库表段中是否存在,默认为true,此属性无法与value合并使用
    private Integer isON;
}
  • 全局配置
mybatis-plus:
  global-config:
    db-config:
      id-type: auto    <!--设置id生成策略-->
      table-prefix: tbl_   <!--设置表之间对应关系,在所有pojo类名前加上"tbl_"-->
5:逻辑查询

5.1:数据库添加字段用于表示是否删除(注意尽量不要用关键字作为字段名)

5.2:用yml文件全局配置

mybatis-plus:
  global-config:
    banner: false
    db-config:
      logic-delete-field: is_delete   #设置表示逻辑删除的字段
      logic-delete-value: 1        #设置表示删除的值
      logic-not-delete-value: 0    #设置表示未删除的值

5.3:pojo类添加相应属性

// @TableLogic(value = "0",delval = "1")
private Integer is_delete;     //表示逻辑删除

5.4:执行删除操作

@Test
void deleteTest(){
    bookDao.deleteById(3);
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ku5rodfz-1653122017080)(SSM笔记.assets/image-20220521115510938.png)]

  • 注意这时执行的是UPDATE操作
    Book::getId,Book::getName,Book::getDescription);
    List bookList = bookDao.selectList(lqw);
    bookList.forEach(System.out::println);
    //查询结果包含模型中未定义的属性
    QueryWrapper qw = new QueryWrapper<>();
    /* qw.select(“id”,“name”,“description”);
    for (Book book : bookDao.selectList(qw)) {
    System.out.println(book);
    } /
    qw.select("count(
    ) as nums,type");
    qw.groupBy(“type”);
    List<Map<String, Object>> maps = bookDao.selectMaps(qw);
    System.out.println(maps);
    /-------------------------其他查询---------------------------------/
    lqw.between(Book::getId,10,20); //查询id在10到20之间,包含10和20
    lqw.like(“计算机”); //like %计算机%
    lqw.likeLeft(“计算机”); //like %计算机
    lqw.likeRigt(“计算机”); //like 计算机%
    }
###### 4:映射匹配兼容性
```java
@Data    //添加Lombok注解
@TableName("tbl_book")   //设置当前类与数据库表之间的对应关系
public class Book {
    private Integer id;
    @TableField(value = "type",select = false)   //value设置数据库表段名称,select:设置是否参与查询
    private String Type;
    private String name;
    private String description;
    @TableField(exist = false)   //设置属性在数据库表段中是否存在,默认为true,此属性无法与value合并使用
    private Integer isON;
}
  • 全局配置
mybatis-plus:
  global-config:
    db-config:
      id-type: auto    <!--设置id生成策略-->
      table-prefix: tbl_   <!--设置表之间对应关系,在所有pojo类名前加上"tbl_"-->
5:逻辑查询

5.1:数据库添加字段用于表示是否删除(注意尽量不要用关键字作为字段名)

5.2:用yml文件全局配置

mybatis-plus:
  global-config:
    banner: false
    db-config:
      logic-delete-field: is_delete   #设置表示逻辑删除的字段
      logic-delete-value: 1        #设置表示删除的值
      logic-not-delete-value: 0    #设置表示未删除的值

5.3:pojo类添加相应属性

// @TableLogic(value = "0",delval = "1")
private Integer is_delete;     //表示逻辑删除

5.4:执行删除操作

@Test
void deleteTest(){
    bookDao.deleteById(3);
}

  • 注意这时执行的是UPDATE操作
相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
5月前
|
Java 编译器 Maven
使用intellij idea搭建SSM架构的maven项目 详细
使用intellij idea搭建SSM架构的maven项目 详细
94 4
|
2月前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
365 37
|
6月前
|
Java 数据库连接 数据库
Mybatis逆向工程笔记小结
Mybatis逆向工程笔记小结
|
2月前
|
SQL Java 数据库连接
【Java笔记+踩坑】MyBatisPlus基础
MyBatisPlus简介、标准数据层开发CRUD、业务层继承IService、ServiceImpl、条件查询、LambdaQueryWrapper、id生成策略、逻辑删除、乐观锁@Version、代码生成器、ActiveRecord
【Java笔记+踩坑】MyBatisPlus基础
|
2月前
|
Java 数据库连接 Maven
【Java笔记+踩坑】Maven高级
分模块开发、依赖传递与冲突问题、 可选依赖和排除依赖、聚合和继承、属性、多环境配置与应用、私服安装和使用
【Java笔记+踩坑】Maven高级
|
2月前
|
前端开发 Java 数据库连接
【Java笔记+踩坑】SSM整合
统一结果封装、统一异常处理、整合图书案例、拦截器
【Java笔记+踩坑】SSM整合
|
2月前
|
Java 数据库连接 数据格式
【Java笔记+踩坑】Spring基础2——IOC,DI注解开发、整合Mybatis,Junit
IOC/DI配置管理DruidDataSource和properties、核心容器的创建、获取bean的方式、spring注解开发、注解开发管理第三方bean、Spring整合Mybatis和Junit
【Java笔记+踩坑】Spring基础2——IOC,DI注解开发、整合Mybatis,Junit
|
3月前
|
Java 应用服务中间件 Maven
Mac使用Idea配置传统SSM项目(非maven项目)
Mac使用Idea配置传统SSM项目(非maven项目)
51 1
|
5月前
|
XML Java 数据库连接
技术笔记:Maven的pom.xml(坐标的gav、依赖的scope)
技术笔记:Maven的pom.xml(坐标的gav、依赖的scope)
159 0
|
5月前
|
JSON JavaScript 前端开发
技术笔记:ssm异步上传图片
技术笔记:ssm异步上传图片
21 0

相关实验场景

更多

推荐镜像

更多