Java文件上传实例并解决跨域问题

简介: 在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传功能的实现。

·.png

在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传功能的实现。

了解MultipartFile接口
我们实现文件的上传用到了Spring-web框架中的 MultipartFile接口,MultipartFile接口的源码注释中说“MultipartFile接口是 在大部分请求中接收的上载文件的表示形式。”

A representation of an uploaded file received in a multipart request.
The file contents are either stored in memory or temporarily on disk. In either case, the user is responsible for copying file contents to a session-level or persistent store as and if desired. The temporary storage will be cleared at the end of request processing.

常用方法如下表

Method Summary
byte[] 获取文件的字节数组 getBytes()

      Return the contents of the file as an array of bytes.

String 获取文件的类型 getContentType()

      Return the content type of the file.

InputStream 获取文件的输入流 getInputStream()

      Return an InputStream to read the contents of the file from.

String 获取文件名 getName()

      Return the name of the parameter in the multipart form.

String 获取原始文件名(防止篡改文件类型) getOriginalFilename()

      Return the original filename in the client's filesystem.

long 获取文件的大小,以字节的形式) getSize()

      Return the size of the file in bytes.

boolean 判断文件是否为空 isEmpty()

      Return whether the uploaded file is empty, that is, either no file has been chosen in the multipart form or the chosen file has no content.

void 将接收到的文件传输到给定的目标文件。 transferTo(File dest)

      Transfer the received file to the given destination file.

文件上传业务代码
Controller类
/**

  • @Author: 富贵论坛www.fgba.net
  • @Date: 2021/7/6 - 20:56
  • @Description: 文件上传
  • @version: 1.0

*/
@Controller
@RequestMapping("upload")
public class UploadController {

@Autowired
private UploadService uploadService;

@PostMapping("image")
public ResponseEntity<String> uploadImage(@RequestParam("file") MultipartFile file){
   String url= this.uploadService.uploadImage(file);
   if (StringUtils.isBlank(url)){
       return ResponseEntity.badRequest().build();
   }
    return ResponseEntity.status(HttpStatus.CREATED).body(url);
}

}
Service类:写了具体的业务逻辑
/**

  • @Author: 富贵论坛www.fgba.net
  • @Date: 2021/7/6 - 21:01
  • @Description: 文件上传
  • @version: 1.0

*/
@Service
public class UploadService {

//用于判断文件的类型,暂时只判断了“image/gif","image/jpeg”
private static final List<String> CONTENT_TYPES= Arrays.asList("image/gif","image/jpeg");

private static final Logger LOGGER= LoggerFactory.getLogger(UploadService.class);

/**
 * 业务逻辑代码
 * @param file 文件的存储的url
 * @return
 */
public String uploadImage(MultipartFile file) {

    String originalFilename = file.getOriginalFilename();
    //校验文件类型
    //方法一:截取字符串
    String afterLast = StringUtils.substringAfterLast(".", originalFilename);
    //方法二:使用getContentType方法
    String contentType = file.getContentType();
    if (!CONTENT_TYPES.contains(contentType)){
        LOGGER.info("文件类型不合法:"+originalFilename);
        return null;
    }
    //校验文件内容
    try {
        //获取文件流
        BufferedImage bufferedImage = ImageIO.read(file.getInputStream());
        if (bufferedImage==null){
            LOGGER.info("文件内容不合法:{}",originalFilename);
            return null;
        }
        //保存到服务器   E:\fgba\image
        //将接收到的文件传输到给定的目标文件。
        file.transferTo(new File("E:\\fgba\\Image\\"+originalFilename));
        
        //返回URL,进行回显
        //可以使用Nginx-图片服务器
        return "fgba"+originalFilename;
    } catch (Exception e) {
        LOGGER.info("服务器内部错误:"+originalFilename);
        e.printStackTrace();
    }
    return null;
}

}
修改nginx配置,将文件存储到文件服务器中
修改Nginx的配置文件nginx.conf,监听80端口,设置root的值为:E盘

  • 图片不能保存在服务器内部,这样会对服务器产生额外的加载负担
  • 一般静态资源都应该使用独立域名,这样访问静态资源时不会携带一些不必要的cookie,减小请求的数据量

server {

    listen       80;
    server_name  ;

    proxy_set_header X-Forwarded-Host $host;
    proxy_set_header X-Forwarded-Server $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    
    location / {
        root E:\\fgba\\image;
    }
}

每次上传文件都会经过网关,必然会给网关带来很大的压力,那我们如何绕过网关呢?
1.在网关中配置白名单 ,这样也会走网关,只是压力少了一点点
@Slf4j
public class AuthorizeFilter implements GlobalFilter, Ordered {

//白名单:存放放行的URL
private List<String> allowPaths;


@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
   
    //获取请求的url路径
    String path = request.getURI().getPath();
    boolean flag=isAllowPath(path);
    if (flag) {
        log.info("请求在白名单中,fgba.filter: {}",path);
        //放行
        return chain.filter(exchange);
    } else {
      //写其他的业务逻辑
        ~~~~
        
    }
}

private boolean isAllowPath(String path) {
 
      //判断是否允许放行
     if (allowPaths.contains(path)){
         return true;
     }
     return  false;

}
2.在nginx做转发,当请求文件上传时,直接转到相应的服务
本实例使用了方法二、需要增加配置

