第一章 SpringBoot介绍

1 简介

Spring Boot是一个便捷搭建基于spring工程的脚手架;作用是帮助开发人员快速搭建大型的spring 项目。简化工程的配置和依赖管理;开发人员把时间都集中在业务开发上。

首页Spring Boot简介可以看到下面的一段介绍:

Spring Boot is designed to get you up and running as quickly as possible, with minimal upfront
confifiguration of Spring. Spring Boot takes an opinionated view of building production-ready
applications.

翻译一下:

Spring Boot的设计目的是让您尽可能快地启动和运行,而无需预先配置Spring。Spring Boot以一种固定的方
式来构建可用于生产级别的应用程序。

一般把Spring Boot称为搭建程序的脚手架或者说是便捷搭建基于Spring的工程脚手架。其最主要作用就是帮助开

发人员快速的构建庞大的spring项目,并且尽可能的减少一切xml配置,做到开箱即用,迅速上手,让开发人员关

注业务而非配置。

2 Spring boot特点

  • 为基于Spring的开发提供更快的入门体验
  • 开箱即用,没有代码生成,也无需XML配置。同时也可以修改默认值来满足特定的需求。
  • 提供了一些大型项目中常见的非功能性特性,如嵌入式服务器、安全、指标,健康检测、外部配置等。
  • Spring Boot并不是不对Spring功能上的增强,而是提供了一种快速使用Spring的方式。

3 为什么要学习Spring Boot

java一直被人诟病的一点就是臃肿、麻烦。当我们还在辛苦的搭建项目时,可能Python程序员已经把功能写好了,究其原因注意是两点:

  • 复杂的配置
    项目各种配置其实是开发时的损耗, 因为在思考 Spring 特性配置和解决业务问题之间需要进行思维切换,所以写配置挤占了写应用程序逻辑的时间。

  • 混乱的依赖管理

    项目的依赖管理也是件吃力不讨好的事情。决定项目里要用哪些库就已经够让人头痛的了,你还要知道这些库
    的哪个版本和其他库不会有冲突,这难题实在太棘手。并且,依赖管理也是一种损耗,添加依赖不是写应用程
    序代码。一旦选错了依赖的版本,随之而来的不兼容问题毫无疑问会是生产力杀手。

而Spring Boot让这一切成为过去!Spring Boot 简化了基于Spring的应用开发,只需要“run”就能创建一个独立的、生产级别的Spring应用。
Spring Boot为Spring平台及第三方库提供开箱即用的设置(提供默认设置,存放默认配置的包就是启动器
starter),这样我们就可以简单的开始。多数Spring Boot应用只需要很少的Spring配置。

我们可以使用Spring Boot创建java应用,并使用java –jar 启动它,就能得到一个生产级别的web工程。

第二章 入门案例

1 创建SpringBoot项目

idea->file->new->project 选中 Srping Initializr,然后一直下一步就可以了

默认生成的Spring Boot项目;

  • java文件夹

  • resources文件夹中目录结构

    • static:保存所有的静态资源,例如: js,css ,images;

    • templates:保存所有的模板页面(Spring Boot默认jar包使用嵌入式的Tomcat,默认不支持JSP页面),可以使用模板引擎(freemarker、thymeleaf);

    • application.properties:Spring Boot应用的配置文件,可以修改一些默认设置;

2 添加依赖

SpringBoot提供了许多“启动器”,启动器在类路径中添加依赖的jar包。

spring-boot-starter-parent是一个特别的启动器,里面已经对各种常用依赖的版本进行了管理,我们的项目需要以这个项目为父工程,这样我们就不用操心依赖的版本问题了。

pom.xml

<?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.4.2</version>
        <relativePath/>
    </parent>

    <groupId>com.xxx</groupId>
    <artifactId>springboot01</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>springboot01</name>
    <description>Demo project for Spring Boot</description>
    
    <properties>
        <java.version>1.8</java.version>
    </properties>
    <dependencies>

        <!-- 启动器 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency> 
        
         <!-- 单元测试 -->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.13.1</version>
            <scope>test</scope>
        </dependency>

        <!-- 增加web依赖 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

添加web依赖后(spring-boot-starter-web),会发现tomcat、springboot、springmvc等依赖已经加进来了。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-3a2ZzNkZ-1656551199494)(image/1.png)]

所以启动SpringBoot项目不需要再像ssm一样需要额外的tomcat了。

3 创建代码

创建启动类Springboot01Application

在com/xxx包中创建启动类,这个类要和controller同级

springboot启动原理: 采用springmvc注解方式启动,内置http服务器(默认tomcat),所以不需要额外配置Tomcat

package com.xxx;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

/**
 * @SpringBootApplication 来标注一个主程序类,说明这是一个Spring Boot应用
 */
@SpringBootApplication
public class Springboot01Application {

    //启动应用
    public static void main(String[] args) {
        SpringApplication.run(Springboot01Application.class, args);
    }

}

说明:

1 @SpringBootApplication注解

