1. @responseBody注解
- @responseBody注解的作用是将controller的方法返回的对象通过适当的转换器转换为指定的格式之后,写入到response对象的body区,通常用来返回JSON数据或者是XML(在使用此注解之后不会再走视图处理器,而是直接将数据写入到输入流中)
- 在springmvc中当返回值是String时,如果不加@ResponseBody的话,返回的字符串就会找这个String对应的页面,如果找不到会报404错误。如果加上@ResponseBody注解的话,返回的就是json字符串,并且把这个json字符串返回给发送请求的原页面。
2. response.getWriter().write()与response.getWriter()的区别
- 不同的是response.getOutputStream()向浏览器输出的是二进制数据,是字节流,可以处理任意类型的数据,而response.getWriter()输出的是字符型数据,是字符流。
在服务器端如何控制客户端浏览器以以UTF-8的编码显示数据呢?
- 可以通过设置响应头控制浏览器的行为,例如:response.setHeader(“content-type”, “text/html;charset=UTF-8”);通过设置响应头控制浏览器以UTF-8的编码显示数据。
2.1使用OutputStream流向浏览器输出中文
public class ServletDemo01 extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String str="使用OutputStream输出中文:我心自在";
//在服务器设置响应头,告诉浏览器以utf-8的编码显示数据,如果不写会出现中文乱码
response.setHeader("content-type", "text/html;charset=UTF-8");
//获取OutputStream输出流
OutputStream os=response.getOutputStream();
/**
* getBytes的作用是将字符转化为字节数组,如果不带参数,默认根据系统环境来进行转化
* 这里指定以utf-8的编码进行转换
*/
byte[]b=str.getBytes("utf-8");
//想客户端(浏览器)输出数据
os.write(b);
}
}
2.2使用PrintWriter流向浏览器输出中文
public class ServletDemo02 extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String str="使用PrintWriter输出中文:我心自在";
//设置中文编码格式
response.setCharacterEncoding("utf-8");
//获取流
PrintWriter pw=response.getWriter();
//输出
pw.write(str);
pw.close();
}
}
这里需要注意的是: 在获取PrintWriter输出流之前首先使用"response.setCharacterEncoding(charset)"设置字符以什么样的编码输出到浏览器,如:response.setCharacterEncoding(“UTF-8”);设置将字符以"UTF-8"编码输出到客户端浏览器,然后再使用response.getWriter();获取PrintWriter输出流,这两个步骤不能颠倒.如果颠倒,设置将无效,还是会出现中文乱码!
通过比较可以看出,当需要向浏览器输出字符数据时,使用PrintWriter比较方便,因为不需要将字符转化为字节这一步!
3、使用OutputStream流和PrintWriter流下载文件
文件下载,在实际开发中用到的很多,最近做的项目便是,查看图片的时候,点击图片的时候进行图片下载
文件下载,运用的步骤比较固定,大致分为以下几步:
-
获取要下载的文件的绝对路径(注意是绝对路径)
-
获取要下载的文件名(获取文件名要注意中文文件名的问题,需进行编码)
-
设置content-disposition响应头控制浏览器,告诉浏览器以下载的形式打开文件
-
获取要下载的文件输入流
-
创建数据缓冲区
-
通过response对象获取OutputStream流
-
将FileInputStream流写入到buffer缓冲区
-
使用OutputStream将缓冲区的数据输出到客户端浏览器
public class ServletDemo03 extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获取文件的下载绝对路径
String downloadPath=this.getServletContext().getRealPath("/download/阳光海滩.jpg");
//获取文件名
String fileName=downloadPath.substring(downloadPath.lastIndexOf("\\")+1);
//设置响应头,告诉浏览器以下载的方式打来文件,设置中文编码,如果不设置会出现乱码
response.setHeader("content-disposition", "attachment;filename="+URLEncoder.encode(fileName, "UTF-8"));
//获取文件流
InputStream is=new FileInputStream(downloadPath);
int len=0;
byte[]b=new byte[1024];
OutputStream os=response.getOutputStream();
while((len=is.read(b))!=-1){
//将缓冲区数据输出到浏览器
os.write(b,0,len);
}
is.close();
}
PrintWriter流处理节数据,会导致数据丢失,因此在编写下载文件功能时,要使用OutputStream流,避免使用PrintWriter流,因为OutputStream流是字节流,可以处理任意类型的数据,而PrintWriter流是字符流,只能处理字符数据,如果用字符流处理字节数据,会导致数据丢失。