文件上传的回顾
- 文件上传的必要前提
- form表单的enctype取值必须是:
multipart/form-data
- method属性取值必须是
post
- 提供一个文件选择域
<input type="file" />
- form表单的enctype取值必须是:
- 导入文件上传的jar包
使用commons-fileupload组件实现文件上传,需要导入该组件响应的支撑jar包,commons-fileupload和commons-io,cmmons-io不属于文件上传组件的开发jar文件,单commons-fileupload组件从1.1版本开始,它工作时需要commons-io包的支持
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency>
<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
<version>2.4</version>
</dependency>
- 编写文件上传的JSP页面
<h3>文件上传</h3>
<form action="user/fileupload" method="post" enctype="multipart/form-data">
选择文件:<input type="file" name="upload"/><br/>
<input type="submit" value="上传文件"/>
</form>
- 编写文件上传的Controller控制器
/**
* 文件上传
* @return
*/
@RequestMapping("/fileUpload1")
public String fileUpload(HttpServletRequest request) throws Exception {
System.out.println("文件上传");
//使用fileupload组件完成文件上传
//上传的位置
String path = request.getSession().getServletContext().getRealPath("/uploads/");
//判断该路径是否存在
File file = new File(path);
if(!file.exists()){
file.mkdirs();
}
//解析request对象,获取上传文件项
DiskFileItemFactory factory = new DiskFileItemFactory();
ServletFileUpload upload = new ServletFileUpload(factory);
//解析request
List<FileItem> items = upload.parseRequest(request);
//遍历
for(FileItem item:items){
//进行判断,当前items对象是否上传文件项
if(item.isFormField()){
//说明是普通表单项
}else{
//说明是上传文件项
//获取上传文件的名称
String filename = item.getName();
//把文件的名称设置成唯一值
String uuid = UUID.randomUUID().toString().replace("-","");
filename = uuid+"_"+filename;
//完成文件上传
item.write(new File(path,filename));
//删除临时文件
item.delete();
}
}
return "success";
}
以上就是传统的文件上传操作
SpringMVC传统方式文件上传
- 原理分析:
- 使用form表单上传文件,会把文件存入request域中,然后经过前端控制器
- 通过前端控制器可以调用一个配置文件解析器组件,当前端上传文件时,前端控制器会调用配置文件解析器,配置文件解析器会解析请求,解析出上传文件对象
- 然后通过参数绑定将文件对象传入Controlelr方法(需要写MultipartFile形参),这个形参就是上传的文件对象
- ++注意:方法的参数名称必须与form表单中file类型的input标签的name值相同++
- 编写Controller代码
/**
* SpringMVC方式实现文件上传
* @param request
* @param upload2
* @return
*/
@RequestMapping("/fileupload2")
public String fileupload(HttpServletRequest request,MultipartFile upload2) throws IOException {
System.out.println("SpringMVC方式上传文件");
//先获取到要上传的文件目录
String path =request.getSession().getServletContext().getRealPath("/uploads/");
//创建File对象,一会向该路径下上传文件
File file = new File(path);
//判断路径是否存在,如果不存在,创建该路径
if(!file.exists()) {
file.mkdirs();
}
//获取到上传文件的名称
String filename = upload2.getOriginalFilename();
String uuid = UUID.randomUUID().toString().replaceAll("-","").toUpperCase();
//把文件名称唯一化
filename = uuid+"_"+filename;
//上传文件
upload2.transferTo(new File(file,filename));
return "success";
}
SpringMVC实现跨服务器方式的文件上传
分服务器的目的
在实际开发中,我们会有很多处理不同功能的服务器,例如:
应用服务器,负责部署我们的应用
数据库服务器:运行我们的数据库
缓存和消息服务器:负责处理大并发访问的缓存和消息
文件服务器:负责存储用户上传文件的服务器。
实现
- 首先要搭建两个Tomcat服务器
- 实现SpringMVC跨服务器方式文件上传
- 导入开发需要的jar包
<dependency> <groupId>com.sun.jersey</groupId> <artifactId>jersey-core</artifactId> <version>1.18.1</version> </dependency> <dependency> <groupId>com.sun.jersey</groupId> <artifactId>jersey-client</artifactId> <version>1.18.1</version> </dependency>
- 编写文件上传的JSP页面
<h3>跨服务器的文件上传</h3> <form action="user/fileupload3" method="post" enctype="multipart/form-data"> 选择文件:<input type="file" name="upload"/> <br/> <input type="submit" value="上传文件"/> </form>
- 编写控制器类
/**
* 跨服务器的文件上传
* @param upload
* @return
* @throws IOException
*/
@RequestMapping("/fileupload3")
public String fileupload3(MultipartFile upload) throws IOException {
System.out.println("SpringMVC跨服务器上传文件");
//定义上传文件服务器的路径
String path = "http://localhost:8888/fileuploadserver/uploads/";
//获取到上传文件的名称
String filename = upload.getOriginalFilename();
String uuid = UUID.randomUUID().toString().replaceAll("-","").toUpperCase();
//把文件名称唯一化
filename = uuid+"_"+filename;
//向图片服务器上传文件
//创建客户端对象
Client client = Client.create();
//连接图片服务器
WebResource webResource = client.resource(path+filename);
//上传文件
webResource.put(upload.getBytes());
return "success";
}
注意如果报405错误,可以尝试修改Tomcat文件夹下的conf/web.xml文件里面加上
<init-param>
<param-name>readonly</param-name>
<param-value>false</param-value>
</init-param>