JavaWeb【day11】--(SpringBootWeb案例)

打印 上一主题 下一主题

主题 972|帖子 972|积分 2916

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

x
SpringBootWeb案例

前面我们已经实现了员工信息的条件分页查询以及删除操作。 关于员工管理的功能,另有两个需要实现:


  • 新增员工
  • 修改员工
起首我们先完成"新增员工"的功能开发,再完成"修改员工"的功能开发。而在"新增员工"中,需要添加头像,而头像需要用到"文件上传"技术。 当整个员工管理功能全部开发完成之后,我们再通过设置文件来优化一些内容。
综上所述,我们今天的课程内容包罗以下四个部分:


  • 新增员工
  • 文件上传
  • 修改员工
  • 设置文件




1. 新增员工

1.1 需求

在新增用户时,我们需要保存用户的基本信息,而且还需要上传的员工的图片,目前我们先完成第一步操作,保存用户的基本信息。

1.2 接口文档

我们参照接口文档来开发新增员工功能


  • 基本信息
    1. 请求路径:/emps
    2. 请求方式:POST
    3. 接口描述:该接口用于添加员工的信息
    复制代码
  • 请求参数
    参数格式:application/json
    参数说明:
       名称类型是否必须备注usernamestring必须用户名namestring必须姓名gendernumber必须性别, 说明: 1 男, 2 女imagestring非必须图像deptIdnumber非必须部门identrydatestring非必须入职日期jobnumber非必须职位, 说明: 1 班主任,2 讲师, 3 学工主管, 4 教研主管, 5 咨询师请求数据样例:
    1. {
    2.  "image": "https://web-framework.oss-cn-hangzhou.aliyuncs.com/2022-09-03-07-37-38222.jpg",
    3.  "username": "linpingzhi",
    4.  "name": "林平之",
    5.  "gender": 1,
    6.  "job": 1,
    7.  "entrydate": "2022-09-18",
    8.  "deptId": 1
    9. }
    复制代码
  • 相应数据
    参数格式:application/json
    参数说明:
       参数名类型是否必须备注codenumber必须相应码,1 代表成功,0 代表失败msgstring非必须提示信息dataobject非必须返回的数据相应数据样例:
    1. {
    2.    "code":1,
    3.    "msg":"success",
    4.    "data":null
    5. }
    复制代码

1.3 思绪分析

新增员工的具体的流程:

   接口文档规定:
  

  • 请求路径:/emps
  • 请求方式:POST
  • 请求参数:Json格式数据
  • 相应数据:Json格式数据
  问题1:如何限定请求方式是POST?
  1. @PostMapping
复制代码
问题2:怎么在controller中接收json格式的请求参数?
  1. @RequestBody  //把前端传递的json数据填充到实体类中
复制代码

1.4 功能开发

EmpController
  1. @Slf4j
  2. @RestController
  3. @RequestMapping("/emps")
  4. public class EmpController {
  5.    @Autowired
  6.    private EmpService empService;
  7.    //新增
  8.    @PostMapping
  9.    public Result save(@RequestBody Emp emp){
  10.        //记录日志
  11.        log.info("新增员工, emp:{}",emp);
  12.        //调用业务层新增功能
  13.        empService.save(emp);
  14.        //响应
  15.        return Result.success();
  16.    }
  17.    //省略...
  18. }
复制代码
EmpService
  1. public interface EmpService {
  2.    /**
  3.     * 保存员工信息
  4.     * @param emp
  5.     */
  6.    void save(Emp emp);
  7.    
  8.    //省略...
  9. }
复制代码
复制代码
EmpServiceImpl
  1. @Slf4j
  2. @Service
  3. public class EmpServiceImpl implements EmpService {
  4.    @Autowired
  5.    private EmpMapper empMapper;
  6.    @Override
  7.    public void save(Emp emp) {
  8.        //补全数据
  9.        emp.setCreateTime(LocalDateTime.now());
  10.        emp.setUpdateTime(LocalDateTime.now());
  11.        //调用添加方法
  12.        empMapper.insert(emp);
  13.    }
  14.    //省略...
  15. }
复制代码
EmpMapper
  1. @Mapper
  2. public interface EmpMapper {
  3.     //新增员工
  4.     @Insert("insert into emp (username, name, gender, image, job, entrydate, dept_id, create_time, update_time) " +
  5.             "values (#{username}, #{name}, #{gender}, #{image}, #{job}, #{entrydate}, #{deptId}, #{createTime}, #{updateTime});")
  6.     void insert(Emp emp);
  7.     //省略...
  8. }
复制代码

1.5 功能测试

代码开发完成后,重启服务器,打开Postman发送 POST 请求,请求路径:http://localhost:8080/emps


1.6 前后端联调

功能测试通过后,我们再举行通过打开浏览器,测试后端功能接口:





2. 文件上传

在我们完成的新增员工功能中,还存在一个问题:没有头像(图片缺失)

上述问题,需要我们通过文件上传技术来解决。下面我们就进入到文件上传技术的学习。
文件上传技术这块我们主要解说三个方面:起首我们先对文件上传做一个整体的介绍,接着再学习文件上传的本地存储方式,最后学习云存储方式。
接下来我们就先来学习下什么是文件上传。

2.1 简介

文件上传,是指将本舆图片、视频、音频等文件上传到服务器,供其他用户浏览或下载的过程。
文件上传在项目中应用非常广泛,我们经常发微博、发微信朋侪圈都用到了文件上传功能。

   在我们的案例中,在新增员工的时间,要上传员工的头像,此时就会涉及到文件上传的功能。在举行文件上传时,我们点击加号或者是点击图片,就可以选择手机或者是电脑本地的图片文件了。当我们选择了某一个图片文件之后,这个文件就会上传到服务器,从而完成文件上传的操作。
  想要完成文件上传这个功能需要涉及到两个部分:

  • 前端步伐
  • 服务端步伐

