• ​在线问题反馈模块实战(九)​:实现图片上传功能(下)


    👨‍🎓作者:bug菌

    ✏️博客:CSDN掘金

    💌公众号:猿圈奇妙屋

    🚫特别声明:原创不易,转载请附上原文出处链接和本文声明,谢谢配合。

    🙏版权声明:文章里可能部分文字或者图片来源于互联网或者百度百科,如有侵权请联系bug菌处理。

    一、前言🔥

           接下来的这几期,bug菌想跟大家分享一下自己昨天刚接到一个临时的需求,热乎着呢,想分享一下自己是如何面对临时需求并制定整个开发周期,其中包括从梳理业务到创建业务表再到实现业务逻辑形成闭环再到与前端对接,其中会穿插一些业务拓展及功能性拓展,这一条龙流程在线与大家一起见证,分享给刚入门的小伙伴,希望对你们有所帮助。

    环境说明:idea2019.3 + springboot2.3.1.REALSE + mybati-plus3.2.0 + mysql5.6 + jdk1.8

           若小伙伴们在批阅文章的过程中觉得文章对您有一丝丝帮助,还请别吝啬您手里的赞呀,大胆的把文章点亮👍吧,您的点赞三连(收藏⭐️+关注👨‍🎓+留言📃)就是对bug菌我创作道路上最好的鼓励与支持😘。时光不弃🏃🏻‍♀️,创作不停💕,加油☘️ 

    二、正文🔥

            上一期我们讲述了如何从一个业务梳理成具体的某些接口,再到接口如何定义,需要注意那些事项,不知道大家是否有记起来,然而这一期,我们就顺着已经确定的接口上实现业务逻辑,可能你们还会问,不就是对于该接口进行一个代码实现么,是的没错,但是我此处讲的肯定是经过了自己改了又改,删了又写最终得出来的,我就要拿最优的代码思路给大家做参考,但是也并不是说我这就是最优的,肯定都是有值得优化的地方,这里就需要发挥大家的思维与创造力了。

            好啦,咱们进入正题吧。手把手带着大家进行coding。

    三、接口逻辑实战🔥

            我们先是在Controller层定义了该问题反馈保存接口及访问接口路径等,具体定义如下:

    1. /**
    2. * 反馈问题保存
    3. *
    4. * @param images img图片数组
    5. * @param inPage 问题反馈/建议所在页面
    6. * @param questionContent 反问题反馈/建议描述
    7. * @param questionType 问题反馈类型(功能异常、体验问题、功能建议、其它)
    8. */
    9. @PostMapping("/save")
    10. @SysLog(logType = LogTypeEnum.LOG_TYPE_IMG_UPLOAD)
    11. @ApiOperation(value = "反馈问题保存", notes = "反馈问题保存")
    12. public ResultResponse saveQuestion(@ApiParam("图片数组") MultipartFile[] images,
    13. @ApiParam("问题反馈类型") String questionType,
    14. @ApiParam("问题反馈/建议描述") String questionContent,
    15. @ApiParam("问题反馈/建议所在页面") String inPage) throws IOException {
    16. return userQuestionsService.saveQuestion(images, questionType, questionContent, inPage);
    17. }

            接着就是实现saveQuestion()方法了。这里对于肯定是直接先定义到userQuestionsService接口层,然后通过实现该接口层进行方法的具体实现。

    定义接口方法如下:

    ResultResponse saveQuestion(MultipartFile[] images, String questionType, String questionContent, String inPage) throws IOException;

            这里我是直接反回了一个自定义类型,且自定义类型是个布尔值。

            然后就是该接口的的重点了,怎么说呢?这也是该接口的核心业务了,如何保存反馈内容,如何保存问题反馈图片数组。切记这里是支持多图片上传。

            接着就是通过UserQuestionsServiceImpl 通过implements关键字来实现 IUserQuestionsService 接口层的所有方法,其中也就包括实现saveQuestion()方法。

    1. @Override
    2. public ResultResponse saveQuestion(MultipartFile[] images, String questionType, String questionContent, String inPage) throws IOException {}

            对于这个业务实现,也是有点坑的。对于业务实现,我们一般要做的步骤就是,第一步干嘛,第二步干嘛...一定要先列举清楚,要不然写一点敲一点,最后肯定是有问题的。

    第一步:遍历MultipartFile[] images 图片数组,分别调用图片上传方法。

    第二步:实现图片上传方法。

    第三部:实现基础字段内容保存并方法返回。

            然后具体对于三步骤,我们也是要具体的业务思维,对于遍历图片保存,这里就需要有一点要注意,你应该要留意,我们一次问题反馈调用,就生成一条记录,但对于一次请求,对于多图片,我们怎么做到一对多呢?其实啊,这就在你定义表字段时,就应该考虑清楚,所以我这里就根据一个字段来处理,也就是在你进行保存图片之后,专属定义一个外循环变量来存放所有的图片保存地址,然后两图片地址之间,你可以用逗号隔开,也可以用别的字段间隔都可。

    imgPaths = imgPaths + "," + saveImgPath;

            就像如上所示,直接用逗号隔开,但是有一点是需要大家注意的,在你进行逗号隔开时,你要考虑如果MultipartFile[] images 图片数组为1,你就不应该有逗号间隔了啊。

            讲到这里,我也诺列的差不多了,想必小伙伴们心里都清楚如何实现了吧?

    第一步:遍历调用图片上传方法

    1. if (images.length > 0) {
    2. //分批处理图片保存
    3. for (int i = 1; i <= images.length; i++) {
    4. //当前图片
    5. MultipartFile img = images[i - 1];
    6. //获取文件后缀。
    7. String imageSuffix = FilenameUtils.getExtension(img.getOriginalFilename()).toLowerCase();
    8. //非以上文件后缀直接跳过。
    9. if (!CONTENT_TYPES.contains(imageSuffix)) {
    10. continue;
    11. }
    12. //获取当前时间进行命名。
    13. String template = TimeUtils.getPrivateTime(ConstantUtils.TIME_STAMP_BEFORE2) + "_" + i;
    14. //重写img文件名 eg:20220527100245.jpg
    15. String saveImgPath = template + "." + imageSuffix;
    16. //图片保存绝对地址
    17. String savePath = this.buildImgPath(saveImgPath, accountId);
    18. //图片上传
    19. uploader.uploadImg(img, savePath);
    20. //路径用逗号拼接。
    21. if (StringUtils.isNotEmpty(imgPaths)) {
    22. imgPaths = imgPaths + "," + saveImgPath;
    23. } else {
    24. imgPaths = saveImgPath;
    25. }
    26. }
    27. }

    注意:这里会涉及到两个比较坑的点。第一个就是对于同一个遍历方法,获取到的时间戳会是一致的,所以如果你想自定义别名图片,你就不能只通过时间戳来进行别名,你应该还得加盐,由于我是想区分用户那天上传的那些图片,才特定通过获取时间戳来进行别名,我这里提供的做法就是在时间戳后再拼接一个遍历下边,这样就保证了同一用户在一次接口调用反馈时,图片是按template_index来命名的。第二点就是需要留意对于图片数组=1的,你就不应该也逗号间隔,加一个判断即可。

    第二步:实现图片上传方法

            对于这一步,实现方式还是相对较多的,我这里为了给大家教学,就使用一种超级简单的实现方式,比如MultipartFile类自带的transferTo()方法。

    1. public void uploadImg(MultipartFile image, String savePath) throws IOException {
    2. File file = new File(savePath);
    3. //检测是否存在该目录
    4. if (!file.getParentFile().exists()) {
    5. file.getParentFile().mkdirs();
    6. }
    7. //写入文件
    8. image.transferTo(file);
    9. }

             这里需要注意的就是一点,要判断该路径存在与否,不存在还是得要进行路径的创建再进行文件写入。

    第三步:实现实体类数据库写入方法

            这一步就相对比较简单了,我们由于将已上传成功返回的图片保存地址进行了路径拼接,我们即可直接定义一个实体,进行数据赋值即可。

    UserQuestionsEntity entity = new UserQuestionsEntity(user, questionType, questionContent, inPage, imgPaths);

    然后在对应的实体类将字段内容进行手动赋值即可。

    1. public UserQuestionsEntity(SysUserEntity user, String questionType, String questionContent, String inPage, String imgPath) {
    2. this.status = UsualStatusEnum.NORMAL.getKey();
    3. this.creatorAccountId = user.getAccountId();
    4. this.creatorName = user.getAccountName();
    5. this.creatorDeptName = user.getDeptName();
    6. this.questionType = questionType;
    7. this.questionContent = questionContent;
    8. this.inPage = inPage;
    9. this.filePaths = imgPath;
    10. }

     这里就不需要再手动赋值那些创建时间、创建人等操作字段了,因为我们前边是已经添加了字段自动填充。

            最后就是直接调用mp提供的save方法即可。

    return new ResultResponse<>(this.save(entity));

    第四步:接口调用测试  

            最后,我们重启项目,通过postman进行接口调用测试。

     我们来检查一下具体保存的图片。

            很完美也是成功将图片保存到了指定位置下。

     ... ...

            好啦,以上就是这期的所有内容啦,你们学废了么?如果对你有所帮助,还请不要忘记给bug菌[三连支持]哟。如果想获得更多的学习资源或者想和更多的技术爱好者一起交流,可以关注我的公众号『猿圈奇妙屋』,后台回复关键词领取学习资料、大厂面经、面试模板等海量资源,就等你来拿。

    四、往期热文推荐🔥

            对于问题反馈模块实战开发,我完整的梳理了每一期的教学及链接地址,仅供参考:希望能对你们有所帮助。

    • 在线问题反馈模块实战(一):梳理业务需求并创建数据库表
    • 在线问题反馈模块实战(二):封装代码自动生成类文件器
    • 在线问题反馈模块实战(三):自动生成所有Controller、Service、Mapper等文件
    • 在线问题反馈模块实战(四):封装通用字段类
    • 在线问题反馈模块实战(五):实现对通用字段内容自动填充功能
    • 在线问题反馈模块实战(六):接口文档定义
    • 在线问题反馈模块实战(七):安装部署swagger2
    • ​在线问题反馈模块实战(八)​:实现图片上传功能(上)
    • ​在线问题反馈模块实战(九)​:实现图片上传功能(下)
    • ​在线问题反馈模块实战(十)​:实现图片预览功能
    • ​在线问题反馈模块实战(十一)​:实现图片下载功能
    • ​在线问题反馈模块实战(十二)​:实现图片删除功能
    • ​在线问题反馈模块实战(十三)​:实现多参数分页查询列表
    • 在线问题反馈模块实战(十四):实现在线答疑功能
    • 在线问题反馈模块实战(十五)​:实现在线更新反馈状态功能
    • 在线问题反馈模块实战(十六)​:实现查详情功能
    • 在线问题反馈模块实战(十七):实现excel模板在线下载功能
    • 在线问题反馈模块实战(十八):实现excel台账文件记录批量导入功能
    • 在线问题反馈模块实战(十九):实现数据批量导出到excel文件中功能
    • 在线问题反馈模块实战(二十):完结篇

            如上是整整二十期内容,每一期都是干货,对于一个模块的开发,如何一点一滴打造并测试部署上线,我再说一遍,这不是演习,是实战!是实战!是实战!

            若你们觉得只是需要了解其中某个知识点或者业务的话,也不反对,你就选择其中的几期进行学习就好,反正都已经完结啦;我只希望你们能有所收获,有所成长,也就不枉我苦心每天下班后给大家总结更新。

    五、文末🔥

            如果你还想要学习更多,小伙伴们大可关注bug菌专门为你们创建的专栏《springboot零基础入门教学》,都是我一手打下的江山,持续更新中,希望能帮助到更多小伙伴们。

           我是bug菌,一名想走👣出大山改变命运的程序猿。接下来的路还很长,都等待着我们去突破、去挑战。来吧,小伙伴们,我们一起加油!未来皆可期,fighting!

            最后送大家两句我很喜欢的话,与诸君共勉!


    ☘️做你想做的人,没有时间限制,只要愿意,什么时候都可以start。

    🍀你能从现在开始改变,也可以一成不变,这件事,没有规矩可言,你可以活出最精彩的自己。


    ​​​​​​

    💌如果文章对您有所帮助,就请留下您的吧!(#^.^#);

    💝如果喜欢bug菌分享的文章,就请给bug菌点个关注吧!(๑′ᴗ‵๑)づ╭❤~;

    💗如果对文章有任何疑问,还请文末留言或者加群吧【QQ交流群:708072830】;

    💞鉴于个人经验有限,所有观点及技术研点,如有异议,请直接回复参与讨论(请勿发表攻击言论,谢谢);

    💕版权声明:原创不易,转载请附上原文出处链接和本文声明,版权所有,盗版必究!!!谢谢。

  • 相关阅读:
    CTFshow 命令执行 web29 30 31
    第二十三课,抗锯齿(Anti Aliasing)
    基于JAVA中山学院教室管理系统计算机毕业设计源码+系统+数据库+lw文档+部署
    Scala in a Nutshell
    CentOS Linux 的安装
    Linux命令大全(超详细版)
    MySQL基础篇之多表查询(内连接、外连接、自连接、子查询、union)
    Calcite RelNode和RexNode 介绍
    尿苷二磷酸修饰阿拉伯糖,阿拉伯糖偶联核苷酸,UDP-B-L-阿拉伯糖二钠盐,15839-78-8
    物联网网关助力生产数据可视化,提升智能管理水平
  • 原文地址:https://blog.csdn.net/weixin_43970743/article/details/125147658