@SpringBootApplication源码

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {
      @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
      @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {

@SpringBootApplication:Spring Boot应用标注在某个类上说明这个类是SpringBoot的主配置类,SpringBoot就应该运行这个类的main方法来启动SpringBoot应用;

@SpringBootApplication组合注解,兼备了@EnableAutoConfiguration、@Configuration和@ComponentScan 注解的功能。

  • @EnableAutoConfiguration: 启动自动配置机制

  • @ComponentScan: 启动组件扫描。

  • @Configuration: 注册额外的bean或者导入其他配置类。

4 测试

直接运行启动类Springboot01Application即可,控制台出现如下提示,说明启动成功

com.xxx.Springboot01Application: Started Springboot01Application in 1.909 seconds (JVM running for 3.553)

测试完毕!

增加Controller,然后继续测试

写法一:

@RestController
public class HelloController {

    @RequestMapping("/")
    public String home() {
        return "第一个SpringBoot!!!";
    }

}

写法二:

@Controller
public class HelloController {

    @RequestMapping("/")
    @ResponseBody
    public String home() {
        return "第一个SpringBoot!!!";
    }

}

由此可见:@RestController = @Controller+@ResponseBody

通过浏览器访问下面地址:http://127.0.0.1:8080/

浏览器页面出现 “第一个SpringBoot!!!” ,说明搭建成功!

注意:默认端口就是8080。

5 配置文件

SpringBoot使用一个全局的配置文件,配置文件名是固定的,因为SpringBoot遵从约定大于配置规则。

配置文件位于resources文件夹下,配置文件支持2种风格:

  • application.properties
  • application.yml

application.properties

SpringBoot会默认扫描这个配置文件,这里的命名只能是application,因为SpringBoot遵从约定大于配置规则。

application.properties示例:

server.port=8080
server.servlet.context-path=/

YAML语法

基本语法:

  • k:(空格)v:表示一对键值对(空格必须有)
  • 以空格的缩进来控制层级关系;只要是左对齐的一列数据,都是同一个层级的
  • 属性和值也是大小写敏感
  • 值中的字符串默认不用加上单引号或者双引号

application.yml示例:

server:
  port: 8080
  servlet-path: /

备注:其实SpringBoot底层会把application.yml文件解析为application.properties

6 @Value读取配置文件

使用@Value可以把配置文件的值直接注入到成员变量中。

案例:

@Controller
//默认获取的是application.properties中的值
//如果不是,则需要加上@PropertySource注解,且指定配置文件名称
//@PropertySource(value = "classpath:application.properties") //可以不配置
//@PropertySource(value = "classpath:user.properties")        //必须配置
public class TestController {

    //方式一
    @Autowired
    private Environment environment;

    //方式二
    @Value("${server.port}")
    private String port;

    @GetMapping("test")
    @ResponseBody
    public void test() {
        System.out.println("获取配置文件中的值:" + environment.getProperty("server.port"));
        System.out.println("获取配置文件中的值port:" + port);
    }

}

7 stater起步器

Starters是一系列很好用的依赖描述符,可以包含在应用程序中。您可以为您需要的所有Spring和相关技术提供一站式服务,而无需搜索示例代码和复制粘贴大量的依赖描述符。例如,如果您想开始使用Spring和JPA进行数据库访问,只需将spring-boot-starter-data-jpa项目中的依赖项即可。

starter包含了一系列依赖,且支持传递依赖。

下面的starter由SpringBoot提供,放在org.springframework.boot包下面。

名称 描述
spring-boot-starter 核心starter,包括自动配置支持、日志记录和YAML。
spring-boot-starter-activemq Starter for JMS messaging using Apache ActiveMQ
spring-boot-starter-amqp Starter for using Spring AMQP and Rabbit MQ
spring-boot-starter-aop Starter for aspect-oriented programming with Spring AOP and AspectJ
spring-boot-starter-artemis Starter for JMS messaging using Apache Artemis
spring-boot-starter-batch Starter for using Spring Batch
spring-boot-starter-cache Starter for using Spring Framework’s caching support
spring-boot-starter-data-cassandra Starter for using Cassandra distributed database and Spring Data Cassandra
spring-boot-starter-data-cassandra-reactive Starter for using Cassandra distributed database and Spring Data Cassandra Reactive
spring-boot-starter-data-couchbase Starter for using Couchbase document-oriented database and Spring Data Couchbase
spring-boot-starter-data-couchbase-reactive Starter for using Couchbase document-oriented database and Spring Data Couchbase Reactive
spring-boot-starter-data-elasticsearch Starter for using Elasticsearch search and analytics engine and Spring Data Elasticsearch
spring-boot-starter-data-jdbc Starter for using Spring Data JDBC
spring-boot-starter-data-jpa Starter for using Spring Data JPA with Hibernate
spring-boot-starter-data-ldap Starter for using Spring Data LDAP
spring-boot-starter-data-mongodb Starter for using MongoDB document-oriented database and Spring Data MongoDB
spring-boot-starter-data-mongodb-reactive Starter for using MongoDB document-oriented database and Spring Data MongoDB Reactive
spring-boot-starter-data-neo4j Starter for using Neo4j graph database and Spring Data Neo4j
spring-boot-starter-data-r2dbc Starter for using Spring Data R2DBC
spring-boot-starter-data-redis Starter for using Redis key-value data store with Spring Data Redis and the Lettuce client
spring-boot-starter-data-redis-reactive Starter for using Redis key-value data store with Spring Data Redis reactive and the Lettuce client
spring-boot-starter-data-rest Starter for exposing Spring Data repositories over REST using Spring Data REST
spring-boot-starter-data-solr Starter for using the Apache Solr search platform with Spring Data Solr
spring-boot-starter-freemarker Starter for building MVC web applications using FreeMarker views
spring-boot-starter-groovy-templates Starter for building MVC web applications using Groovy Templates views
spring-boot-starter-hateoas Starter for building hypermedia-based RESTful web application with Spring MVC and Spring HATEOAS
spring-boot-starter-integration Starter for using Spring Integration
spring-boot-starter-jdbc Starter for using JDBC with the HikariCP connection pool
spring-boot-starter-jersey Starter for building RESTful web applications using JAX-RS and Jersey. An alternative to spring-boot-starter-web
spring-boot-starter-jooq Starter for using jOOQ to access SQL databases. An alternative to spring-boot-starter-data-jpa or spring-boot-starter-jdbc
spring-boot-starter-json Starter for reading and writing json
spring-boot-starter-jta-atomikos Starter for JTA transactions using Atomikos
spring-boot-starter-jta-bitronix Starter for JTA transactions using Bitronix. Deprecated since 2.3.0
spring-boot-starter-mail Starter for using Java Mail and Spring Framework’s email sending support
spring-boot-starter-mustache Starter for building web applications using Mustache views
spring-boot-starter-oauth2-client Starter for using Spring Security’s OAuth2/OpenID Connect client features
spring-boot-starter-oauth2-resource-server Starter for using Spring Security’s OAuth2 resource server features
spring-boot-starter-quartz Starter for using the Quartz scheduler
spring-boot-starter-rsocket Starter for building RSocket clients and servers
spring-boot-starter-security Starter for using Spring Security
spring-boot-starter-test Starter for testing Spring Boot applications with libraries including JUnit Jupiter, Hamcrest and Mockito
spring-boot-starter-thymeleaf Starter for building MVC web applications using Thymeleaf views
spring-boot-starter-validation Starter for using Java Bean Validation with Hibernate Validator
spring-boot-starter-web Starter for building web, including RESTful, applications using Spring MVC. Uses Tomcat as the default embedded container
spring-boot-starter-web-services Starter for using Spring Web Services
spring-boot-starter-webflux Starter for building WebFlux applications using Spring Framework’s Reactive Web support
spring-boot-starter-websocket Starter for building WebSocket applications using Spring Framework’s WebSocket support

后面会继续讲解这些起步器的使用。

8 热部署

在pom.xml中添加依赖

<!-- 热部署 -->
        <!-- devtools可以实现页面热部署(即页面修改后会立即生效,
            这个可以直接在application.properties文件中配置spring.thymeleaf.cache=false来实现) -->
        <!-- 实现类文件热部署(类文件修改后不会立即生效),实现对属性文件的热部署。 -->
        <!-- 即devtools会监听classpath下的文件变动,并且会立即重启应用(发生在保存时机),
            注意:因为其采用的虚拟机机制,该项重启是很快的 -->
        <!-- (1)base classloader (Base类加载器):加载不改变的Class,例如:第三方提供的jar包。 -->
        <!-- (2)restart classloader(Restart类加载器):加载正在开发的Class。 -->
        <!-- 为什么重启很快,因为重启的时候只是加载了在开发的Class,没有重新加载第三方的jar包。 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <!-- optional=true, 依赖不会传递, 该项目依赖devtools;
                之后依赖boot项目的项目如果想要使用devtools, 需要重新引入 -->
            <optional>true</optional>
        </dependency>

在application.properties中添加配置信息

#热部署生效
spring.devtools.restart.enabled=true
#设置重启的目录,添加那个目录的文件需要restart
spring.devtools.restart.additional-paths=src/main/java
#如使用 thymeleaf 模板,记得在配置文件中关掉 thymeleaf 缓存
#spring.thymeleaf.cache=false

idea的设置

  1. File->Settings->Build ,Exception,Deployment->Compiler-> 勾上Build Project automatically

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-P7zA03Ua-1656551199502)(image/Springboot基础/image-20210723071801569.png)]

  2. ctrl + shift + alt + / 然后选择Registry,勾上 Compiler.autoMake.allow.when.app.running

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-HBblvEGn-1656551199503)(image/Springboot基础/image-20210723071815906.png)]