我们先来看看在前端步伐中要完成哪些代码:
  1. <form action="/upload" method="post" enctype="multipart/form-data">
  2.         姓名: <input type="text" name="username"><br>
  3.     年龄: <input type="text" name="age"><br>
  4.     头像: <input type="file" name="image"><br>
  5.     <input type="submit" value="提交">
  6. </form>
复制代码
上传文件的原始form表单,要求表单必须具备以下三点(上传文件页面三要素):


  • 表单必须有file域,用于选择要上传的文件
          
    1. <input type="file" name="image"/>
    复制代码
  • 表单提交方式必须为POST
           通常上传的文件会比较大,所以需要利用 POST 提交方式
  • 表单的编码类型enctype必须要设置为:multipart/form-data
           普通默认的编码格式是不适合传输大型的二进制数据的,所以在文件上传时,表单的编码格式必须设置为multipart/form-data


前端页面的3要素我们相识后,接下来我们就来验证下所解说的文件上传3要素。
在提供的"课程资料"中有一个名叫"文件上传"的文件夹,直接将里的"upload.html"文件,复制到springboot项目工程下的static目录里面。



下面我们来验证:删除form表单中enctype属性值,会是什么环境?

  • 在IDEA中直接利用浏览器打开upload.html页面



  • 选择要上传的本地文件



  • 点击"提交"按钮,进入到开发者模式观察



我们再来验证:设置form表单中enctype属性值为multipart/form-data,会是什么环境?
  1. <form action="/upload" method="post" enctype="multipart/form-data">
  2.         姓名: <input type="text" name="username"><br>
  3.         年龄: <input type="text" name="age"><br>
  4.         头像: <input type="file" name="image"><br>
  5.         <input type="submit" value="提交">
  6.     </form>
复制代码
  1. [/code]
  2. 知道了前端步伐中需要设置上传文件页面三要素,那我们的后端步伐又是如何实现的呢?
  3. [list]
  4. [*] 起首在服务端定义这么一个controller,用来举行文件上传,然后在controller当中定义一个方法来处理/upload 请求
  5. [*] 在定义的方法中接收提交过来的数据 (方法中的形参名和请求参数的名字保持同等)
  6. [list]
  7. [*] 用户名:String name
  8. [*] 年龄: Integer age
  9. [*] 文件: MultipartFile image
  10. [/list]       Spring中提供了一个API:MultipartFile,利用这个API就可以来接收到上传的文件
  11.    
  12. [/list] [align=center][img=722,342]https://i-blog.csdnimg.cn/direct/c4afdeb7f8c7466981b41a30484f5325.png[/img][/align] 
  13.    问题:如果表单项的名字和方法中形参名差别等,该怎么办?
  14.   
  15. [list]
  16. [*] [code]public Result upload(String username,
  17.                     Integer age,
  18.                     MultipartFile file) //file形参名和请求参数名image不一致
复制代码
  解决:利用@RequestParam注解举行参数绑定
  

    1. public Result upload(String username,
    2.                     Integer age,
    3.                     @RequestParam("image") MultipartFile file)
    复制代码
  
UploadController代码:
  1. @Slf4j
  2. @RestController
  3. public class UploadController {
  4.    @PostMapping("/upload")
  5.    public Result upload(String username, Integer age, MultipartFile image)  {
  6.        log.info("文件上传:{},{},{}",username,age,image);
  7.        return Result.success();
  8.    }
  9. }
复制代码
  后端步伐编写完成之后,打个断点,以debug方式启动SpringBoot项目
  
 
   打开浏览器输入:http://localhost:8080/upload.html , 录入数据并提交
   

 
通过后端步伐控制台可以看到,上传的文件是存放在一个临时目录
 
   打开临时目录可以看到以下内容:
  
 
   表单提交的三项数据(姓名、年龄、文件),分别存储在差别的临时文件中:
  
 
   当我们步伐运行完毕之后,这个临时文件会自动删除。
  所以,我们如果想要实现文件上传,需要将这个临时文件,要转存到我们的磁盘目录中。
  
2.2 本地存储

前面我们已分析了文件上传功能前端和后端的基础代码实现,文件上传时在服务端会产生一个临时文件,请求相应完成之后,这个临时文件被自动删除,并没有举行保存。下面呢,我们就需要完成将上传的文件保存在服务器的本地磁盘上。
代码实现:

  • 在服务器本地磁盘上创建images目录,用来存储上传的文件(例:E盘创建images目录)
  • 利用MultipartFile类提供的API方法,把临时文件转存到本地磁盘目录下
   MultipartFile 常见方法:
  

  • String getOriginalFilename(); //获取原始文件名
  • void transferTo(File dest); //将接收的文件转存到磁盘文件中
  • long getSize(); //获取文件的大小,单位:字节
  • byte[] getBytes(); //获取文件内容的字节数组
  • InputStream getInputStream(); //获取接收到的文件内容的输入流
  1. @Slf4j
  2. @RestController
  3. public class UploadController {
  4.     @PostMapping("/upload")
  5.     public Result upload(String username, Integer age, MultipartFile image) throws IOException {
  6.         log.info("文件上传:{},{},{}",username,age,image);
  7.         //获取原始文件名
  8.         String originalFilename = image.getOriginalFilename();
  9.         //将文件存储在服务器的磁盘目录
  10.         image.transferTo(new File("E:/images/"+originalFilename));
  11.         return Result.success();
  12.     }
  13. }
复制代码
利用postman测试:
   注意:请求参数名和controller方法形参名保持同等
  
 