server {
    listen       80;
    server_name  fgba;

    proxy_set_header X-Forwarded-Host $host;
    proxy_set_header X-Forwarded-Server $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    # 新增加的配置,用于文件上传
    location /api/upload {
        proxy_pass;
        proxy_connect_timeout 600;
        proxy_read_timeout 600;
        
        rewrite "^/api/(.*)$" /$1 break;
    }
    # 网关的配置
    location / {
        proxy_pass;
        proxy_connect_timeout 600;
        proxy_read_timeout 600;
    }
}

当这样配置之后,文件上传就不会过网关,减少了网关的压力。但是有引来了一个新问题那就是跨域。

解决上传文件出现跨域问题
由于Nginx将文件上传的请求直接转发到了具体服务中,不再走gateway,所以gateway中的跨域配置,不再生效了。 需要在文件上传这个服务中单独配置跨域。

写配置类CorsFilter
/**

  • @Author: 富贵论坛www.fgba.net
  • @Date: 2021/6/15 - 11:12
  • @Description: 解决 跨域问题
  • @version: 1.0

*/
@Configuration
public class fgbaCorsConfiguration {

@Bean
public CorsFilter corsFilter(){
    //初始化配置对象
    CorsConfiguration configuration = new CorsConfiguration();
    //允许跨域访问的域名
    configuration.addAllowedOrigin("*");
   // configuration.setAllowCredentials(true);  //运行携带cookie
    configuration.addAllowedMethod("*"); //代表所有请求方法
    configuration.addAllowedHeader("*"); //允许携带任何头信息

    //初始化cors配置源对象
    UrlBasedCorsConfigurationSource configurationSource=new UrlBasedCorsConfigurationSource();
    configurationSource.registerCorsConfiguration("/**",configuration);

    //返回CorSfilter实例,参数
    return new CorsFilter(configurationSource);
}

}
到此应该就可以上传了,但是还是报跨域,我已经配置好了啊,为什么还是报跨域呢?

在nginx配置中配置请求实体大小
我就想是不是Nginx的问题,然后我就一行一行的读配置,最后发现

nginx配置中没有配置请求实体大小

加上这行配置就好了
client_max_body_size 1024m;

目录
相关文章
|
3月前
|
Java
Java关键字 —— super 详细解释!一看就懂 有代码实例运行!
文章详细解释了Java关键字`super`的用途,包括访问父类的成员变量、调用父类的构造方法和方法,并提供了相应的代码实例。
215 5
Java关键字 —— super 详细解释!一看就懂 有代码实例运行!
|
3月前
|
JavaScript 前端开发 Java
解决跨域问题大集合:vue-cli项目 和 java/springboot(6种方式) 两端解决(完美解决)
这篇文章详细介绍了如何在前端Vue项目和后端Spring Boot项目中通过多种方式解决跨域问题。
419 1
解决跨域问题大集合:vue-cli项目 和 java/springboot(6种方式) 两端解决(完美解决)
|
3月前
|
Java
通过Java代码解释成员变量(实例变量)和局部变量的区别
本文通过一个Java示例,详细解释了成员变量(实例变量)和局部变量的区别。成员变量属于类的一部分,每个对象有独立的副本;局部变量则在方法或代码块内部声明,作用范围仅限于此。示例代码展示了如何在类中声明和使用这两种变量。
|
4月前
|
Java
Java——接口的使用实例
Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。 接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。 最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。
43 4
Java——接口的使用实例
|
3月前
|
存储 前端开发 Java
Java后端如何进行文件上传和下载 —— 本地版(文末配绝对能用的源码,超详细,超好用,一看就懂,博主在线解答) 文件如何预览和下载?(超简单教程)
本文详细介绍了在Java后端进行文件上传和下载的实现方法,包括文件上传保存到本地的完整流程、文件下载的代码实现,以及如何处理文件预览、下载大小限制和运行失败的问题,并提供了完整的代码示例。
1052 2
|
3月前
|
存储 Java 数据安全/隐私保护
Java中的域,什么是域?计算机语言中的域是什么?(有代码实例)
文章解释了Java中域的概念,包括实例域、静态域、常量域和局部域,以及它们的特点和使用场景。
92 2
|
3月前
|
Java
Java关键字 —— super 与 this 详细解释!一看就懂 有代码实例运行!
本文介绍了Java中this和super关键字的用法,包括在构造方法中使用this来区分参数和成员变量、使用super调用父类构造方法和方法,以及它们在同一个方法中同时使用的场景。
172 0
Java关键字 —— super 与 this 详细解释!一看就懂 有代码实例运行!
|
3月前
|
Java
Java关键字 —— static 与 final 详细解释!一看就懂 有代码实例运行!
这篇文章详细解释了Java中static和final关键字的用法,包括它们修饰类、方法、变量和代码块时的行为,并通过代码示例展示了它们的具体应用。
276 0
Java关键字 —— static 与 final 详细解释!一看就懂 有代码实例运行!
|
4月前
|
存储 Java
Java之静态(static)与实例(instance)
Java之静态(static)与实例(instance)
|
4月前
|
存储 Java
Java内置数据类型和实例的详解
Java内置数据类型分为基本和引用两类。基本数据类型包括整型(`byte`、`short`、`int`、`long`)、浮点型(`float`、`double`)、字符型(`char`)和布尔型(`boolean`),用于存储简单的数值;引用数据类型则用于存储对象的引用,包括类(如`String`)、接口和数组。掌握这两类数据类型是Java编程的基础。以下示例展示了各种数据类型的使用方法。
48 16