勾上 Compiler.autoMake.allow.when.app.running

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6T7xWhVu-1656551199503)(image/Springboot基础/image-20210723071936776.png)]

测试

每次修改完代码,要ctrl+s保存,然后热部署才会生效

9 返回json格式数据

springboot返回json格式数据

1 返回对象数据

增加POJO:

public class Student {
    private String name;
    private int age;
}

UserController

@RestController
public class UserController {
    
	@RequestMapping("/getStudent")
	public Student getStudent() {
    	return new Student("张三5",44);
	}
}

在浏览器访问:http://127.0.0.1:8080/getStudent,即可以看到返回的json数据

2 返回数组数据

在TestController增加方法getStudents():

@RequestMapping("/getStudents")
public List<Student> getStudents() {
    List students=new ArrayList();

    students.add(new Student("张三",12));
    students.add(new Student("张三2",13));
    students.add(new Student("张三3",22));
    students.add(new Student("张三4",33));
    students.add(new Student("张三5",44));

    return students;
}

在浏览器访问:http://127.0.0.1:8080/getStudents,即可以看到返回的数组数据

3 返回HashMap数据

增加方法:

@RequestMapping("/getStudentMap")
public Map getStudentMap() {
    Map map = new HashMap();
    List students=new ArrayList();
    students.add(new Student("张三",12));
    students.add(new Student("张三2",13));
    students.add(new Student("张三3",22));
    students.add(new Student("张三4",33));
    students.add(new Student("张三5",44));

    map.put("retcode",1);
    map.put("students",students);
    return map;
}

10、springboot使用静态资源

在 Spring Boot 中,默认情况下,一共有5个位置可以放静态资源,五个路径分别是如下5个:

classpath:/META-INF/resources/
classpath:/resources/
classpath:/static/
classpath:/public/
/:当前项目的根路径

优先级别是从上往下,即如果在以上5个目录出现相同的文件,会按照这个优先级别来显示。

在资源文件resources目录下建立如下四个目录:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VkAixILm-1656551199504)(image/Springboot/image-20210223144841813.png)]

我们项目的静态资源存放在 static下面。

第三章 集成数据库

1、springboot集成jdbcTemplate

1)增加依赖

在原有的springboot的基础上,增加下面的依赖:

<dependency>
	<groupId>mysql</groupId>
	<artifactId>mysql-connector-java</artifactId>
	<version>8.0.11</version>    
    <scope>runtime</scope>
</dependency>
<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>druid</artifactId>
	<version>1.1.6</version>
</dependency>
<!-- jdbcTemplate -->
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>

2)增加数据库配置

修改application.yml

​ 其中如果是mysql8的版本,驱动和url要做相应的修改

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

mysql8的配置:

驱动:com.mysql.cj.jdbc.Driver

url : jdbc:mysql://localhost:3306/db_user?serverTimezone=UTC&characterEncoding=utf8&useUnicode=true&useSSL=false

3)创建表和增加数据