通过postman测试,我们发现文件上传是没有问题的。但是由于我们是利用原始文件名作为所上传文件的存储名字,当我们再次上传一个名为1.jpg文件时,发现会把之前已经上传成功的文件覆盖掉。

解决方案:保证每次上传文件时文件名都唯一的(利用UUID获取随机文件名)
  1. @Slf4j
  2. @RestController
  3. public class UploadController {
  4.    @PostMapping("/upload")
  5.    public Result upload(String username, Integer age, MultipartFile image) throws IOException {
  6.        log.info("文件上传:{},{},{}",username,age,image);
  7.        //获取原始文件名
  8.        String originalFilename = image.getOriginalFilename();
  9.        //构建新的文件名
  10.        String extname = originalFilename.substring(originalFilename.lastIndexOf("."));//文件扩展名
  11.        String newFileName = UUID.randomUUID().toString()+extname;//随机名+文件扩展名
  12.        //将文件存储在服务器的磁盘目录
  13.        image.transferTo(new File("E:/images/"+newFileName));
  14.        return Result.success();
  15.    }
  16. }
复制代码
在解决了文件名唯一性的问题后,我们再次上传一个较大的文件(超出1M)时发现,后端步伐报错:
 
报错原因呢是因为:在SpringBoot中,文件上传时默认单个文件最大大小为1M
那么如果需要上传大文件,可以在application.properties举行如下设置:
  1. #配置单个文件最大上传大小
  2. spring.servlet.multipart.max-file-size=10MB
  3. #配置单个请求最大上传大小(一次请求可以上传多个文件)
  4. spring.servlet.multipart.max-request-size=100MB
复制代码

到时此,我们文件上传的本地存储方式已完成了。但是这种本地存储方式还存在一问题:

如果直接存储在服务器的磁盘目录中,存在以下缺点:


  • 不安全:磁盘如果粉碎,所有的文件就会丢失
  • 容量有限:如果存储大量的图片,磁盘空间有限(磁盘不可能无限定扩容)
  • 无法直接访问
为相识决上述问题呢,通常有两种解决方案:


  • 自己搭建存储服务器,如:fastDFS 、MinIO
  • 利用现成的云服务,如:阿里云,腾讯云,华为云



2.3 阿里云OSS

2.3.1 准备

阿里云是阿里巴巴集团旗下全球领先的云盘算公司,也是国内最大的云服务提供商 。

   云服务指的就是通过互联网对外提供的各种各样的服务,比如像:语音服务、短信服务、邮件服务、视频直播服务、笔墨识别服务、对象存储服务等等。
  当我们在项目开发时需要用到某个或某些服务,就不需要自己来开发了,可以直接利用阿里云提供好的这些现成服务就可以了。比如:在项目开发当中,我们要实现一个短信发送的功能,如果我们项目组自己实现,将会非常繁琐,因为你需要和各个运营商举行对接。而此时阿里云完成了和三大运营商对接,并对外提供了一个短信服务。我们项目组只需要调用阿里云提供的短信服务,就可以很方便的来发送短信了。这样就低落了我们项目标开发难度,同时也提高了项目标开发效率。(大白话:别人帮我们实现好了功能,我们只要调用即可)
  云服务提供商给我们提供的软件服务通常是需要收取一部分费用的。
  阿里云对象存储OSS(Object Storage Service),是一款海量、安全、低成本、高可靠的云存储服务。利用OSS,您可以通过网络随时存储和调用包括文本、图片、音频和视频等在内的各种文件。

在我们利用了阿里云OSS对象存储服务之后,我们的项目当中如果涉及到文件上传这样的业务,在前端举行文件上传并请求到服务端时,在服务器本地磁盘当中就不需要再来存储文件了。我们直接将接收到的文件上传到oss,由 oss帮我们存储和管理,同时阿里云的oss存储服务还保障了我们所存储内容的安全可靠。


那我们学习利用这类云服务,我们主要学习什么呢?着实我们主要学习的是如何在项目当中来利用云服务完成具体的业务功能。而无论利用什么样的云服务,阿里云也好,腾讯云、华为云也罢,在利用第三方的服务时,操作的思绪都是一样的。

   SDK:Software Development Kit 的缩写,软件开发工具包,包括辅助软件开发的依赖(jar包)、代码示例等,都可以叫做SDK。
  简朴说,sdk中包罗了我们利用第三方云服务时所需要的依赖,以及一些示例代码。我们可以参照sdk所提供的示例代码就可以完成入门步伐。
  第三方服务利用的通用思绪,我们做一个简朴介绍之后,接下来我们就来介绍一下我们当前要利用的阿里云oss对象存储服务具体的利用步骤。

   Bucket:存储空间是用户用于存储对象(Object,就是文件)的容器,所有的对象都必须从属于某个存储空间。
  下面我们根据之前介绍的利用步骤,完成准备工作:

  • 注册阿里云账户(注册完成后需要实名认证)
  • 注册完账号之后,就可以登录阿里云


  • 通过控制台找到对象存储OSS服务

   如果是第一次访问,还需要开通对象存储服务OSS
  
 

  • 开通OSS服务之后,就可以进入到阿里云对象存储的控制台


  • 点击左侧的 "Bucket列表",创建一个Bucket

   大家可以参照"资料\04. 阿里云oss"中提供的文档,开通阿里云OSS服务。
  


2.3.2 入门

阿里云oss 对象存储服务的准备工作我们已经完成了,接下来我们就来完成第二步操作:参照官方所提供的sdk示例来编写入门步伐。
起首我们需要来打开阿里云OSS的官方文档,在官方文档中找到 SDK 的示例代码:

   如果是在实际开发当中,我们是需要从前以后仔细的去阅读这一份文档的,但是由于现在是讲授,我们就只挑重点的去看。有兴趣的同砚大家下来也可以自己去看一下这份官方文档。
  
 
