Servlet学习笔记(一)

对java web学习的记录,有关的一些教学视频和源代码可以加qq 1269557707 咨询,如果对本教程满意的话给个赞,留个好评吧,欢迎转载,但请注明出处!

目录:

  • 什么是Servlet?
    • 实现Servlet的方法
  • Servlet API
    • Servlet生命周期
    • ServletConfig
    • GenericServlet
    • HttpServlet
    • < url-pattern >
    • ServletContext
    • request和response
      • HttpServletRequest
        • 1.request域方法
        • 2. request获取请求头
        • 3.request请求数据相关其他方法
        • 4. 请求参数
        • 5. 请求包含和请求转发
        • 6. 请求转发与重定向
        • 7.request域
        • 9.请求参数和request域的不同( request.getParameter()和request.getAttribute())
      • HttpServletResponse
        • 1.响应正文
        • 2.response字符编码
        • 3. response字符流缓冲区
      • 4.设置响应头
        • 5.指定状态码
        • 6.重定向
  • 路径
    • 客户端路径和服务器路径
    • 对以上的解释(有点废话)
  • 编码
    • 乱码解决方案
    • 乱码详解(有点废话)
  • 补充:Get请求和Post请求

下面是正文:

-------------------------------------------------------------------------------------------------------------------------------

什么是Servlet?

Servlet 并不难理解,其实就是一个java应用程序,但是它和普通的java程序不同,它是由开发者创建,但是是由Servlet容器(Tomcat)运行的,它不能自动运行。Servlet容器将用户的请求传给Servlet应用程序,并把Servlet应用程序的响应回传给用户
Servlet学习笔记(一)_第1张图片
  • Servlet是JavaWeb三大组件之一(Servlet、Filter、Listener)
  • Servlet是用来处理客户端请求的动态资源
  • Servlet的任务有:
    获取请求数据
    处理请求
    完成响应
  • Servlet接口方法:
    void init(ServletConfig)
    void service(ServletRequest,ServletResponse)
    void destory()
    ServletConfig getServletConfig()
    String getServletInfo()

实现Servlet的方法

  • 实现Servlet接口(不方便)
  • 继承GenericServlet类(不方便)
  • 继承HttpServlet类(方便)

Servlet API

Servlet生命周期

Servlet接口一共5个方法,但其中只有三个是生命周期方法:
  • void init(ServletConfig)
  • void service(ServletRequest,ServletResponse)
  • void destory()
Servlet运行过程:
1). 服务器创建Servlet:
  • 当Servlet第一次被请求时,或服务器启动时,服务器会创建Servlet实例。
  • 服务器默认是在servlet第一次被请求时创建Servlet实例,如果希望服务器启动时就创建Servlet实现需要在web.xml中配置
  • 服务器只为一个类型的Servlet创建一个实例对象,所以Servlet是单例的;
2). 服务器初始化Servlet:
  • 当服务器创建Servlet实例后会马上调用Servlet的init(ServletConfig)方法,完成对Servlet的初始化;
  • init(ServletConfig)只会被调用一次
  • 服务器会在调用init()方法时传递ServletConfig参数
3). 服务器使用Servlet处理请求:
  • 当Servlet被请求时,服务器会调用Servlet的service(ServletRequest,ServletResponse)方法
  • service(ServletRequest,ServletResponse)方法每处理一次请求,就会被调用一次,所以它可能会被调用N次
  • 因为Servlet是单例的,所以可能在同一时刻一个Servlet对象会被多个请求同时访问,所以这可能出现线程案例问题
  • Servlet不是线程案例的,这有助与提高效率,但不能让Servlet具有状态,以免多个线程争抢数据
4). 服务器销毁Servlet
  • 服务器通常不会销毁Servlet,通常只有在服务器关闭时才会销毁Servlet
  • 服务器会在销毁Servlet之前调用Servlet的destory()方法
  • 可以在destory()方法中给出释放Servlet占有的资源,但通常Servlet是没什么可要释放的,所以该方法一般都是空的

ServletConfig