DROP TABLE IF EXISTS `tb_user`;
CREATE TABLE `tb_user` (
  `id` int(11) NOT NULL AUTO_INCREMENT COMMENT 'ID',
  `username` varchar(50) NOT NULL COMMENT '用户名',
  `age` int(11) NOT NULL COMMENT '年龄',
  `ctm` datetime NOT NULL COMMENT '创建时间',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8;



INSERT INTO `tb_user` VALUES ('1', '张三', '18', '2019-01-24 09:07:41');
INSERT INTO `tb_user` VALUES ('2', '李四', '20', '2019-01-24 09:07:41');
INSERT INTO `tb_user` VALUES ('3', '王五', '19', '2019-01-24 09:07:41');

4)增加实体类

public class User {
    private int id;
    private String username;
    private int age;
    private Date ctm;
 
    public User() {
    }
 
    public User(String username, int age) {
        this.username = username;
        this.age = age;
        this.ctm = new Date();
    }
 
    // Getter、Setter
}

5)增加dao接口

public interface UserDao {
    User getUserById(Integer id);
    public List<User> getUserList();
    public int add(User user);
    public int update(Integer id, User user);
    public int delete(Integer id);
}

5)增加dao实现类

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;
 
import java.util.Date;
import java.util.List;
 
@Repository
public class UserDaoImpl implements UserDao {
 
    @Autowired
    private JdbcTemplate jdbcTemplate;
 
    @Override
    public User getUserById(Integer id) {
        List<User> list = jdbcTemplate.query("select * from tb_user where id = ?", new Object[]{id}, new BeanPropertyRowMapper(User.class));
        if(list!=null && list.size()>0){
            return list.get(0);
        }else{
            return null;
        }
    }
 
    @Override
    public List<User> getUserList() {
        List<User> list = jdbcTemplate.query("select * from tb_user", new Object[]{}, new BeanPropertyRowMapper(User.class));
        if(list!=null && list.size()>0){
            return list;
        }else{
            return null;
        }
    }
 
 
    @Override
    public int add(User user) {
        return jdbcTemplate.update("insert into tb_user(username, age, ctm) values(?, ?, ?)",
                user.getUsername(),user.getAge(), new Date());
    }
 
    @Override
    public int update(Integer id, User user) {
        return jdbcTemplate.update("UPDATE tb_user SET username = ? , age = ? WHERE id=?",
                user.getUsername(),user.getAge(), id);
    }
 
    @Override
    public int delete(Integer id) {
        return jdbcTemplate.update("DELETE from tb_user where id = ? ",id);
    }
 
}

7)测试

a、修改controller,增加getUserById()方法和成员变量userDao

public class UserController {
    @Autowired
    UserDao userDao;//由springboot自动注入dao对象给contrroller

    //测试访问数据获取对象
    @RequestMapping("/getUserById")
    public User getUserById(int id){
        return userDao.getUserById(id);
    }

b、访问url获取用户信息

http://127.0.0.1:8080/user/getUserById?id=3

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Neh9O6ms-1656551199506)(…/…/…/…/work/03-课件/MD版本/04-核心框架课件/09-springboot v2/image/springboot/image-20210225180238184.png)]

8)解决pojo属性和数据库字段名不一致问题

方法1:使用别名
 public Project getProjectById2(Integer id) {
        String sql = "select id,name,engineer,start_time startDate,cost_months costMonths," +
                "project_desc projectDesc, complete_rate completeRate,working_rate workingRate," +
                "waiting_rate waitingRate " +
                "from tb_project where id=?";
        System.out.println(sql);
        List<Project> list = jdbcTemplate.query(sql,
                new Object[]{id},new BeanPropertyRowMapper(Project.class));
        if(list != null && list.size()>0)
            return list.get(0);
        else
            return null;
    }
方法2:自定义mapper
public Project getProjectById(Integer id) {
        String sql = "select  * from tb_project where id=?";
        System.out.println(sql);
        List<Project> list = jdbcTemplate.query(sql,
                new Object[]{id}, new ProjectRomapper());

        if(list != null && list.size()>0)
            return list.get(0);
        else
            return null;
    }

ProjectRomapper 代码

public class ProjectRomapper implements RowMapper<Project> {
    @Override
    public Project mapRow(ResultSet resultSet, int i) throws SQLException {
        Project project = new Project();
        project.setId(resultSet.getInt("id"));
        project.setName(resultSet.getString("name"));
        project.setEngineer(resultSet.getString("engineer"));
        project.setStartDate(resultSet.getDate("start_time"));
        project.setCostMonths(resultSet.getInt("cost_months"));
        project.setCompleteRate(resultSet.getDouble("complete_rate"));
        project.setWaitingRate(resultSet.getDouble("working_rate"));
        project.setWorkingRate(resultSet.getDouble("waiting_rate"));
        return project;
    }
}

2 集成Mybatis

1 依赖

pom.xml

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <!--spring-mybatis-->
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>1.1.1</version>
        </dependency>

        <!--mysql-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.37</version>
        </dependency>

        <!--druid-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.0.16</version>
        </dependency>
		<!-- jpa依赖 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>

2 配置文件

在resources目录下建立mybatis-config.xml

mybatis-config.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>

    <!--需要内容自己添加,可以什么都不写-->
    <settings>
        <!--开启驼峰命名匹配规则-->
        <setting name="mapUnderscoreToCamelCase" value="true"/>
    </settings>

</configuration>

application.properties