参照官方提供的SDK,改造一下,即可实现文件上传功能:
  1. [/code]   在以上代码中,需要替换的内容为:
  2.   
  3. [list]
  4. [*] accessKeyId:阿里云账号AccessKey
  5. [*] accessKeySecret:阿里云账号AccessKey对应的秘钥
  6. [*] bucketName:Bucket名称
  7. [*] objectName:对象名称,在Bucket中存储的对象的名称
  8. [*] filePath:文件路径
  9. [/list]  AccessKey :[align=center][img=690,153]https://i-blog.csdnimg.cn/direct/05c4947015cd4d9b906a9d4fec3575f8.png[/img][/align]
  10.   运行以上步伐后,会把本地的文件上传到阿里云OSS服务器上:[align=center][img=733,290]https://i-blog.csdnimg.cn/direct/8f5c0611e3994913bc675234ccfb607d.png[/img][/align]
  11. [size=2]2.3.3 集成[/size]
  12. 阿里云oss对象存储服务的准备工作以及入门步伐我们都已经完成了,接下来我们就需要在案例当中集成oss对象存储服务,来存储和管理案例中上传的图片。
  13. [code]import com.aliyun.oss.ClientException;
  14. import com.aliyun.oss.OSS;
  15. import com.aliyun.oss.OSSClientBuilder;
  16. import com.aliyun.oss.OSSException;
  17. import com.aliyun.oss.model.PutObjectRequest;
  18. import com.aliyun.oss.model.PutObjectResult;
  19. import java.io.FileInputStream;
  20. import java.io.InputStream;
  21. public class AliOssTest {
  22.     public static void main(String[] args) throws Exception {
  23.         // Endpoint以华东1(杭州)为例,其它Region请按实际情况填写。
  24.         String endpoint = "oss-cn-shanghai.aliyuncs.com";
  25.         
  26.         // 阿里云账号AccessKey拥有所有API的访问权限,风险很高。强烈建议您创建并使用RAM用户进行API访问或日常运维,请登录RAM控制台创建RAM用户。
  27.         String accessKeyId = "LTAI5t9MZK8iq5T2Av5GLDxX";
  28.         String accessKeySecret = "C0IrHzKZGKqU8S7YQcevcotD3Zd5Tc";
  29.         
  30.         // 填写Bucket名称,例如examplebucket。
  31.         String bucketName = "web-framework01";
  32.         // 填写Object完整路径,完整路径中不能包含Bucket名称,例如exampledir/exampleobject.txt。
  33.         String objectName = "1.jpg";
  34.         // 填写本地文件的完整路径,例如D:\\localpath\\examplefile.txt。
  35.         // 如果未指定本地路径,则默认从示例程序所属项目对应本地路径中上传文件流。
  36.         String filePath= "C:\\Users\\Administrator\\Pictures\\1.jpg";
  37.         // 创建OSSClient实例。
  38.         OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);
  39.         try {
  40.             InputStream inputStream = new FileInputStream(filePath);
  41.             // 创建PutObjectRequest对象。
  42.             PutObjectRequest putObjectRequest = new PutObjectRequest(bucketName, objectName, inputStream);
  43.             // 设置该属性可以返回response。如果不设置,则返回的response为空。
  44.             putObjectRequest.setProcess("true");
  45.             // 创建PutObject请求。
  46.             PutObjectResult result = ossClient.putObject(putObjectRequest);
  47.             // 如果上传成功,则返回200。
  48.             System.out.println(result.getResponse().getStatusCode());
  49.         } catch (OSSException oe) {
  50.             System.out.println("Caught an OSSException, which means your request made it to OSS, "
  51.                     + "but was rejected with an error response for some reason.");
  52.             System.out.println("Error Message:" + oe.getErrorMessage());
  53.             System.out.println("Error Code:" + oe.getErrorCode());
  54.             System.out.println("Request ID:" + oe.getRequestId());
  55.             System.out.println("Host ID:" + oe.getHostId());
  56.         } catch (ClientException ce) {
  57.             System.out.println("Caught an ClientException, which means the client encountered "
  58.                     + "a serious internal problem while trying to communicate with OSS, "
  59.                     + "such as not being able to access the network.");
  60.             System.out.println("Error Message:" + ce.getMessage());
  61.         } finally {
  62.             if (ossClient != null) {
  63.                 ossClient.shutdown();
  64.             }
  65.         }
  66.     }
  67. }
复制代码
  在新增员工的时间,上传员工的图像,而之所以需要上传员工的图像,是因为将来我们需要在体系页面当中访问并展示员工的图像。而要想完成这个操作,需要做两件事:
  

  • 需要上传员工的图像,并把图像保存起来(存储到阿里云OSS)
  • 访问员工图像(通过图像在阿里云OSS的存储地点访问图像)

    • OSS中的每一个文件都会分配一个访问的url,通过这个url就可以访问到存储在阿里云上的图片。所以需要把url返回给前端,这样前端就可以通过url获取到图像。

  
我们参照接口文档来开发文件上传功能:


  • 基本信息
    1. 请求路径:/upload
    2. 请求方式:POST
    3. 接口描述:上传图片接口
    复制代码
  • 请求参数
    参数格式:multipart/form-data
    参数说明:
       参数名称参数类型是否必须示例备注imagefile是
  • 相应数据
    参数格式:application/json
    参数说明:
       参数名类型是否必须备注codenumber必须相应码,1 代表成功,0 代表失败msgstring非必须提示信息dataobject非必须返回的数据,上传图片的访问路径相应数据样例:
    1. {
    2.     "code": 1,
    3.     "msg": "success",
    4.     "data": "https://web-framework.oss-cn-hangzhou.aliyuncs.com/2022-09-02-00-27-0400.jpg"
    5. }
    复制代码

