SpringMVC 实现文件的上传和下载

练习源码【upload_download】

文件上传

SpringMVC 的文件上传是基于commons-fileupload组件的文件上传。SpringMVC在原有文件上传组件做了进一步封装,简化了文件上传的代码实现。
因此,实现文件上传我们需要以下相关jar包:

基于表单的文件上传

<form action="xxx" method="post" enctype="multipart/form-data">
	选择文件:<input type="file" name="myfile"><br>
	......
	<input type="submit" value="提交">
</form>

要注意的是,基于表单的文件上传,需要设置enctype属性,并将它的值设置为multipart/form-data,同时将表单设置为post提交方式。
表单的enctype属性指定的是表单数据的编码方式,该属性有3个值:

  • application/x-www-form-urlencoded :默认编码方式,只处理表单域的value属性值。
  • multipart/form-data :该编码方式以二进制流的方式处理表单数据,并将文件域指定文件的内容封装到请求参数里。
  • text/plain :该编码方式只有在action属性值为mailto:URL的形式时才使用,主要适用于直接通过表单发送邮件的方式。

MultipartFile 接口

在SpringMVC中,上传文件时将文件的相关信息以及操作封装到了MultipartFile对象中,因此,只需要使用MultipartFile类型声明模型类的一个属性即可对被上传文件进行操作。

MultipartFile接口的相关方法如下:

  • byte[] getBytes() :以字节数组的形式返回文件内容。
  • String getContentType() :返回文件内容类型。
  • InputStream getInputStream() :返回一个InputStream,从中读取文件内容。
  • String getName() :返回请求参数的名称。
  • String getOriginalFilename() :返回上传文件的文件名。
  • long getSize() :返回文件的大小,单位为字节。
  • boolean isEmpty() :判断被上传文件是否为空。
  • void transferTo(File destination) :将上传文件保存到目标目录下。

配置

在上传文件时需要在配置文件中对org.springframework.web.multipart.commons.CommonsMultipartResolver类进行相关配置。

	<!-- 使用Spring的commonsMultipartResovler配置MultipartResovler用于文件上传  -->
	<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
		<!-- 设置请求的编码格式, 默认为iso-8859-1 -->
		<property name="defaultEncoding" value="utf-8"/>
		<!-- 设置允许上传文件的最大值, 单位为 字节 -->
		<property name="maxUploadSize" value="5400000"/>
		<!-- 设置上传文件的临时路径 -->
		<property name="uploadTempDir" value="upload/temp"/>
	</bean>

单文件上传Demo

  • 创建单文件的上传模型类 SingleFile.java
public class SingleFile {
	private String description;
	private MultipartFile myfile;
	//省略getter 和setter方法
}
  • 创建前台页面选择文件上传 oneFileUpload.jsp
<form action="${ pageContext.request.contextPath }/upload/singlefile" method="post" enctype="multipart/form-data">
	选择文件:<input type="file" name="myfile"><br>
	文件描述:<textarea rows="3" name="description"></textarea><br>
	<input type="submit" value="提交">
</form>
  • 编写处理上传文件的控制器 UploadController.java
@Controller
@RequestMapping("/upload")
public class UploadController {
	// 创建日志记录对象
	private static final Log logger = LogFactory.getLog(UploadController.class);
	/**
	 * 单文件上传处理 
	 */
	@RequestMapping(value = "/singlefile", method = RequestMethod.POST)
	public String singleFileUpload(@ModelAttribute SingleFile singleFile, HttpServletRequest request) {
		// 获取文件上传到具体文件夹的绝对路径
		String realpath = request.getServletContext().getRealPath("upload");
		// 获取上传的文件名
		String fileName = singleFile.getMyfile().getOriginalFilename();
		// 根据路径构建文件对象
		File uploadFile = new File(realpath, fileName);
		// 判断指定文件夹uploadfiles是否存在,不存在就创建
		if (!uploadFile.exists()) {
			uploadFile.mkdir();
		}
		// 上传文件
		try {
			singleFile.getMyfile().transferTo(uploadFile);
			logger.info("上传成功");
		} catch (Exception e) {
			e.printStackTrace();
		}
		return "result";
	}

多文件上传Demo