server.port=8080
server.servlet.context-path=/
#
#数据源
spring.datasource.url=jdbc:mysql://localhost:3306/travel?useUnicode=true&amp;characterEncoding=utf-8&zeroDateTimeBehavior=convertToNull
spring.datasource.username=root
spring.datasource.password=root
spring.datasource.driverClassName=com.mysql.jdbc.Driver
#
#mybatis
mybatis.config-location=classpath:mybatis-config.xml
mybatis.checkConfigLocation=true
# mybatis 别名扫描
mybatis.type-aliases-package=edu.ln.travel.pojo
# mapper.xml文件全部放在resources/mappers目录中
mybatis.mapper-locations=classpath:mappers/*.xml

注意:如果是mysql8,

driver-class-name= com.mysql.cj.jdbc.Driver

url=

jdbc:mysql://localhost:3306/travel?serverTimezone=UTC&characterEncoding=utf8&useUnicode=true&useSSL=false

3 model层

UserDao需要加入@Mapper注解

如果UserDao不加@Mapper注解 ,也可以在springboot的启动类上加上@MapperScan(“com.dao”)注解

@Mapper
public interface UserDao {

    public List<User> findAll();

}

pojo

@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
    private Integer uid;
    private String name;
    private Integer telephone;
}

resources/mappers/UserMapper.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "mybatis-3-mapper.dtd" >
    <mapper namespace="edu.ln.travel.dao.UserDao">

        <select id="findAll" resultType="User">
            select * from tab_user
        </select>
    </mapper>

4、测试集成mybatis

@RunWith(SpringRunner.class)
@SpringBootTest(classes = MybatisApplication.class)
class MybatisApplicationTest {
    @Autowired
    UserDao userDao;

    @Test
    public void test1(){
        List<User> users = userDao.findAll();
        System.out.println(users);
    }

    @Test
    public void test2(){
        User user = userDao.findById(4);
        System.out.println(user);
    }
}

3、集成通用mapper

1)引入依赖

通用Mapper的作者也为自己的插件编写了启动器,我们直接引入即可:

<!-- 通用mapper -->
<dependency>
    <groupId>tk.mybatis</groupId>
    <artifactId>mapper-spring-boot-starter</artifactId>
    <version>4.2.1</version>
</dependency>

不需要做任何配置就可以使用了。

@Mapper
public interface UserMapper extends tk.mybatis.mapper.common.Mapper<User>{
}

2)修改pojo增加jpa的设置

pojo

@Data
@AllArgsConstructor
@NoArgsConstructor
@Table(name="tab_user")
public class User {
    @Id
    private Integer uid;
    private String name;
    private Integer telephone;
}

3)测试集成通用mapper

@RunWith(SpringRunner.class)
@SpringBootTest(classes = MybatisApplication.class)
class MybatisApplicationTest {
    @Autowired
    UserDao userDao;

    @Autowired
    UserMapper userMapper;

    @Test
    public void test1(){
        List<User> users = userDao.findAll();
        System.out.println(users);
    }

    @Test
    public void test2(){
        User user = userDao.findById(4);
        System.out.println(user);
    }

    ///============user mapper的测试
    @Test
    public void testMapper1(){
        User user = new User(null,"杰伦-布朗",28);
        int ret = userMapper.insert(user);
        System.out.println("usrmapper 插入用户: " + ret);
    }
}

3、集成PageHelper

1 增加依赖

注意,springboot2.6.0和springboot2.6.1以上只能使用pagehelper-spring-boot-starter1.4.1及以上版本,如果用低版本的pagehelper-spring-boot-starter,springboot需要降版本。

  <dependency>
      <groupId>com.github.pagehelper</groupId>
      <artifactId>pagehelper-spring-boot-starter</artifactId>
      <version>1.4.3</version>
  </dependency>

2、application配置文件中配置

pagehelper分页插件配置,这些属性不加也可以实现分页功能

pagehelper.helperDialect=mysql
pagehelper.reasonable=true
pagehelper.supportMethodsArguments=true
pagehelper.params=count=countSql

3、PageHelper的使用

在userService增加分页功能。

@Override
public Page<User> listByPage(int pageNo, int pageSize) {
    PageHelper.startPage(pageNo, pageSize);
    Page<User> pageUsers =(Page<User> ) userDao.findAll();
    System.out.println("总条数:"+pageUsers.getTotal());
    return pageUsers;
}

第四章 使用springboot

1 模板引擎

1)介绍

常见的模板引擎有JSP、Velocity、Freemarker、Thymeleaf, 使用springboot整合jsp并不是很好,因为springboot默认没有对jsp有很大的支持,SpringBoot推荐使用Thymeleaf

2)Thymeleaf的特点

  • 动静结合:Thymeleaf 在有网络和无网络的环境下皆可运行,即它可以让美工在浏览器查看页面的静态效果,也可以让程序员在服务器查看带数据的动态页面效果。这是由于它支持 html 原型,然后在 html 标签里增加额外的属性来达到模板+数据的展示方式。浏览器解释 html 时会忽略未定义的标签属性,所以 thymeleaf 的模板可以静态地运行;当有数据返回到页面时,Thymeleaf 标签会动态地替换掉静态内容,使页面动态显示。
  • 开箱即用:它提供标准和spring标准两种方言,可以直接套用模板实现JSTL、 OGNL表达式效果,避免每天套模板、该jstl、改标签的困扰。同时开发人员也可以扩展和创建自定义的方言。
  • 多方言支持:Thymeleaf 提供spring标准方言和一个与 SpringMVC 完美集成的可选模块,可以快速的实现表单绑定、属性编辑器、国际化等功能。
  • 与SpringBoot完美整合,SpringBoot提供了Thymeleaf的默认配置,并且为Thymeleaf设置了视图解析器,我们可以像以前操作jsp一样来操作Thymeleaf。代码几乎没有任何区别,就是在模板语法上有区别。

3)Thymeleaf的使用

首先引入依赖

<dependency>
     <groupId>org.springframework.boot</groupId>
     <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>

框架自动给我们默认分配了模版的前缀和后缀,我们只需要按部就班的将模版丢进去即可

image-20210921224719009

具体见:ThymeLeafProperties类默认ed配置。

修改配置文件

spring.thymeleaf.prefix=classpath:/templates/
spring.thymeleaf.check-template-location=true
spring.thymeleaf.suffix=.html
spring.thymeleaf.encoding=UTF-8
spring.thymeleaf.content-type=text/html 
spring.thymeleaf.mode=HTML5
spring.thymeleaf.cache=false

Thymeleaf语法规则

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wjjN4B7D-1656551199507)(image/2.png)]

案例1:基本使用

在resources/templates目录下创建index.html文件,只要我们把HTML页面放在classpath:/templates/下,thymeleaf就能自动渲染;

IndexController

@Controller
@RequestMapping("index")
public class IndexController {

    @RequestMapping("index")
    public String hello(Model model) {
        //跳转到index.html页面
         model.addAttribute("hello","你好");
        return "index";
    }
}

index.html

<!DOCTYPE html>
<!-- 导入thymeleaf的名称空间 -->
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <h1> SpringBoot集成Thymeleaf成功!</h1>
    <!--th:text 将div里面的文本内容设置为 -->
    <div th:text="${hello}">这是显示欢迎信息</div>
</body>
</html>

说明:

导入thymeleaf的名称空间,不导入则无法渲染

<html lang="en" xmlns:th="http://www.thymeleaf.org">
案例2:循环

html代码:

<tr th:each="user : ${users}">
    <td th:text="${user.name}">Onions</td>
    <td th:text="${user.age}">2.41</td>
</tr>

${users} 是要遍历的集合,可以是以下类型:

  • Iterable,实现了Iterable接口的类
  • Enumeration,枚举
  • Interator,迭代器
  • Map,遍历得到的是Map.Entry
  • Array,数组及其它一切符合数组结果的对象

在迭代的同时,我们也可以获取迭代的状态对象:

<tr th:each="user,stat : ${users}">
    <td th:text="${user.name}">Onions</td>
    <td th:text="${user.age}">2.41</td>
</tr>

stat对象包含以下属性:

  • index,从0开始的角标
  • count,元素的个数,从1开始
  • size,总元素个数
  • current,当前遍历到的元素
  • even/odd,返回是否为奇偶,boolean值
  • first/last,返回是否为第一或最后,boolean值
案例3:逻辑判断

逻辑判断Thymeleaf中使用th:if 或者 th:unless ,两者的意思恰好相反。

<span th:if="${user.age} > 24">老油条</span>

如果表达式的值为true,则标签会渲染到页面,否则不进行渲染。

以下情况被认定为true:

  • 表达式值为true
  • 表达式值为非0数值
  • 表达式值为非0字符
  • 表达式值为字符串,但不是"false","no","off"
  • 表达式不是布尔、字符串、数字、字符中的任何一种

其它情况包括null都被认定为false。

案例4:JS模板

模板引擎不仅可以渲染html,也可以对JS中的进行预处理。而且为了在纯静态环境下可以运行,其Thymeleaf代码可以被注释起来:

<script th:inline="javascript">
    const user = /*[[${user}]]*/ {};
    const age = /*[[${user.age}]]*/ 20;
    console.log(user);
    console.log(age)