引入阿里云OSS上传文件工具类(由官方的示例代码改造而来)
  1. import com.aliyun.oss.OSS;
  2. import com.aliyun.oss.OSSClientBuilder;
  3. import org.springframework.stereotype.Component;
  4. import org.springframework.web.multipart.MultipartFile;
  5. import java.io.IOException;
  6. import java.io.InputStream;
  7. import java.util.UUID;
  8. @Component
  9. public class AliOSSUtils {
  10.     private String endpoint = "https://oss-cn-shanghai.aliyuncs.com";
  11.     private String accessKeyId = "LTAI5t9MZK8iq5T2Av5GLDxX";
  12.     private String accessKeySecret = "C0IrHzKZGKqU8S7YQcevcotD3Zd5Tc";
  13.     private String bucketName = "web-framework01";
  14.     /**
  15.      * 实现上传图片到OSS
  16.      */
  17.     public String upload(MultipartFile multipartFile) throws IOException {
  18.         // 获取上传的文件的输入流
  19.         InputStream inputStream = multipartFile.getInputStream();
  20.         // 避免文件覆盖
  21.         String originalFilename = multipartFile.getOriginalFilename();
  22.         String fileName = UUID.randomUUID().toString() + originalFilename.substring(originalFilename.lastIndexOf("."));
  23.         //上传文件到 OSS
  24.         OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);
  25.         ossClient.putObject(bucketName, fileName, inputStream);
  26.         //文件访问路径
  27.         String url = endpoint.split("//")[0] + "//" + bucketName + "." + endpoint.split("//")[1] + "/" + fileName;
  28.         // 关闭ossClient
  29.         ossClient.shutdown();
  30.         return url;// 把上传到oss的路径返回
  31.     }
  32. }
复制代码
修改UploadController代码:
  1. import com.itheima.pojo.Result;
  2. import com.itheima.utils.AliOSSUtils;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.web.bind.annotation.PostMapping;
  6. import org.springframework.web.bind.annotation.RestController;
  7. import org.springframework.web.multipart.MultipartFile;
  8. import java.io.IOException;
  9. @Slf4j
  10. @RestController
  11. public class UploadController {
  12.     @Autowired
  13.     private AliOSSUtils aliOSSUtils;
  14.     @PostMapping("/upload")
  15.     public Result upload(MultipartFile image) throws IOException {
  16.         //调用阿里云OSS工具类,将上传上来的文件存入阿里云
  17.         String url = aliOSSUtils.upload(image);
  18.         //将图片上传完成后的url返回,用于浏览器回显展示
  19.         return Result.success(url);
  20.     }
  21.    
  22. }
复制代码
利用postman测试:






3. 修改员工

需求:修改员工信息

在举行修改员工信息的时间,我们起首先要根据员工的ID查询员工的信息用于页面回显展示,然后用户修改员工数据之后,点击保存按钮,就可以将修改的数据提交到服务端,保存到数据库。 具体操作为:

  • 根据ID查询员工信息
  • 保存修改的员工信息

3.1 查询回显

3.1.1 接口文档

根据ID查询员工数据


  • 基本信息
    1. 请求路径:/emps/{id}
    2. 请求方式:GET
    3. 接口描述:该接口用于根据主键ID查询员工的信息
    复制代码
  • 请求参数
    参数格式:路径参数
    参数说明:
       参数名类型是否必须备注idnumber必须员工ID请求参数样例:
    1. /emps/1
    复制代码
  • 相应数据
    参数格式:application/json
    参数说明:
       名称类型是否必须默认值备注codenumber必须相应码, 1 成功 , 0 失败msgstring非必须提示信息dataobject必须返回的数据|- idnumber非必须id|- usernamestring非必须用户名|- namestring非必须姓名|- passwordstring非必须密码|- entrydatestring非必须入职日期|- gendernumber非必须性别 , 1 男 ; 2 女|- imagestring非必须图像|- jobnumber非必须职位, 说明: 1 班主任,2 讲师, 3 学工主管, 4 教研主管, 5 咨询师|- deptIdnumber非必须部门id|- createTimestring非必须创建时间|- updateTimestring非必须更新时间相应数据样例:
    1. {
    2.   "code": 1,
    3.   "msg": "success",
    4.   "data": {
    5.     "id": 2,
    6.     "username": "zhangwuji",
    7.     "password": "123456",
    8.     "name": "张无忌",
    9.     "gender": 1,
    10.     "image": "https://web-framework.oss-cn-hangzhou.aliyuncs.com/2022-09-02-00-27-53B.jpg",
    11.     "job": 2,
    12.     "entrydate": "2015-01-01",
    13.     "deptId": 2,
    14.     "createTime": "2022-09-01T23:06:30",
    15.     "updateTime": "2022-09-02T00:29:04"
    16.   }
    17. }
    复制代码

3.1.2 实现思绪


3.1.3 代码实现



  • EmpMapper
  1. @Mapper
  2. public interface EmpMapper {
  3.     //根据ID查询员工信息
  4.     @Select("select id, username, password, name, gender, image, job, entrydate, dept_id, create_time, update_time " +
  5.             "from emp " +
  6.             "where id = #{id}")
  7.     public Emp findById(Integer id);
  8.    
  9.     //省略...
  10. }
复制代码


  • EmpService
  1. public interface EmpService {
  2.     /**
  3.      * 根据ID查询员工
  4.      * @param id
  5.      * @return
  6.      */
  7.     public Emp getById(Integer id);
  8.    
  9.     //省略...
  10. }