  • 创建单文件的上传模型类 MultipleFiles.java
public class MultipleFiles {
	private List<String> description;
	private List<MultipartFile> myfiles;
	//省略getter 和setter方法
}
  • 创建前台页面选择文件上传 multipartFileUpload.jsp
<form action="${ pageContext.request.contextPath }/upload/multipartfile" method="post" enctype="multipart/form-data">
	<fieldset>
		<legend>上传文件1</legend>
		选择文件:<input type="file" name="myfiles"><br>
		文件描述:<textarea rows="3" name="description"></textarea><br>
	</fieldset>
	<fieldset>
		<legend>上传文件2</legend>
		选择文件:<input type="file" name="myfiles"><br>
		文件描述:<textarea rows="3" name="description"></textarea><br>
	</fieldset>
	<fieldset>
		<legend>上传文件3</legend>
		选择文件:<input type="file" name="myfiles"><br>
		文件描述:<textarea rows="3" name="description"></textarea><br>
	</fieldset>
	<input type="submit" value="提交">
</form>
  • 编写多文件上传处理方法
@Controller
@RequestMapping("/upload")
public class UploadController {
	//省略单文件上传处理方法
	
	/**
	 * 多文件上传处理
	 */
	@RequestMapping(value = "/multipartfile", method = RequestMethod.POST)
	public String multipartFileUpload(@ModelAttribute MultipleFiles multipleFiles, HttpServletRequest request) {
		// 获取上传路径
		String realPath = request.getServletContext().getRealPath("upload");
		// 构建文件对象
		File uploadDir = new File(realPath);
		// 判断文件夹是否存在,不存在则创建
		if (!uploadDir.exists()) {
			uploadDir.mkdir();
		}
		// 获取上传文件列表
		List<MultipartFile> myfiles = multipleFiles.getMyfiles();
		// 上传文件
		for (MultipartFile myfile : myfiles) {
			File targetFile = new File(realPath, myfile.getOriginalFilename());
			try {
				myfile.transferTo(targetFile);
				logger.info(myfile.getOriginalFilename() + "上传成功");
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
		return "result";
	}	
}

文件下载

文件下载的实现有两种方式,一种是通过超链接实现下载,另一种是通过程序编码方式实现下载。前者虽然实现简单,但暴露了文件的真实位置,而且文件需要放在web程序的目录下才能下载。后者能够增加安全访问控制,还可以从任意位置提供下载数据,可以将文件存放web程序目录以外,也可以将文件存放在数据库中。

通过程序编码实现下载需要设置两个响应报头:

  • Content-Type :告诉浏览器其输出的内容不是普通文本文件或者HTML文件,而是一个保存到本地的文件,需要设置该报头的值为application/x-msdownload。在不知道文件的mime类型时,也可以设置值为application/octet-stream,该属性值表示任意的字节流。
  • Content-Disposition :告诉浏览器不直接处理相应实体内容,由用户选择将相应的实体内容保存到一个文件中,需要设置值为attachment;filename=xxx,指定接收程序处理数据内容的方式。在HTTP应用中,attachment是标准的方式,表示要求用户干预。attachment后面指定的filename参数表示服务器建议浏览器将内容保存到文件的文件名称(对于中文名称,需要通过编码转换,否则会出现乱码)。
response.setHeader("Content-Type", "application/x-msdownload");
response.setHeader("Content-Disposition", "attachment;filename=" + toUTF8String(filename));

实现的关键代码

通过请求的文件名参数,去相关目录下查找该文件,并将该文件写进Response的响应输出流,响应给浏览器。

@Controller
public class DownloadController {
	// 创建日志记录对象
	private static final Log logger = LogFactory.getLog(DownloadController.class);
	/**
	 * 执行下载
	 */
	@RequestMapping("/download")
	public String downloadFile(@RequestParam String filename, HttpServletRequest request,
			HttpServletResponse response) {
		// 获取下载的文件路径
		String uploadFilePath = request.getServletContext().getRealPath("upload");
		// 设置下载文件时的响应报头
		response.setHeader("Content-Type", "application/x-msdownload");
		response.setHeader("Content-Disposition", "attachment;filename=" + toUTF8String(filename));
		try {
			// 获取文件输入流
			FileInputStream in = new FileInputStream(new File(uploadFilePath, filename));
			// 获得响应对象的输出流,用于向客户端输出二进制数据
			ServletOutputStream out = response.getOutputStream();
			out.flush();

			int aRead = 0;
			byte[] b = new byte[1024];
			// 写到响应输出流
			while ((aRead = in.read(b)) != -1 && in != null) {
				out.write(b, 0, aRead);
			}
			out.flush();
			// 关闭IO对象
			in.close();
			out.close();
			logger.info("下载成功");
		} catch (Throwable e) {
			// TODO: handle exception
			e.printStackTrace();
		}
		return null;
	}
	
	/**
	 * 下载时保存中文文件名的字符编码转换方法
	 */
	public String toUTF8String(String str) {
		StringBuffer sb = new StringBuffer();
		int len = str.length();
		for (int i = 0; i < len; i++) {
			// 取出字符串中的每个字符
			char c = str.charAt(i);
			// Unicode码值为0-255时,不作处理
			if (c >= 0 && c <= 255) {
				sb.append(c);
			} else {
				// 转换成utf-8编码
				byte[] b;
				try {
					b = Character.toString(c).getBytes("UTF-8");
				} catch (UnsupportedEncodingException e) {
					// TODO: handle exception
					e.printStackTrace();
					b = null;
				}
				// 转换为%HH的字符串形式
				for (int j = 0; j < b.length; j++) {
					int k = b[j];
					if (k < 0) {
						k &= 255;
					}
					sb.append("%" + Integer.toHexString(k).toUpperCase());
				}
			}
		}
		return sb.toString();
	}
}

运行效果

单文件上传

在这里插入图片描述在这里插入图片描述

多文件上传

在这里插入图片描述在这里插入图片描述

文件下载

在这里插入图片描述
当点击download时
在这里插入图片描述

  • 6
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
实现Spring MVC的文件上传下载功能,需要进行以下步骤: 1、引入Apache Commons FileUpload组件的依赖。在pom.xml文件中添加以下依赖: ``` <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.3.3</version> </dependency> ``` 2、配置文件上传解析器。在Spring MVC的配置文件中,配置一个MultipartResolver的Bean用于处理文件上传请求。 3、创建文件上传的表单。在HTML表单中,设置enctype为"multipart/form-data",并添加一个文件选择框。 4、创建文件上传的控制器。在控制器中,使用MultipartFile参数接收上传文件,并执行相应的操作,比如保存文件到指定位置。 5、创建文件下载的控制器。在控制器中,使用ResponseEntity<byte[]>对象来实现文件下载,设置相应的响应头信息,如Content-Disposition和文件名。 下面是一个示例的代码,演示了如何实现文件上传下载: ``` // 文件上传的控制器 @Controller public class FileUploadController { @RequestMapping("/fileUpload") public String testFileUpload(MultipartFile photo, HttpSession session) throws IOException { String filename = photo.getOriginalFilename(); ServletContext servletContext = session.getServletContext(); String realPath = servletContext.getRealPath("photo"); File file = new File(realPath); if (!file.exists()) { file.mkdir(); } String finalPath = realPath + File.separator + filename; photo.transferTo(new File(finalPath)); return "success"; } } // 文件下载的控制器 @Controller public class FileDownloadController { @RequestMapping("/fileDownload") public ResponseEntity<byte[]> testFileDownload(HttpSession session) throws IOException { ServletContext servletContext = session.getServletContext(); String realPath = servletContext.getRealPath("static/img/a.jpg"); InputStream inputStream = new FileInputStream(realPath); byte[] bytes = new byte[inputStream.available()]; inputStream.read(bytes); MultiValueMap<String, String> headers = new HttpHeaders(); headers.add("Content-Disposition", "attachment;filename=a.jpg"); HttpStatus status = HttpStatus.OK; ResponseEntity<byte[]> responseEntity = new ResponseEntity<>(bytes, headers, status); inputStream.close(); return responseEntity; } } // 文件上传的表单 <form action="${pageContext.request.contextPath}/fileUpload" method="post" enctype="multipart/form-data"> <input type="file" name="photo" multiple> <input type="submit" value="上传"/> </form> ``` 通过以上步骤,你可以在Spring MVC中实现文件上传下载功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值