</script>

在script标签中通过th:inline="javascript"来声明这是要特殊处理的js脚本

语法结构:

const user = /*[[Thymeleaf表达式]]*/ "静态环境下的默认值";

因为Thymeleaf被注释起来,因此即便是静态环境下, js代码也不会报错,而是采用表达式后面跟着的默认值。且User对象会被直接处理为json格式。

2 全局异常捕获

如果每个方法都可能会发生异常,每个方法都加上try不好,因此使用全局捕获异常处理

全局捕获异常: 整个web请求项目全局捕获异常。

/**
 * @auth admin
 * @date
 * @Description 全局异常捕获类
 */
@ControllerAdvice
public class GlobalException {

    private static final Logger log = LoggerFactory.getLogger(GlobalException.class);

    @ResponseBody
    @ExceptionHandler(NullPointerException.class)
    public Map<String, Object> nullPointerException() {
        //实际开发中,会将错误记录在日志中,每天检测有哪些错误报告,通过邮件发送给你
        Map<String, Object> errorResultMap = new HashMap<String, Object>();
        errorResultMap.put("code", "500");
        errorResultMap.put("msg", "全局捕获异常-NullPointerException");
        return errorResultMap;
    }

    @ResponseBody
    @ExceptionHandler(RuntimeException.class)
    public Map<String, Object> runtimeException() {
        Map<String, Object> errorResultMap = new HashMap<String, Object>();
        errorResultMap.put("code", "500");
        errorResultMap.put("msg", "全局捕获异常-runtimeException");
        return errorResultMap;
    }

    @ResponseBody
    @ExceptionHandler(Exception.class)
    public Map<String, Object> exception() {
        Map<String, Object> errorResultMap = new HashMap<String, Object>();
        errorResultMap.put("code", "500");
        errorResultMap.put("msg", "全局捕获异常-exception");
        return errorResultMap;
    }

}

测试:

在任何一个类中手动写一个异常,例如 int a=1/0,发现会被全局异常类捕获到。

说明:

1 发生异常会首先被子类捕获,如果子类捕获不到,由父类捕获,直至根类Exception捕获

2 可以自定义异常类

3 AOP统一处理日志

pom依赖

        <!-- springboot aop技术 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

aop类

package com.aop;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * @auth admin
 * @date
 * @Description aop类
 */
@Aspect
@Component
public class WebLogAspect {

    private static final Logger log = LoggerFactory.getLogger(WebLogAspect.class);

    @Pointcut("execution(* com.controller.*.*(..))")
    public void webLog() {
    }

    @Before("webLog()")
    public void doBefore(JoinPoint joinPoint) throws Throwable {
        //接收到请求,记录请求内容,可以保存到nosql中
        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = attributes.getRequest();
        //记录下请求内容
        log.info("URL:" + request.getRequestURL().toString());
        log.info("HTTP_METHOD:" + request.getMethod());
        log.info("IP:" + request.getRemoteAddr());
        Enumeration<String> enu = request.getParameterNames();
        while (enu.hasMoreElements()) {
            String name = (String) enu.nextElement();
            log.info("name:{},value:{}", name, request.getParameter(name));
        }
    }