复制代码


  • EmpServiceImpl
  1. @Slf4j
  2. @Service
  3. public class EmpServiceImpl implements EmpService {
  4.     @Autowired
  5.     private EmpMapper empMapper;
  6.     @Override
  7.     public Emp getById(Integer id) {
  8.         return empMapper.findById(id);
  9.     }
  10.    
  11.     //省略...
  12. }
复制代码


  • EmpController
  1. @Slf4j
  2. @RestController
  3. @RequestMapping("/emps")
  4. public class EmpController {
  5.    @Autowired
  6.    private EmpService empService;
  7.    //根据id查询
  8.    @GetMapping("/{id}")
  9.    public Result getById(@PathVariable Integer id){
  10.        Emp emp = empService.getById(id);
  11.        return Result.success(emp);
  12.    }
  13.    
  14.    //省略...
  15. }
复制代码

3.1.4 postman测试




3.2 修改员工


   当用户修改完数据之后,点击保存按钮,就需要将数据提交到服务端,然后服务端需要将修改后的数据更新到数据库中。
  3.2.1 接口文档



  • 基本信息
    1. 请求路径:/emps
    2. 请求方式:PUT
    3. 接口描述:该接口用于修改员工的数据信息
    复制代码
  • 请求参数
    参数格式:application/json
    参数说明:
       名称类型是否必须备注idnumber必须idusernamestring必须用户名namestring必须姓名gendernumber必须性别, 说明: 1 男, 2 女imagestring非必须图像deptIdnumber非必须部门identrydatestring非必须入职日期jobnumber非必须职位, 说明: 1 班主任,2 讲师, 3 学工主管, 4 教研主管, 5 咨询师请求数据样例:
    1. {
    2.   "id": 1,
    3.   "image": "https://web-framework.oss-cn-hangzhou.aliyuncs.com/2022-09-03-07-37-38222.jpg",
    4.   "username": "linpingzhi",
    5.   "name": "林平之",
    6.   "gender": 1,
    7.   "job": 1,
    8.   "entrydate": "2022-09-18",
    9.   "deptId": 1
    10. }
    复制代码
  • 相应数据
    参数格式:application/json
    参数说明:
       参数名类型是否必须备注codenumber必须相应码,1 代表成功,0 代表失败msgstring非必须提示信息dataobject非必须返回的数据相应数据样例:
    1. {
    2.     "code":1,
    3.     "msg":"success",
    4.     "data":null
    5. }
    复制代码

3.2.2 实现思绪


3.2.3 代码实现



  • EmpMapper
  1. @Mapper
  2. public interface EmpMapper {
  3.     //修改员工信息
  4.     public void update(Emp emp);
  5.    
  6.     //省略...
  7. }
复制代码


  • EmpMapper.xml
  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  4.        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.itheima.mapper.EmpMapper">
  6.    <!--更新员工信息-->
  7.    <update id="update">
  8.        update emp
  9.        <set>
  10.            <if test="username != null and username != ''">
  11.                username = #{username},
  12.            </if>
  13.            <if test="password != null and password != ''">
  14.                password = #{password},
  15.            </if>
  16.            <if test="name != null and name != ''">
  17.                name = #{name},
  18.            </if>
  19.            <if test="gender != null">
  20.                gender = #{gender},
  21.            </if>
  22.            <if test="image != null and image != ''">
  23.                image = #{image},
  24.            </if>
  25.            <if test="job != null">
  26.                job = #{job},
  27.            </if>
  28.            <if test="entrydate != null">
  29.                entrydate = #{entrydate},
  30.            </if>
  31.            <if test="deptId != null">
  32.                dept_id = #{deptId},
  33.            </if>
  34.            <if test="updateTime != null">
  35.                update_time = #{updateTime}
  36.            </if>
  37.        </set>
  38.        where id = #{id}
  39.    </update>
  40.    <!-- 省略... -->
  41.  
  42. </mapper>
复制代码


  • EmpService
  1. public interface EmpService {
  2.     /**
  3.      * 更新员工
  4.      * @param emp
  5.      */
  6.     public void update(Emp emp);
  7.    
  8.     //省略...
  9. }
复制代码


  • EmpServiceImpl
  1. @Slf4j
  2. @Service
  3. public class EmpServiceImpl implements EmpService {
  4.     @Autowired
  5.     private EmpMapper empMapper;
  6.     @Override
  7.     public void update(Emp emp) {
  8.         emp.setUpdateTime(LocalDateTime.now()); //更新修改时间为当前时间
  9.         
  10.         empMapper.update(emp);
  11.     }
  12.    
  13.     //省略...
  14. }
复制代码


  • EmpController
  1. @Slf4j
  2. @RestController
  3. @RequestMapping("/emps")
  4. public class EmpController {
  5.    @Autowired
  6.    private EmpService empService;
  7.    //修改员工
  8.    @PutMapping
  9.    public Result update(@RequestBody Emp emp){
  10.        empService.update(emp);
  11.        return Result.success();
  12.    }
  13.    
  14.    //省略...
  15. }
复制代码

3.2.4 postman测试


3.2.5 前后端联调测试






4. 设置文件

员工管理的增删改查功能我们已开发完成,但在我们所开发的步伐中还一些小问题,下面我们就来分析一下当前案例中存在的问题以及如何优化解决。
4.1 参数设置化


在我们之前编写的步伐中举行文件上传时,需要调用AliOSSUtils工具类,将文件上传到阿里云OSS对象存储服务当中。而在调用工具类举行文件上传时,需要一些参数:


  • endpoint //阿里云OSS域名
  • accessKeyID //用户身份ID
  • accessKeySecret //用户密钥
  • bucketName //存储空间的名字