ServletConfig是Servlet中的init()方法的参数类型,服务器会在调用init()方法时传递ServletConfig对象给init()方法。
ServletConfig对象封装了Servlet在web.xml中的配置信息,它对应< servlet >元素。ServletConfig类的功能有:
  • String getServletName():获取Servlet配置名,即< servlet-name >的值;
  • ServletContext getServletContext():获取ServletContext对象,这个对象稍后介绍
  • String getInitParameter(String name):获取初始化参数
  • Enumeration getInitParameterNames():获取所有初始化参数的名称
在web.xml文件中,配置< servlet >时可以为配置0~N个初始化参数,例如:
< servlet >
< servlet-name > xxx servlet-name >
< servlet-class > cn.itcast.servlet.MyServlet servlet-class >
< init-param >
< param-name > p1 param-name >
< param-value > v1 param-value >
init-param >
< init-param >
< param-name > p2 param-name >
< param-value > v2 param-value >
init-param >
servlet >

GenericServlet

GenericServlet是Servlet接口的实现类,但它是一个抽象类,它唯一的抽象方法就是service()方法
GenericServlet实现了Servlet接口:
  • 实现了String getServletInfo()方法
  • 实现了void destory()方法,空实现
  • 实现了void init(ServletConfig)方法,用来保存ServletConfig参数
  • 实现了ServletConfig getServletConfig()方法
GenericServlet实现了ServletConfig接口:
  • 实现了ServletContext getServletContext()方法
  • 实现了String getInitParameter()方法
  • 实现了String getServletName()方法
  • 实现了Enumeration getInitParameterNames()方法
GenericServlet添加了init()方法:
  • 该方法会被init(ServletConfig)方法调用
  • 如果希望对Servlet进行初始化,那么应该覆盖init()方法,而不是init(ServletConfig)方法

HttpServlet

HttpServlet是GenericServlet的子类,它专注HTTP请求
HttpServlet类的方法:
  • 实现了void service(ServletRequest,ServletResponse)方法,实现内容是:
    • 把ServletRequest强转成HttpServletRequest
    • 把ServletResponse强转成HttpServletResponse
    • 调用本类添加的void service(HttpServletRequest,HttpServletResponse)方法
  • 添加了void service(HttpServletRequest,HttpServletResponse)方法,内容是:
    • 调用request的getMethod()获取请求方式
    • 如果请求方式为GET,那么调用本类添加的doGet(HttpServletRequest,HttpServletResponse)方法
    • 如果请求方式为POST,那么调用本类添加的doPost(HttpServletRequest,HttpServletResponse)方法
  • 添加了doGet(HttpServletRequest,HttpServletResponse)方法,内容是响应405,表示错误,所以我们应该去覆盖这个方法
  • 添加了doPost(HttpServletRequest,HttpServletResponse)方法,内容是响应405,表示错误,所以我们应用去覆盖这个方法
 如果是通过继承HttpServlet类来自定义Sevlet的话,那么:
  • 不要去覆盖void service(ServletRequest,ServletResponse)
  • 不要去覆盖void service(HttpServletRequest, HttpServletResponse)
  • 而应该去覆盖doGet()或doPost()方法。

< url-pattern >

