如何解决MultipartFile类的transferTo()上传图片报“系统找不到指定的路径”问题【亲测有效】

一、前言🔥

在使用MultipartFile类调用transferTo()方法时,报错了,如何解决,此文告诉你。

环境说明:springboot2.3.1.REALSE + mysql5.6 + jdk1.8

二、报错展示🔥

在使用MultipartFile类调用transferTo()方法时,却报错了。

如下是重点报错内容:

java.io.IOException: java.io.FileNotFoundException: C:\Users\86157\AppData\Local\Temp\tomcat.11159546904116402439.8889\work\Tomcat\localhost\review\.\template\question-img\admin\20220527101841.jpg (系统找不到指定的路径。)

如下是控制台报错完整截图:

三、报错排查🔥

不假思索的一看到这“系统找不到指定的路径”,肯定是路径问题呗,我以为是我传的路径地址不对,那我debug看看,看看到最后传入到transferTo()方法时的入参是啥。很明显,确实是这串相对地址啊。

    ![](https://ooolo.net/usr/uploads/2024/02/3107862054.png)
        emmm,好吧,还是报错了,我刨根问底,debug进transferTo()方法里,然后看到它有执行 this.part.write(dest.getpath 方法,进入write()方法内部,到这里,我们看到了我们想要的答案!你们看到了么?

如下是write()源码:

@Override
public void write(String fileName) throws IOException {
    File file = new File(fileName);
    if (!file.isAbsolute()) {
        file = new File(location, fileName);
    }
    try {
        fileItem.write(file);
    } catch (Exception e) {
        throw new IOException(e);
    }
}

这段方法非常明显,先是判断传入的参数是否是相对路径,如果是相对路径,它会自己给我们拼接一个父路径(location)! 所以你应该知道那个奇怪的路径是从哪里来的了吧!

好了,现在是知道了问题在哪了吧,罪魁祸首就是transferTo()参数,如果是相对路径的话,它会自己拼接一个父路径,因为指定的相对路径中拼接我自定义的路径已经变成了一个不会存在的路径,保存肯定是失败告终。但如果你传入的参数只是一个文件名,那应该就能保存成功。但是这样,取文件的时候,又会遇到问题了,你可能都不知道文件在哪里!所以最直观的解决方案就是,给定一个绝对路径,这样妥妥是不会有问题的。
四、解决方案🔥

所以综上,问题就在于你调用transferTo()方法,参数路径得定义一个绝对路径方可行。

接下来,为了验证我的观点是否正确,我给图片保存目录设置一个绝对路径。那我先这样,为了给大家看的更直观一点,我直接通过获取项目根路径然后再拼接自定义路径吧!

具体演示请看如下:

我是以postman测试上传了两张图片,然后如下是我debug第一次循环时的截图,大家可以看到,savePath地址这肯定是没问题的。因为我是通过System.getProperty("user.dir") 来获取的项目根目录,然后拼接自定义路径,如果路径不存在,我是有些自动创建目录。

    ![](https://ooolo.net/usr/uploads/2024/02/1262177626.png)

接下来,给你们最直观的验证,就是查看下该保存目录下是否已经上传了图片?

MultipartFile的transferto报错IO路径错误

问题描述:

在使用MultipartFile.transferto准备保存上传文件的时候报错IO路径错误(IO路径不存在),但是我肯定我指定的路径是存在的

出现原因:

MultipartFile.transferto(file)该方法内参数需要的是绝对路径而不是相对路径,如果我们传入的是相对路径的话,那么该方法就会去tomcat内的webapp下去寻找该相对路径。这样肯定是找不到对应路径的,所以会出现该报错

解决方案:

选择传入绝对路径而不是相对路径

file类有获取绝对值的方法getCanonicalPath()

修改后的代码

@PostMapping("/upload")
    public myResult upLoadFiles(MultipartFile[] files, HttpServletRequest request) throws IOException {
        // 1.获取所有文件的文件名 分配文件路径
        String parentPath = "uploadFiles";
        File parent = new File(parentPath);
        if (!parent.exists()){
            parent.mkdir();
            log.info("开始创建parent文件夹:{}",parent.getPath());
        }
        for (MultipartFile file : files){
            String fileName = file.getOriginalFilename();
            String new_FileName = parent.getCanonicalPath()+File.separator+fileName;
            log.info(new_FileName);
            try {
                file.transferTo(new File(new_FileName));
                log.info("文件{}上传成功!",new_FileName);
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
        return new myResult().ok();
    }

Ubuntu——系统版本、显存、CPU型号、GPU型号查询

一、系统版本查询
lsb_release -a

查询结果如下:系统版本为Ubuntu 20.04.6 LTS


二、CPU信息查询

cat /proc/cpuinfo


如上图所示,本机的CPU型号信息为 i7-11700k

三、显存查询

nvidia-smi


如上图所示,其显存为12G(12050MiB)

四、GPU显卡信息查询
如下图所示,NVIDIA GeForce RTX 3080 Ti 即为本机显卡型号信息
nvidia-smi -L

dobbo 集成seata框架

seata server:

#AT 模式
#seata 管理面板 http://192.168.xx:7091
seata:
  enabled: true
  application-id: oa-server
  tx-service-group: oa-server-group
  enable-auto-data-source-proxy: true
  client:
    support:
      spring:
        datasource-autoproxy: true
  registry:
    type: zk
    zk:
      server-addr: 192.168.xx:2181
      connect-timeout: 50000
      session-timeout: 60000
      username: ""
      password: ""
  service:
    grouplist:
      default: 192.168.xx:8091
    vgroup-mapping:
      oa-server-group: default

client:

#AT 模式
#seata 管理面板 http://192.168.xx:7091
seata:
  enabled: true
  application-id: oa-client-9999
  tx-service-group: oa-client-9999-group
  enable-auto-data-source-proxy: true
  client:
    support:
      spring:
        datasource-autoproxy: true
  registry:
    type: zk
    zk:
      server-addr: 192.168.xx:2181
      connect-timeout: 50000
      session-timeout: 60000
      username: ""
      password: ""
  service:
    grouplist:
      default: 192.168.xx:8091
    vgroup-mapping:
      oa-client-9999-group: default

seata代码要写在
servver端
注解用
@GlobalTransactional

特色功能:

两台Linux服务器之间怎么传输文件

scp

【优点】简单方便,安全可靠;支持限速参数

【缺点】不支持排除目录

【用法】

scp就是secure copy,是用来进行远程文件拷贝的。数据传输使用 ssh,并且和ssh 使用相同的认证方式,提供相同的安全保证 。

命令格式:

scp [参数] <源地址(用户名@IP地址或主机名)>:<文件路径> <目的地址(用户名 @IP 地址或主机名)>:<文件路径>

举例:

scp /home/work/source.txt work@192.168.0.10:/home/work/ #把本地的source.txt文件拷贝到192.168.0.10机器上的/home/work目录下

scp work@192.168.0.10:/home/work/source.txt /home/work/ #把192.168.0.10机器上的source.txt文件拷贝到本地的/home/work目录下

scp work@192.168.0.10:/home/work/source.txt work@192.168.0.11:/home/work/ #把192.168.0.10机器上的source.txt文件拷贝到192.168.0.11机器的/home/work目录下

scp -r /home/work/sourcedir work@192.168.0.10:/home/work/ #拷贝文件夹,加-r参数

scp -r /home/work/sourcedir work@www.myhost.com:/home/work/ #使用主机名

scp -r -v /home/work/sourcedir work@www.myhost.com:/home/work/ #显示详情,加-v参数