关于以上的这些阿里云相关设置信息,我们是直接写死在java代码中了(硬编码),如果我们在做项目时每涉及到一个第三方技术服务,就将其参数硬编码,那么在Java步伐中会存在两个问题:

  • 如果这些参数发生变革了,就必须在源步伐代码中改动这些参数,然后需要重新举行代码的编译,将Java代码编译成class字节码文件再重新运行步伐。(比较繁琐)
  • 如果我们开发的是一个真实的企业级项目, Java类可能会有很多,如果将这些参数分散的定义在各个Java类当中,我们要修改一个参数值,我们就需要在众多的Java代码当中来定位到对应的位置,再来修改参数,修改完毕之后再重新编译再运行。(参数设置过于分散,是不方便集中的管理和维护)

为相识决以上分析的问题,我们可以将参数设置在设置文件中。如下:
  1. #自定义的阿里云OSS配置信息
  2. aliyun.oss.endpoint=https://oss-cn-hangzhou.aliyuncs.com
  3. aliyun.oss.accessKeyId=LTAI4GCH1vX6DKqJWxd6nEuW
  4. aliyun.oss.accessKeySecret=yBshYweHOpqDuhCArrVHwIiBKpyqSL
  5. aliyun.oss.bucketName=web-tlias
复制代码

在将阿里云OSS设置参数交给properties设置文件来管理之后,我们的AliOSSUtils工具类就变为以下情势:
  1. @Component
  2. public class AliOSSUtils {
  3.    /*以下4个参数没有指定值(默认值:null)*/
  4.    private String endpoint;
  5.    private String accessKeyId;
  6.    private String accessKeySecret;
  7.    private String bucketName;
  8.    //省略其他代码...
  9. }
复制代码
  而此时如果直接调用AliOSSUtils类当中的upload方法举行文件上传时,这4项参数全部为null,原因是因为并没有给它赋值。
  此时我们是不是需要将设置文件当中所设置的属性值读取出来,并分别赋值给AliOSSUtils工具类当中的各个属性呢?那应该怎么做呢?
  
因为application.properties是springboot项目默认的设置文件,所以springboot步伐在启动时会默认读取application.properties设置文件,而我们可以利用一个现成的注解:@Value,获取设置文件中的数据。
@Value 注解通常用于外部设置的属性注入,具体用法为: @Value("${设置文件中的key}")
  1. @Component
  2. public class AliOSSUtils {
  3.    @Value("${aliyun.oss.endpoint}")
  4.    private String endpoint;
  5.    
  6.    @Value("${aliyun.oss.accessKeyId}")
  7.    private String accessKeyId;
  8.    
  9.    @Value("${aliyun.oss.accessKeySecret}")
  10.    private String accessKeySecret;
  11.    
  12.    @Value("${aliyun.oss.bucketName}")
  13.    private String bucketName;
  14.    
  15.     //省略其他代码...
  16. }  
复制代码
  1. [/code] 利用postman测试:[align=center][img=729,296]https://i-blog.csdnimg.cn/direct/5e20e43436204091a5402a27f5afc603.png[/img][/align]
  2. [size=3]4.2 yml设置文件[/size]
  3. 前面我们一直利用springboot项目创建完毕后自带的application.properties举行属性的设置,那着实呢,在springboot项目当中是支持多种设置方式的,除了支持properties设置文件以外,还支持另外一种类型的设置文件,就是我们接下来要解说的yml格式的设置文件。
  4. [list]
  5. [*] application.properties
  6. [code]server.port=8080
  7. server.address=127.0.0.1