    @AfterReturning(returning = "ret", pointcut = "webLog()")
    public void doAfterReturning(Object ret) throws Throwable {
        //处理完请求,返回内容
        log.info("返回内容 :" + ret);
    }

}

4 集成Swagger3

Swagger是一个可以根据你的代码,自动生成接口文档的一个工具,并且可以用作接口测试工具,Swagger 3.0版本是在Swagger2的基础上进行了部分升级, 使用和Swagger2没有多少区别

一个重要的优化是依赖的引入,由之前的多个依赖变更为一个依赖,跟随springboot-starter风格,同时引入了新的开关注解 @EnableOpenApi 以代替@EnableSwagger2 。

必要工作只有两个:添加swagger3的starter依赖包,在springboot主程序类添加@EnableOpenApi开关注解。

第1步:增加依赖,修改pom.xml

<!-- swagger -->
<dependency>
     <groupId>io.springfox</groupId>
     <artifactId>springfox-boot-starter</artifactId>
     <version>3.0.0</version>
</dependency>

第2步:添加开关注解@EnableOpenApi

@SpringBootApplication
    @EnableOpenApi //启动swagger3
public class DTourApp {

    public static void main(String[] args) {
        SpringApplication.run(DemoSwagger3Application.class, args);
    }

}

第3步:添加Swagger配置类(可选步骤)

自定义首页属性 Docket配置

package edu.ln.tour.config;
@Configuration
public class Swagger3 {
    @Bean
    public Docket docket() {
        return new Docket(DocumentationType.OAS_30).apiInfo(
                new ApiInfoBuilder()
                        .contact(new Contact("黄豆", "", "88779900@qq.com"))
                        .title("岭南旅游网")
                        .build()
                );
    }
}

第4步:配置路径匹配策略,解决空指针问题

springboot2.6.x版本以上配置swagger3.0会提示[空指针异常]

错误如下

org.springframework.context.ApplicationContextException: Failed to start bean 'documentationPluginsBootstrapper'; nested exception is java.lang.NullPointerException

原因: 这是因为Springfox使用的路径匹配是基于AntPathMatcher的,而Spring Boot 2.6.X使用的是PathPatternMatcher。
解决:在application.properties里配置:spring.mvc.pathmatch.matching-strategy=ANT_PATH_MATCHER

则需要在applocation.properties增加如下配置:

spring.mvc.pathmatch.matching-strategy=ant_path_matcher

第5步

为了增加文档的可读性,我们还需要在接口中增加一些说明

UserController可以添加@Api和 @ApiOperation注解

@Api(description = "用户管理api")
@Controller
public class UserController {

    Logger logger = LoggerFactory.getLogger(getClass());

    @Resource
    private UserService userService;

    @ApiOperation("查询所有用户信息")
    @GetMapping("getAll")
    @ResponseBody
    public List<User> getAll() {
        return userService.findAll();
    }
    
    @ApiOperation("添加用户")
    @GetMapping("save")
    @ResponseBody
    public void save(User user) {
    }

}
@ApiModel("用户实体类")
public class User {

    @ApiModelProperty("id")
    private Long id;

    @ApiModelProperty(value = "用户姓名", required = true)
    private String name;

    @ApiModelProperty("用户年龄")
    private Integer age;

第6步:测试

启动springboot

swagger-ui访问地址:http://127.0.0.1:8080/swagger-ui/index.html

常用注解如下:

注解 使用的地方 用途
@Api 类/接口 描述类/接口主要用途
@ApiOperation 方法 描述方法的用途
@ApiImplicitParam 方法 用于描述接口的非对象参数
@ApiImplicitParams 方法 用于描述接口的非对象参数集
@ApiIgnore 类/方法/参数 Swagger 文档不会显示拥有该注解的接口
@ApiModel 参数实体类 可设置接口相关实体的描述
@ApiModelProperty 参数实体类属性 可设置实体属性的相关描述

第五章 日志

1 日志框架

在项目的开发中,日志是必不可少的一个记录事件的组件,所以也会相应的在项目中实现和构建日志框架。

市面上的日志框架:

JUL、JCL、Jboss-logging、logback、log4j、log4j2、slf4j…

通常情况下,日志是由一个抽象层+实现层的组合来搭建的。

日志门面 (日志的抽象层) 日志实现
JCL(Jakarta Commons Logging) 、SLF4j(Simple Logging Facade for Java) 、jboss-logging Log4j、JUL(java.util.logging) 、Log4j2 、 Logback

而SpringBoot选择了SLF4J+Logback的组合,这个组合是当下比较合适的一组。

2 SLF4j使用

application.properties加入日志配置信息

#
#日志的级别:由低到高trace<debug<info<warn<error
#SpringBoot默认是设置info级别
logging.level.edu.ln.tour=trace  #包的日志级别
#logging.level.root=trace    #root日志级别

日志的级别:优先级依次升高

image-20220626095802434

测试

@RunWith(SpringRunner.class)
@SpringBootTest(classes = TourApp.class)
public class TestLogger {
//记录器
	Logger logger = LoggerFactory.getLogger(getClass());

