在前面的文章中,我们学习了Feign声明式服务调用,但是在Spring Cloud封装的Feign中并不直接支持传文件,但Feign官方提供了子项目Feign-form来实现跨服务的文件上传,具体如下:
服务提供方(接收文件)
服务提供方的实现比较简单,就按Spring MVC的正常实现方式即可,就在我们之前的eurekaclient项目添加,如下:
@PostMapping(value = "/fileupload", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
public String fileupload(@RequestPart("file") MultipartFile file){
if(file == null){
return "附件为空,上传文件失败";
}
String filename = file.getOriginalFilename();
long fileSize = file.getSize();
return "上传成功,文件名为:"+filename+" 文件大小:"+fileSize;
}
服务消费方(发送文件)
在服务消费方由于会使用Feign客户端,所以在这里需要在引入feign对表单提交的依赖,在我们之前serice-feign项目中添加,具体如下:
<dependency>
<groupId>io.github.openfeign.form</groupId>
<artifactId>feign-form</artifactId>
<version>3.0.3</version>
</dependency>
<dependency>
<groupId>io.github.openfeign.form</groupId>
<artifactId>feign-form-spring</artifactId>
<version>3.0.3</version>
</dependency>
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.3</version>
</dependency>
定义文件上传方的应用主类和FeignClient,并且设置我们服务提供方的服务名为service-hi
@FeignClient(value = "service-hi")
public interface SchedualServiceHi {
@PostMapping(value = "/fileupload",
consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
public String fileupload(@RequestPart("file") MultipartFile file);
@Configuration
class MultipartSupportConfig {
@Bean
public Encoder feignFormEncoder() {
return new SpringFormEncoder();
}
}
}
服务消费端接口,同样按照Spring MVC的正常实现方式即可,如下:
@RestController
public class HiController {
@Autowired
SchedualServiceHi schedualServiceHi;
@PostMapping(value = "/fileupload")
public String fileupload(@RequestPart("file") MultipartFile file){
return schedualServiceHi.fileupload(file);
}
}
然后我们依次启动服务注册中心、服务提供方和服务消费方进行文件上传测试,测试结果如下:
注意点:
1.接口注解PostMapping需要加入consumes = MediaType.MULTIPART_FORM_DATA_VALUE,如下:
@PostMapping(value = "/fileupload", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)2.参数注解@RequestPart(“file”)不能写成@RequestParam("file")。
3.如果文件较大,最好将Hystrix的超时时间设置的长一些,否则文件还没有上传完,Hystrix就超时,导致文件上传失败。