复制代码
  • application.yml
    1. server:
    2.   port: 8080
    3.   address: 127.0.0.1
    复制代码
  • application.yaml
    1. server:
    2.   port: 8080
    3.   address: 127.0.0.1
    复制代码
       yml 格式的设置文件,后缀名有两种:
      

    • yml (保举)
    • yaml
      常见设置文件格式对比:

    我们可以看到设置同样的数据信息,yml格式的数据有以下特点:


    • 容易阅读
    • 容易与脚本语言交互
    • 以数据为焦点,重数据轻格式

    简朴的相识过springboot所支持的设置文件,以及差别类型设置文件之间的优缺点之后,接下来我们就来相识下yml设置文件的基本语法:


    • 大小写敏感
    • 数值前边必须有空格,作为分隔符
    • 利用缩进表示层级关系,缩进时,不答应利用Tab键,只能用空格(idea中会自动将Tab转换为空格)
    • 缩进的空格数目不告急,只要相同层级的元素左侧对齐即可
    • #表示注释,从这个字符一直到行尾,都会被解析器忽略


    相识完yml格式设置文件的基本语法之后,接下来我们再来看下yml文件中常见的数据格式。在这里我们主要介绍最为常见的两类:

    • 定义对象或Map集合
    • 定义数组、list或set集合
    对象/Map集合
    1. user:
    2.   name: zhangsan
    3.   age: 18
    4.   password: 123456
    复制代码
    数组/List/Set集合
    1. hobby:
    2.   - java
    3.   - game
    4.   - sport
    复制代码

    熟悉完了yml文件的基本语法后,我们修改下之前案例中利用的设置文件,变更为application.yml设置方式:

    • 修改application.properties名字为:_application.properties(名字随便更换,只要加载不到即可)
    • 创建新的设置文件: application.yml
    原有application.properties文件:

    新建的application.yml文件:
    1. spring:
    2.   datasource:
    3.     driver-class-name: com.mysql.cj.jdbc.Driver
    4.     url: jdbc:mysql://localhost:3306/tlias
    5.     username: root
    6.     password: 1234
    7.   servlet:
    8.     multipart:
    9.       max-file-size: 10MB
    10.       max-request-size: 100MB
    11.       
    12. mybatis:
    13.   configuration:
    14.     log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
    15.     map-underscore-to-camel-case: true
    16.        
    17. aliyun:
    18.   oss:
    19.     endpoint: https://oss-cn-hangzhou.aliyuncs.com
    20.     accessKeyId: LTAI4GCH1vX6DKqJWxd6nEuW
    21.     accessKeySecret: yBshYweHOpqDuhCArrVHwIiBKpyqSL
    22.     bucketName: web-397
    复制代码



    4.3 @ConfigurationProperties

    解说完了yml设置文件之后,最后再来介绍一个注解@ConfigurationProperties。在介绍注解之前,我们先来看一个场景,分析下代码当中可能存在的问题:

    我们在application.properties或者application.yml中设置了阿里云OSS的四项参数之后,如果java步伐中需要这四项参数数据,我们直接通过@Value注解来举行注入。这种方式本身没有什么问题问题,但是如果说需要注入的属性较多(例:需要20多个参数数据),我们写起来就会比较繁琐。
    那么有没有一种方式可以简化这些设置参数的注入呢?答案是肯定有,在Spring中给我们提供了一种简化方式,可以直接将设置文件中设置项的值自动的注入到对象的属性中。

    Spring提供的简化方式套路:

    • 需要创建一个实现类,且实体类中的属性名和设置文件当中key的名字必须要同等
             比如:设置文件当中叫endpoints,实体类当中的属性也得叫endpoints,另外实体类当中的属性还需要提供 getter / setter方法
    • 需要将实体类交给Spring的IOC容器管理,成为IOC容器当中的bean对象
    • 在实体类上添加@ConfigurationProperties注解,并通过perfect属性来指定设置参数项的前缀


    实体类:AliOSSProperties
    1. import lombok.Data;
    2. import org.springframework.boot.context.properties.ConfigurationProperties;
    3. import org.springframework.stereotype.Component;
    4. /*阿里云OSS相关配置*/
    5. @Data
    6. @Component
    7. @ConfigurationProperties(prefix = "aliyun.oss")
    8. public class AliOSSProperties {
    9.     //区域
    10.     private String endpoint;
    11.     //身份ID
    12.     private String accessKeyId ;
    13.     //身份密钥
    14.     private String accessKeySecret ;
    15.     //存储空间
    16.     private String bucketName;
    17. }
    复制代码

    AliOSSUtils工具类:
    1. import com.aliyun.oss.OSS;
    2. import com.aliyun.oss.OSSClientBuilder;
    3. import org.springframework.beans.factory.annotation.Autowired;
    4. import org.springframework.stereotype.Component;
    5. import org.springframework.web.multipart.MultipartFile;
    6. import java.io.IOException;
    7. import java.io.InputStream;
    8. import java.util.UUID;
    9. @Component //当前类对象由Spring创建和管理
    10. public class AliOSSUtils {
    11.     //注入配置参数实体类对象
    12.     @Autowired
    13.     private AliOSSProperties aliOSSProperties;
    14.    
    15.    
    16.     /**
    17.      * 实现上传图片到OSS
    18.      */
    19.     public String upload(MultipartFile multipartFile) throws IOException {
    20.         // 获取上传的文件的输入流
    21.         InputStream inputStream = multipartFile.getInputStream();
    22.         // 避免文件覆盖
    23.         String originalFilename = multipartFile.getOriginalFilename();
    24.         String fileName = UUID.randomUUID().toString() + originalFilename.substring(originalFilename.lastIndexOf("."));
    25.         //上传文件到 OSS
    26.         OSS ossClient = new OSSClientBuilder().build(aliOSSProperties.getEndpoint(),
    27.                 aliOSSProperties.getAccessKeyId(), aliOSSProperties.getAccessKeySecret());
    28.         ossClient.putObject(aliOSSProperties.getBucketName(), fileName, inputStream);
    29.         //文件访问路径
    30.         String url =aliOSSProperties.getEndpoint().split("//")[0] + "//" + aliOSSProperties.getBucketName() + "." + aliOSSProperties.getEndpoint().split("//")[1] + "/" + fileName;
    31.         // 关闭ossClient
    32.         ossClient.shutdown();
    33.         return url;// 把上传到oss的路径返回
    34.     }
    35. }
    复制代码

    在我们添加上注解后,会发现idea窗口上面出现一个红色警告:

    这个警告提示是告知我们还需要引入一个依赖:
    1. <dependency>
    2.     <groupId>org.springframework.boot</groupId>
    3.     <artifactId>spring-boot-configuration-processor</artifactId>
    4. </dependency>
    复制代码
    当我们在pom.xml文件当中设置了这项依赖之后,我们重新启动服务,大家就会看到在properties或者是yml设置文件当中,就会提示阿里云 OSS 相关的设置项。所以这项依赖它的作用就是会自动的识别被@Configuration Properties注解标识的bean对象。

       刚才的红色警告,已经变成了一个灰色的提示,提示我们需要重新运行springboot服务
      @ConfigurationProperties注解我们已经介绍完了,接下来我们就来区分一下@ConfigurationProperties注解以及我们前面所介绍的另外一个@Value注解:
    相同点:都是用来注入外部设置的属性的。
    差别点:


    • @Value注解只能一个一个的举行外部属性的注入。
    • @ConfigurationProperties可以批量的将外部的属性设置注入到bean对象的属性中。

    如果要注入的属性非常的多,而且还想做到复用,就可以定义这么一个bean对象。通过 configuration properties 批量的将外部的属性设置直接注入到 bin 对象的属性当中。在其他的类当中,我要想获取到注入进来的属性,我直接注入 bin 对象,然后调用 get 方法,就可以获取到对应的属性值了





    免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
  • 回复

    使用道具 举报

    0 个回复

    倒序浏览

    快速回复

    您需要登录后才可以回帖 登录 or 立即注册

    本版积分规则

    前进之路

    金牌会员
    这个人很懒什么都没写!
    快速回复 返回顶部 返回列表