	@Test
	public void contextLoads() {
		//日志的级别;
		//由低到高   trace<debug<info<warn<error
		//可以调整输出的日志级别;日志就只会在这个级别以以后的高级别生效
		logger.trace("这是trace日志...");
		logger.debug("这是debug日志...");
		//SpringBoot默认使用的是info级别的,没有指定级别的就用SpringBoot默认规定的级别
		logger.info("这是info日志...");
		logger.warn("这是warn日志...");
		logger.error("这是error日志...");
	}
}

这些日志信息默认都会在控制台进行输出打印。

通过修改logging.level.com.xxx=日志级别,可以看到打印的信息也不同。

3 配置日志生成路径和名称

在项目的运行中,我们不可能一直看着控制台,而且日志数量会很大,那么我们需要指定我们需要的日志名称以及

日志生成的路径,用到两个配置都是在application.properties/yml中写,如下:(都不设置的话,不生成日志)

#
#日志的级别:由低到高trace<debug<info<warn<error
#SpringBoot默认是设置info级别
logging.level.edu.ln.tour=info
#
#在当前磁盘的根路径下创建spring文件夹和里面的log文件夹,并使用spring.log作为默认文件
logging.file.path=/spring/log
#
#在当前项目的根目录中生成a.log文件(生成完刷新一下项目)
#logging.file.name=a.log
#注意:logging.file.path和logging.file.name如果同时配置,则只有logging.file.name有效
#
# 设置控制台输出的日志的格式
#logging.pattern.console=%d{yyyy-MM-dd} [%thread] %-5level %logger{50} - %msg%n
# 设置指定文件中日志输出的格式
#logging.pattern.file=%d{yyyy-MM-dd} === [%thread] === %-5level === %logger{50} ==== %msg%n
logging.file.name logging.file.path Example Description
(none) (none) 只在控制台输出
指定文件名 (none) my.log 输出日志到当前项目的根目录中my.log文件
(none) 指定目录 /var/log 输出到当前磁盘/var/log的spring.log 文件中
指定文件名 指定目录 my.log 输出日志到当前项目的根目录中my.log文件

4 在service中使用logger

在service使用logger如下。然后观察spring.log的输出。

@Service
public class UserServiceImpl implements UserService {
    Logger logger = LoggerFactory.getLogger(getClass());

    @Autowired
    UserDao userDao;

    @Override
    public Page<User> listByPage(int pageNo, int pageSize) {
        logger.debug("分页:"+pageNo+"---"+pageSize);
        PageHelper.startPage(pageNo, pageSize);
        Page<User> pageUsers =(Page<User> ) userDao.findAll();
        System.out.println("总条数:"+pageUsers.getTotal());
        return pageUsers;
    }
}

spring.log的输出:

...
e.ln.tour.service.impl.UserServiceImpl   : 分页:2---3
....

第六章、springboot文件上传

1、文件上传

增加依赖
 <dependency>
      <groupId>commons-fileupload</groupId>
      <artifactId>commons-fileupload</artifactId>
      <version>1.3.1</version>
    </dependency>
    <dependency>
      <groupId>commons-io</groupId>
      <artifactId>commons-io</artifactId>
      <version>2.3</version>
    </dependency>
html代码
<form>
    单选:
    <input type="file" id="uploadFile" name="uploadFile"><br>
    <button type="button"  id="btn1">保存单选</button><br>
</form>
js代码
$(function(){
    $('#btn1').click(function(){
        var files = $('#uploadFile').prop('files');
        var data = new FormData();
        //注意uploadFile和controller中的参数名要一致
        data.append('uploadFile', files[0]);
        data.append('username',"zhangsan");
        $.ajax({
            url: '/fileupload',
            type: 'POST',
            data: data,
            cache: false,
            processData: false,
            contentType: false,
            success: function(msg){
                console.log( "Data Saved: " + msg );
            }
        });
    });
})
Controller代码
    @RequestMapping(value="/fileupload")
    @ResponseBody
    public void save22(String username, MultipartFile uploadFile) throws IOException {
        System.out.println(username);
       	System.out.println(uploadFile);
       	
       	String originalFilename = uploadFile.getOriginalFilename();
        uploadFile.transferTo(new File("C:\\upload\\"+originalFilename));
    }

2、多个文件上传

html

<form id="uploadForm" action="Upload" method="post" enctype="multipart/form-data">
    <input id="File1" name="uploadFile" accept="image/gif, image/jpeg" multiple="multiple" type="file" value="" />
    <input type="text" name="username" id="username" value="zhansan">
     <input id="btn" type="button" value="上传" />
</form>

js

$(function(){
	$('#btn').click(function(){
		var formData = new FormData($("#uploadForm")[0]);
		$.ajax({
			url: '/fileuploadMul',
			type: 'POST',
			data: formData,
			cache: false,
			processData: false,
			contentType: false,
			success: function(msg){
				console.log( "Data Saved: " + msg );
			}
		});
	});
})

controller

@RequestMapping(value="/fileuploadMul")
@ResponseBody
public void save23(String username, MultipartFile[] uploadFile) throws IOException {
	System.out.println(username);
	for (MultipartFile multipartFile : uploadFile) {
		String originalFilename = multipartFile.getOriginalFilename();
		multipartFile.transferTo(new File("C:\\upload\\"+originalFilename));
	}
}

3、文件下载

通过封装ResponseEntity,将文件流写入body中。这里注意一点,就是文件的格式需要根据具体文件的类型来设置,一般默认为application/octet-stream。文件头中设置缓存,以及文件的名字。

@RequestMapping(value = "/downloadFile", method = RequestMethod.GET)  
public ResponseEntity<InputStreamResource> downloadFile( String fileName)  
		throws IOException {  
	String filePath = "E:/test/"+fileName;  
	FileSystemResource file = new FileSystemResource(filePath);  
	HttpHeaders headers = new HttpHeaders();  
	headers.add("Cache-Control", "no-cache, no-store, must-revalidate");  
	headers.add("Content-Disposition", String.format("attachment; filename=\"%s\"", file.getFilename()));  
	headers.add("Pragma", "no-cache");  
	headers.add("Expires", "0");  

	return ResponseEntity  
			.ok()  
			.headers(headers)  
			.contentLength(file.contentLength())  
			.contentType(MediaType.parseMediaType("application/octet-stream"))  
			.body(new InputStreamResource(file.getInputStream()));  

}  
Logo

魔乐社区(Modelers.cn) 是一个中立、公益的人工智能社区,提供人工智能工具、模型、数据的托管、展示与应用协同服务,为人工智能开发及爱好者搭建开放的学习交流平台。社区通过理事会方式运作,由全产业链共同建设、共同运营、共同享有,推动国产AI生态繁荣发展。

更多推荐