< url-pattern >是的子元素,用来绑定Servlet的访问路径
可以在一个中给出多个< url-pattern >,也就是说一个Servlet可以有多个访问路径:
< servlet-mapping >
< servlet-name > xxx servlet-name >
< url-pattern > /helo1 < url-pattern >
< url-pattern > /hello2 < url-pattern >
servlet-mapping >
还可以在中使用通配符,即“*”。
  • < url-pattern > /* < url-pattern >:表示匹配任何路径
  • < url-pattern >/do/*< url-pattern >:表示匹配以/do开头的任何路径
  • < url-pattern >*.do< url-pattern >:表示匹配任何以“.do”结尾的路径
注意:
  • 通配符要么在开头,要么在结尾,不能在中间,例如:/*.do就是错误的使用。
  • 如果不使用通配符,那么< url-pattern >必须以“/”开头,例如:< url-pattern >abc< /url-pattern >就是错误的

ServletContext

ServletContext是Servlet三大域对象之一
ServletContext在服务器启动时创建,在服务器关闭时销毁,一个JavaWeb应用只创建一个ServletContext对象
它的功能分类:
  • 存取数据
  • 读取web.xml中的应用初始化参数
  • 读取应用资源
  1. 获取ServletContext对象
    在HttpServlet中可以通过以下方法来获取ServletContext对象
  • ServletContext sc = this.getServletContext()
  • ServletContext sc = this.getServletConfig().getServletContext()
  1. 存取数据
    因为在一个JavaWeb应用中,只有一个ServletContext对象,所以在ServletContext中保存的数据可以共整个JavaWeb应用中的动态资源共享
    ServletContext是Servlet三大域对象之一,域对象内部有一个Map,用来保存数据
  • void setAttribute(String name, Object value):用来添加或替换ServletContext域数据
    servletContext.setAttribute("xxx", "XXX"),添加域数据
    servletContext.setAttribute("xxx", "XXXX"),覆盖域数据,因为在域中已经存在了名为xxx的数据,所以这次就是覆盖了
  • Object getAttribute(String name):通过名称来获取域数据
  • void removeAttribute(String name):通过名称移除域数据
  • Enumeration getAttributeNames():获取所有ServletContext域数据的名称
注:Web资源
  • 静态资源:html、css、javascript、图片等;
  • 动态资源:javaweb的动态资源有jsp/servlet,其他web程序动态资源有asp、php等。
  1. 读取web.xml中配置的应用初始化参数
< context-param >
< param-name > p1 param-name >
< param-value > v1 param-value >     
context-param >
< context-param >
< param-name > p2 param-name >
< param-value > v2 param-value >     
context-param >
  • servletContext.getInitParameter("p1"),返回v1
  • servletContext.getInitParameter("p2"),返回v2
  • servletContext.getInitParameterNames(),返回Enumeration,包含p1和p2
  1. 获取项目资源
  • String getRealPath(String path):获取资源的真实名称
    String path = servletContext.getRealPath("/WEB-INF/a.jpg");
    返回值为/WEB-INF/a.jpg真实路径,即磁盘路径:C:/tomcat6/wabapps/hello/WEB-INF/a.jpg
  • InputStream getResourceAsStream(String path):获取资源的输入流
    InputStream in = servletContext.getResourceAsStream("/WEB-INF/a.jpg");
    返回的是a.jpg的输入流对象,可以从流中得到a.jpg的数据
  • Set getResourcePaths(String path):获取指定目录下的所有资源路径
    Set paths = servletContext.getResourcePaths("/WEB-INF");
    返回的Set中包含如下字符串:

    /WEB-INF/lib/
    /WEB-INF/classes/
    /WEB-INF/web.xml
    /WEB-INF/a.jpg
  1. 获取类路径资源
可以通过Class类的对象来获取类路径下的资源,对应JavaWeb应用的类路径就是classes目录下的资源
例如:
InputStream in = cn.itcast.servlet.MyServlet.class.getResourceAsStream("a.jpg");
获取的是:/WEB-INF/classes/cn/itcast/servlet/a.jpg,即与MyServlet.class同目录下的资源
例如:
InputStream in = cn.itcast.servlet.MyServlet.class.getResourceAsStream("/a.jpg");
获取的是:/WEB-INF/classes/a.jpg,即类路径的根目录下的资源,类路径的根目录就是/classes目录

request和response

  • 当服务器接收到请求后,服务器会创建request和response对象,把请求数据封装到request对象中;
  • 然后调用Servlet的service()方法时把这两个对象传递给service()方法;
  • 在service()方法中可以通过request对象获取请求数据,可以使用response对象向客户端完成响应;
  • 每次请求服务器都会创建新的request和response对象,即每个请求有自己独自的request和response对象

HttpServletRequest

HttpServletRequest表示HTTP环境中的Servlet请求
常用方法:
  • Cookie[] getCookies() 返回一个Cookie数组
  • java.lang.String getContextPath() 返回表示请求Context的URI部分
  • java.lang.String getHeader(java.lang.String name) 返回指定的HTTP请求头
  • HttpSession getSession() 返回与这个请求有关的session对象,如果没有就新建一个
request功能介绍
  • 获取请求头
  • 获取请求参数
  • Servlet三大域对象之一
  • 请求包含和请求转发

1.request域方法

  • void setAttribute(String name, Object value):添加或替换request域属性
  • Object getAttribute(String name):获取request域指定名称的域属性
  • void removeAttribute(String name):移除request域指定名称的域属性
  • Enumeration getAttributeNames():获取所有request域的属性名称

2. request获取请求头

  • String getHeader(String name):获取指定名称的请求头
  • int getIntHeader(String name):获取指定名称的请求头,把值转换成int类型。
  • Enumeration getHeaderNames():获取所有请求头名称

3.request请求数据相关其他方法

重点:
  • String getMethod():获取请求方式
  • String getContextPath():获取上下文路径,即“/” + 应用名称,例如:/day05_1
  • void setCharacterEncoding(String):设置请求体的编码
  • String getRemoteAddr():获取客户端IP地址
非重点:
  • int getContentLength():获取请求体字节数
  • Locale getLocale():获取请求Locale,例如zh_CN表示中文,中国
  • String getCharacterEncoding():获取请求体编码,在没有调用setCharacterEncoding()之前该方法返回null
  • String getQueryString():获取参数列表,例如:username=zhangSan&password=123
  • String getRequestURI():返回请求URI路径,从应用名称开始,到参数之前这一段,例如:/day05_1/AServlet
  • StringBuffer getRequestURL():整个请求URL,不包含参数部分
  • String getServletPath():返回Servlet路径,从应用名称后开始,到参数之前这一段,不包含应用名称。
  • String getServerName():返回主机名,例如:localhost
  • int getServerPort():返回服务器端口号,例如:8080

4. 请求参数

获取请求参数,即 获取超链接上的参数和表单中的参数
  • String getParameter(String name):获取指定名称的参数,如果存在同名参数,那么该方法只获取第一个参数值
  • String[] getParameterValues(String name):获取指定名称的参数,因为同名参数的存在,所以返回值为String[]
  • Enumeration getParameterNames():获取所有参数名称
  • Map getParameterMap():获取所有参数,封装到Map中,key为参数名称,value为参数值。

5. 请求包含和请求转发

有时一个请求需要多个Servlet协作才能完成,所以需要在一个Servlet跳到另一个Servlet!
  • 请求包含和请求转发都是在一个请求中,访问两个Servlet
  • 请求包含和请求转发都是有一个Servlet去调用执行另一个Servlet
  • 请求包含和请求转发都可以共享request中的数据,因为都是一个请求。
    Servlet学习笔记(一)_第2张图片
转发和包含总结:
一个请求跨多个Servlet,需要使用转发和包含。
请求转发:由下一个Servlet完成响应体!当前Servlet可以设置响应头!(留头不留体)
请求包含:由两个Servlet共同未完成响应体!(都留)
无论是请求转发还是请求包含,都在一个请求范围内!使用同一个request和response!
eg:
  • 从AServlet请求转发到BServlet
    在AServlet中可以设置响应头
    在AServlet中不能使用响应流输出
如果在AServlet中执行了响应操作,那么有两种可能:
  • 如果在AServlet中响应的数据导致response提交,那么在转发时抛出异常;
  • 如果在AServlet中响应的数据没有导致response提交,那么response中的数据会被清空。
  • 从AServlet请求包含BServlet
    在AServlet可以设置响应头
    在AServlet可以使用响应流输出
  • 请求转发和请求包含都要使用RequestDispatcher对象:RequestDispatcher rd = request.getRequestDispatcher("/BServlet");使用request获取RequestDispatcher对象,方法的参数是被转发或被包含的Servlet路径
  • 请求转发执行RequestDispatcher的forward()方法:rd.forward(request,response);
  • 请求包含执行RequestDispatcher的include()方法:rd.include(request,response);
  • 请求转发和请求包含的路径都是服务器端路径,相对当前应用

6. 请求转发与重定向

Servlet学习笔记(一)_第3张图片

重定向
1)请求转发是一个请求一次响应,而重定向是两次请求两次响应
2)请求转发地址栏不变化,而重定向会显示后一个请求的地址
3)请求转发只能转发到本项目其他Servlet,而重定向不只能重定向到本项目的其他Servlet,还能定向到其他项目
4)请求转发是服务器端行为,只需给出转发的Servlet路径,而重定向需要给出requestURI,即包含项目名!
5)请求转发和重定向效率是转发高!因为是一个请求!
如何选择?
   需要地址栏发生变化,那么必须使用重定向!
   需要在下一个Servlet中获取request域中的数据,必须要使用转发!

7.request域

Servlet中三大域对象:request、session、application,都有如下三个方法:
1)void setAttribute(String name, Object value)
2)Object getAttribute(String name)
3)void removeAttribute(String name);
同一请求范围内使用request.setAttribute()、request.getAttribute()来传值!前一个Servlet调用setAttribute()保存值,后一个Servlet调用getAttribute()获取值。

9.请求参数和request域的不同( request.getParameter()和request.getAttribute())

Servlet学习笔记(一)_第4张图片

请求参数和request域
  • getParameter()是获取客户端参数,它是从客户端传递给服务器的数据。
  • getAttribute()是获取服务器端自己设置的数据,而不是客户端的数据。
  • request没有setParameter()方法,不能自己设置参数,参数都由客户端传递
  • request有setAttribute()方法,在getAttribute()之前,需要先setAttribute()才能获取到。
  • getAttribute()和setAttribute()是用来在请求转发和请求包含中的多个Servlet中共享数据。

HttpServletResponse

response是响应对象,用来在Servlet的service()方法中向客户端响应数据
常用方法:
  • void addCookie(Cookie cookie) 为响应对象添加Cookie
  • void addHeader(java.lang.String name, java.lang.String value) 为响应对象添加标头
  • void sendRedirect(java.lang.String location) 发送响应代号,把浏览器重定向到指定位置
response的功能如下:
  • 设置响应头
  • 发送状态码
  • 设置响应正文
  • 重定向

1.响应正文

  • ServletOutputStream getOutputStream():用来向客户端响应字节数据;
  • PrintWriter getWriter():用来向客户端响应字符数据;
response.getWriter().print("你好");//向客户端响应字符数据
byte[] bytes = ...;
response.getOutputStream().write(bytes);//向客户端响应字节数据

2.response字符编码

  • Tomcat响应数据默认使用ISO-8859-1
  • 通常浏览器默认使用GBK编码
  • response.setCharacterEncoding("utf-8"); / /设置response.getWriter()的字符编码
response.getWriter().print("大家好");
因为Tomcat默认使用的是ISO-8859-1编码,不支持中文。所以一定编码!
response.setCharacterEncoding("utf-8");
response.getWriter().print("大家好");
因为已经设置了字符流编码为utf-8,所以响应给客户端的数据为utf-8编码!
但因为浏览器默认使用的是gbk来解析响应数据,所以乱码!如果浏览器使用utf-8编码,那么就不会乱码了。
response.setCharacterEncoding("gbk");
response.getWriter().print("大家好");
因为设置了字符流编码为gbk,所以响应给客户端的数据为gbk编码!
因为浏览器默认使用gbk来解析数据,所以不会出现乱码!如果浏览器使用utf-8编码,那么就会出现乱码!
4)
response.setContentType("text/html;charset=utf-8");
response.getWriter().print("大家好");
setContentType()方法有两个作用:
  • 设置字符流编码。等同与调用了response.setCharacterEncoding("utf-8");
  • 设置Content-type响应头,即通知浏览器响应数据的编码为utf-8。
    因为设置字符流的编码为utf-8,所以响应给客户端数据为utf-8编码
    因为设置了Content-type头为utf-8,所以浏览器会使用utf-8来解析响应数据
    没有乱码!
response.setContentType("text/html;charset=gbk");
response.getWriter().print("大家好");
  • 设置了字符流为gbk,所以响应给客户端的数据为gbk
  • 设置了Content-type头为gbk,所以通知浏览器响应数据为gbk编码
    没有乱码!
response.setHeader("Content-type", "text/html;charset=utf-8")
等同于
repsonse.setContentType("text/html;charset=utf-8")
response.getOutputStream().write("大家好".getBytes("gbk"));
响应的数据是gbk编码
客户端浏览器默认使用gbk编码
所以没有乱码

3. response字符流缓冲区

response字符流缓冲区大小为8KB,当向字符流中写入数据后,数据可能只在缓冲区中,而没有发送到浏览器。
可以调用response.flushBuffer()或response.getWriter().flush()方法刷新缓冲区,把数据发送到浏览器。

4.设置响应头

response.setHeader("Content-type", "text/html;charset=utf-8");
等同与
response.setContentType("text/html;charset=utf-8");
response.setHeader("Refresh", "5; URL=http://www.itcast.cn");

5.指定状态码

response.setStatus(200):设置状态码为200
response.sendError(404, “您要查找的资源不存在”):设置状态码为404
repsonse.sendError(500, “服务器出氏了”):设置状态码为500
在调用sendError()方法时,不只是设置了状态码,而且还会给浏览器一个显示错误的页面。

6.重定向

Servlet学习笔记(一)_第5张图片

重定向

response.sendStatus(302);
repsonse.setHeader("Location", " http://www.baidu.com");
快捷的方法重定向:
response.sendRedirect(" http://www.baidu.com ");
  • 重定向是两次请求
  • 重定向不局限与当前应用,也可以是其他应用,例如重定向到百度
  • 重定向响应码为302,而且必须有Location响应头
  • 重定向与response响应流同时使用。

路径

客户端路径和服务器路径

  1. 客户端路径和服务器端路径
  • 客户端路径需要给出应用名称,例如:/day05_1/AServlet
  • 服务器端路径无需给出应用名称,例如:/AServlet
  1. 客户端路径
    1). 页面中都是客户端路径:
  • 超链接的href
  • 表单的action
  • 的src
    2). 重定向也是客户端路径:response.sendRedirect("/day05_1/BServlet");
  1. 服务器端路径
  • 请求转发和请求包含
  • ServletContext获取资源等

对以上的解释(有点废话)

  • web.xml中路径,(叫它Servlet路径!)
要么以“ * ”开关,要么为“ / ”开头
  • 转发和包含路径
以“/”开头:相对当前项目路径,例如: http://localhost:8080/项目名/  request.getRequestdispacher("BServlet").for...();
不以“/”开头:相对当前Servlet路径。 request.getRequestdispacher("/BServlet").for...();,假如当前Servlet是: http://localhost:8080/项目名/servlet/AServlet,  就是http://localhost:8080/项目名/servlet/BServlet
  • 重定向路径(客户端路径)
     > 以“ / ”开头:相对当前主机,例如:http: //localhost:8080/, 所以需要自己手动添加项目名,例如;response.sendRedirect(" /day10_1/Bservlet");
  • 页面中超链接和表单路径
     > 与重定向相同,都是客户端路径!需要添加项目名
    < form action="/day10_1/AServlet">
    < a href="/day10_/AServlet">
    < a href="AServlet" > ,如果不以“/”开头,那么相对当前页面所在路径。如果是http://localhost:8080/day10_1/html/form.html。 即: http://localhost:8080/day10_1/html/ASevlet
    建议使用以“/”开头的路径,即绝对路径!
      * ServletContext获取资源路径
        > 相对当前项目目录,即当然index.jsp所在目录
      * ClassLoader获取资源路径
        > 相对classes目录
      * Class获取资源路径
        > 以“/”开头相对classes目录
        > 不以“/”开头相对当前.class文件所在目录。

编码

  常见字符编码:iso-8859-1(不支持中文)、gb2312、gbk、gb18030(系统默认编码,中国的国标码)、utf-8(万国码,支持全世界的编码,所以我们使用这个)

乱码解决方案

一、浏览器和服务器之间传输乱码
  1. 请求编码
  • 客户端发送的数据编码:由浏览器来决定:
    1). 如果是在地址栏中直接给出url,那么一般都是默认为GBK,但这个可能不太大。
    2). 如果是通过页面上的表单或超链接发出请求,那么由当前页面的编码来决定发送的参数的编码。
  • 无论浏览器发送过来的是什么编码的数据,Tomcat都默认使用ISO-8859-1来解码
    1). POST:可以使用request.setCharacterEncoding()方法来设置请求体数据的编码,因为POST请求参数在请求体中,所以是可以设置编码的。在使用request.getParameter()方法获取参数之前,先使用request.setCharacterEncoding("UTF-8")方法来设置编码即可。
    2). GET:没有方法可以设置它,因为参数在url中。所以使用request.getParameter()获取到的数据一定是错误的使用了iso-8859-1解码的。可以再使用iso-8859-1把字符串转回到byte[],再重新使用正确的编码来解码即可。
    String s = request.getParameter("s");//使用iso-8859-1错误的解码了
    byte[] bytes = s.getBytes("iso-8859-1");//退回错误的解码,让字符串通过iso-8859-1返回到字节数据,即还原字节数据
    s = new String(bytes, "utf-8");//重新使用正确的utf-8来解码。
2.响应编码
在使用response.getWriter()之前可以使用setContentType("text/html;charset=utf-8)。或者response.setCharacterEncoding("UTF-8");
  1. 使用过滤器
二、文件乱码
1、项目文本文件默认编码:【右击项目】->【Properties】->【Resource】->【Text file encoding】
2、文件默认编码:默认使用项目的默认编码:【右击文件】->【Properties】->【Resource】->【Text file encoding】
3、 JSP 文件编码:【右击文件】->【Properties】->【Resource】->【Text file encoding】
4、JSP翻译为Servlet时的编码:
< %@ page language="java" pageEncoding="utf-8"% >

乱码详解(有点废话)

  1. 响应编码
    Servlet学习笔记(一)_第6张图片
      * 当使用response.getWriter()来向客户端发送字符数据时,如果在之前没有设置编码,那么默认使用iso,因为iso不支持中文,一定乱码
      * 在使用response.getWriter()之前可以使用response.setCharaceterEncoding()来设置字符流的编码为gbk或utf-8,当然我们通常会选择utf-8。这样使用response.getWriter()发送的字符就是使用utf-8编码的。但还是会出现乱码!因为浏览器并不知道服务器发送过来的是什么编码的数据!这时浏览器会使用gbk来解码,所以乱码!
      * 在使用response.getWriter()之前可以使用response.setHeader("Content-type","text/html;charset=utf-8")来设置响应头,通知浏览器服务器这边使用的是utf-8编码,而且在调用setHeader()后,还会自动执行setCharacterEncding()方法。这样浏览器会使用utf-8解码,所以就不会乱码了!
      * setHeader("Content-Type", "text/html;charset=utf-8")的快捷方法是:setContentType("text/html;charset=utf-8)。
  2. 请求编码
      * 客户端发送给服务器的请求参数是什么编码:
        客户端首先要打开一个页面,然后在页面中提交表单或点击超链接!在请求这个页面时,服务器响应的编码是什么,那么客户端发送请求时的编码就是什么。
      * 服务器端默认使用什么编码来解码参数:
        服务器端默认使用ISO-8859-1来解码!所以这一定会出现乱码的!因为iso不支持中文!
      * 请求编码处理分为两种:GET和POST:GET请求参数不在请求体中,而POST请求参数在请求体中,所以它们的处理方式是不同的!
      * GET请求编码处理:
      Servlet学习笔记(一)_第7张图片
        > String username = new String(request.getParameter("iso-8859-1"), "utf-8");
        > 在server.xml中配置URIEncoding=utf-8
      * POST请求编码处理:
        > String username = new String(request.getParameter("iso-8859-1"), "utf-8");
        > 在获取参数之前调用request.setCharacterEncoding("utf-8");
  3. URL编码
      表单的类型:Content-Type: application/x-www-form-urlencoded,就是把中文转换成%后面跟随两位的16进制。
      为什么要用它:在客户端和服务器之间传递中文时需要把它转换成网络适合的方式。
  * 它不是字符编码!
  * 它是用来在客户端与服务器之间传递参数用的一种方式!
  * URL编码需要先指定一种字符编码,把字符串解码后,得到byte[],然后把小于0的字节+256,再转换成16进制。前面再添加一个%。
  * POST请求默认就使用URL编码!tomcat会自动使用URL解码!
  * URL编码:String username = URLEncoder.encode(username, "utf-8");
  * URL解码:String username = URLDecoder.decode(username, "utf-8");
  最后我们需要把链接中的中文参数,使用url来编码!

补充:Get请求和Post请求

最为常见的客户端传递参数方式有两种:
  • 浏览器地址栏直接输入:一定是GET请求;
  • 超链接:一定是GET请求;
  • 表单:可以是GET,也可以是POST,这取决与的method属性值;
GET请求和POST请求的区别:
GET请求:
  • 请求参数会在浏览器的地址栏中显示,所以不安全;
  • 请求参数长度限制长度在1K之内;
  • GET请求没有请求体,无法通过request.setCharacterEncoding()来设置参数的编码;
    POST请求:
  • 请求参数不会显示浏览器的地址栏,相对安全;
  • 请求参数长度没有限制;

你可能感兴趣的:(JavaWeb)