问题:使用字节流输出,产生了中文乱码?

举个栗子

如下面的形式就会输出中文乱码

public class ResponseDemo1 extends HttpServlet {

    /**
     * 演示字节流输出的乱码问题
     */
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
       
        String str = "字节流输出中文的乱码问题";//UTF-8的字符集,此时浏览器显示也需要使用UTF-8的字符集。
        //1.拿到字节流输出对象
        ServletOutputStream sos = response.getOutputStream();

        //2.把str转换成字节数组之后输出到浏览器
        sos.write(str.getBytes("UTF-8")); 
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }
}
原因:存和取用的不是同一个码表
  • 在保存时用的是IDEA创建文件使用的字符集UTF-8
  • 到浏览器上显示,chrome浏览器和ie浏览器默认的字符集是GB2312(其实就是GBK),存和取用的不是同一个码表,就会产生乱码。
解决办法:把存和取的码表统一

第一种解决办法:

修改浏览器的编码,使用右键——编码——改成UTF-8。(不推荐使用,我们的应用尽量不要求用户取做什么事情)

ie和火狐浏览器可以直接右键设置字符集。而chrome需要安装插件,很麻烦。

Javaweb 响应字节流输出中文乱码问题_Java

第二种解决办法: (不建议使用,因为不好记)

向页面上输出一个meta标签,内容如下:

 <meta http-equiv="content-type" content="text/html;charset=UTF-8">

其实它就是指挥了浏览器,使用哪个编码进行显示。

在java中是这样写的:

//1.拿到字节流输出对象
ServletOutputStream sos = response.getOutputStream();

sos.write("<meta http-equiv='content-type' content='text/html;charset=UTF-8'>".getBytes());
        

第三种解决办法:

设置响应消息头,告知浏览器响应正文的MIME类型和字符集

response.setHeader("Content-Type","text/html;charset=UTF-8");

在Java中是这样实现的:

//1.拿到字节流输出对象
ServletOutputStream sos = response.getOutputStream();
      
response.setHeader("Content-Type","text/html;charset=UTF-8");

第四种解决办法:我们推荐使用的办法

它的本质就是设置了一个响应消息头

response.setContentType("text/html;charset=UTF-8");

在java中是这样实现的

//1.拿到字节流输出对象
ServletOutputStream sos = response.getOutputStream();

response.setContentType("text/html;charset=UTF-8");
//2.把str转换成字节数组之后输出到浏览器
sos.write(str.getBytes("UTF-8")); 

案例

上述四种解决办法的案例:

public class ResponseDemo1 extends HttpServlet {

    /**
     * 演示字节流输出的乱码问题
     */
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        /**
         * 问题:
         * 	  String str = "字节流中文乱码问题";
         * 	     使用字节流输出,会不会产生中文乱码?
         * 答案:
         * 	  会产生乱码
         * 原因:
         * 	String str = "字节流中文乱码问题"; 在保存时用的是IDEA创建文件使用的字符集UTF-8。
         * 	到浏览器上显示,chrome浏览器和ie浏览器默认的字符集是GB2312(其实就是GBK),存和取用的不是同一个码表,就会产生乱码。
         *
         * 引申:
         *   如果产生了乱码,就是存和取用的不是同一个码表
         * 解决办法:
         *   把存和取的码表统一。
         */
        String str = "字节流输出中文的乱码问题";//UTF-8的字符集,此时浏览器显示也需要使用UTF-8的字符集。
        //1.拿到字节流输出对象
        ServletOutputStream sos = response.getOutputStream();
        /**
         * 解决办法:
         * 	第一种解决办法:
         *      修改浏览器的编码,使用右键——编码——改成UTF-8。(不推荐使用,我们的应用尽量不要求用户取做什么事情)
         *      ie和火狐浏览器可以直接右键设置字符集。而chrome需要安装插件,很麻烦。
         * 	第二种解决办法: (不建议使用,因为不好记)
         *  	向页面上输出一个meta标签,内容如下: <meta http-equiv="content-type" content="text/html;charset=UTF-8">
         *      其实它就是指挥了浏览器,使用哪个编码进行显示。
         *  第三种解决办法:
         * 		设置响应消息头,告知浏览器响应正文的MIME类型和字符集
         * 		response.setHeader("Content-Type","text/html;charset=UTF-8");
         * 	第四种解决办法:我们推荐使用的办法
         * 	    它的本质就是设置了一个响应消息头
         *  	response.setContentType("text/html;charset=UTF-8");
         */
        //第二种解决办法:sos.write("<meta http-equiv='content-type' content='text/html;charset=UTF-8'>".getBytes());
        //第三种解决办法:response.setHeader("Content-Type","text/html;charset=UTF-8");
        //第四种解决办法:
        response.setContentType("text/html;charset=UTF-8");
        //2.把str转换成字节数组之后输出到浏览器
        sos.write(str.getBytes("UTF-8")); 
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }
}

问题解决后案例:正常输出中文

public class ResponseDemo2 extends HttpServlet {

    /**
     * 字符流输出中文乱码
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        String str = "字符流输出中文乱码";
        //response.setCharacterEncoding("UTF-8");

        //设置响应正文的MIME类型和字符集
        response.setContentType("text/html;charset=UTF-8");
        //1.获取字符输出流
        PrintWriter out = response.getWriter();
        //2.使用字符流输出中文
        /**
         * 问题:
         * 	out.write(str); 直接输出,会不会产生乱码
         * 答案:
         * 	会产生乱码
         * 原因:
         * 	存用的什么码表:UTF-8
         *  在浏览器取之前,字符流PrintWriter已经获取过一次了,PrintWriter它在取的时候出现了乱码。
         * 	浏览器取默认用的是GBK。(本地系统字符集)
         *
         *  UTF-8(存)————>PrintWriter ISO-8859-1(取)					乱
         *  PrintWirter ISO-8859-1(存)————>浏览器 GBK(取)				乱
         *
         * 解决办法:
         * 	改变PrintWriter的字符集,PrintWriter是从response对象中获取的,其实设置response的字符集。
         *  注意:设置response的字符集,需要在拿流之前。
         *  response.setCharacterEncoding("UTF-8");
         *
         * response.setContentType("text/html;charset=UTF-8");
         * 此方法,其实是做了两件事:
         * 		1.设置响应对象的字符集(包括响应对象取出的字符输出流)
         * 		2.告知浏览器响应正文的MIME类型和字符集
         */

        out.write(str);
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }
}