后端框架学习--servlet (1)

前言

之前有过flask,django的小玩具的开发经验,由于觉得只了解一门语言好像不太有用。因为最近能接到的单子越来越多的需要java方向的内容。比如在写python爬虫的时候经常需要逆向app,虽然现在很多app都采用了JNI的方法来保护源代码,但还是有一部分app是没有采用的。因此熟悉java很有必要。
之前已经花了一段时间来学习过java的基础语法,只是学习到java的io流部分。看来看去还是觉得没有实际写出来有用的代码,是记不住经常使用的语法什么的。于是便开始学习servlet这个后端框架。
在这里只写出个人理解的部分,不代表servlet真实的思想...

servlet

每个servlet请求都会开启一个线程来处理

生命周期

  • init()
    在客户端第一次发起请求的时候会初始化servlet
    然后调用这个方法内的逻辑
    可以用来做些变量的初始化之类的放到内存之中
    避免多次重复初始化
  • service()
    检查客户端响应的类型
    根据类型调用 servlet容器内的doGet(),doPost()....之类的http method
  • destory()
    在servlet销毁的时候会调用这个钩子函数。
    可以用来做些内存的释放或者其他操作

Http method

如果servlet中存在service()方法会直接调用service方法而不会去调用doGet,doPost方法,若还需要调用可以直接使用父类的service()方法,例如 super.service(req,resp)

  • GET
    在servlet中会使用doGet方法进行处理
  • POST
    在servlet中会使用doPost方法进行处理

表单数据

http://localhost:8080/MyForm?name=monster&age=18

  1. request.getParameter() 方法来获取表单参数的值,这个方法需要一个字符串的参数来获取值
    request.getParameter("name")
  2. request.getParameterValues() 如果参数出现一次以上,则调用该方法,并返回多个值,例如复选框.
    http://localhost:8080/MyForm?name=monster&age=18&age=58
    request.getParameter("name")
  3. request.getParameterNames() 获取请求来的所有参数名

请求头

http请求中会包含客户端的请求头


firefox的请求头
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        Enumeration header_name = request.getHeaderNames();
        while (true){
            if(header_name.hasMoreElements()){
                String header_iter = header_name.nextElement();
                System.out.println(header_iter+ " : "+request.getHeader(header_iter));
            }else {
                break;
            }
        }
    }

这段示例代码可以在控制台输出当前http请求的所有请求头.
可以通过请求头来简单的防止初级爬虫

响应头

可以在响应头中写入cookie或者其他自己需要的信息
可以在响应头中写入各种各样的信息,客户端可以解析响应头

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        response.setCharacterEncoding("utf-8");
        response.setContentType("text/html");
        response.setIntHeader("refresh",5);
        Date now = new Date();
        SimpleDateFormat s = new SimpleDateFormat ("E yyyy.MM.dd 'at' hh:mm:ss a zzz");
        PrintWriter out = response.getWriter();
        out.println("

wsw

"); out.println("

"+s.format(now)+"

"); }

这个示例代码在响应头中写入了值为5的refresh,客户端会根据这个响应头每5s刷新一次页面

Http状态码

在服务运行过程中可能会出现各种各样的错误,
此时返回Http状态码能够更快的通过Http状态码找到相应出错的地方,前提是需要返回正确的状态码.

public class HttpCode extends HttpServlet {
    private String my_parameter;
    public void init() throws ServletException {
        my_parameter = "monster";
    }

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        Enumeration names = request.getParameterNames();
        boolean flag = false;
        while (true){
            if(names.hasMoreElements()){
                if(my_parameter.equals(names.nextElement())){
                    flag = true;
                    break;
                }
            }else {
                break;
            }
        }
        if(flag){
            response.sendError(400,"

parameter True!

"); }else { response.sendError(404,"

Error!!!!!!

"); } } }

这段程序先初始化了一个自定义的字符串为monster
如果客户端发来的请求没有monster这个参数,则返回404状态码,否则返回400
这里的状态码是随便写的,没有实际意义...

过滤器

使用过滤器可以实现身份验证,数据压缩,日志等等功能

Servlet 过滤器是可用于 Servlet 编程的 Java 类,可以实现以下目的:

  • 在客户端的请求访问后端资源之前,拦截这些请求。
  • 在服务器的响应发送回客户端之前,处理这些响应。
@WebFilter(filterName = "MyFilter",urlPatterns = "/*")
public class MyFilter implements Filter {
    public void destroy() {
    }

    public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws ServletException, IOException {
        System.out.println("myfilter doing....");
        chain.doFilter(req, resp);
    }

    public void init(FilterConfig config) throws ServletException {

    }

}

之前都是在web.xml中配置servlet,其实也可以通过使用注解方式来实现url映射
使用@WebServlet,@WebFilter 就能够轻松配置url
但是个人感觉这样写好像不方便修改,需要进入到每个类内去寻找相应的路由.
不如在web.xml中写的来得简洁明了

这个程序在每个Http请求到来之后都会先通过Myfilter类的doFilter的逻辑
可以在这里实现各种身份验证或者其他功能的逻辑
如果通过了这些逻辑判断,
需要进行对get,post,或者其他方法的处理就可以调用chain.doFilter(req, resp)来进行下一步.
如果没有通过就可以在这里对resp进行写操作 返回错误内容

Cookie

Http是一种无状态的请求,所以需要通过携带其他信息提交给服务器来判断是否登录是哪个用户等等功能
其中最常用的就是通过cookie判断

未设置cookie前
@WebServlet(name = "MyCookie",urlPatterns = "/cook")
public class MyCookie extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doGet(request,response);
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        Cookie[] cookies = request.getCookies();
        for(Cookie c : cookies){
            System.out.println(c.getName() + " : " + c.getValue());
        }
        Cookie usermsg = new Cookie("hello","monster");
        response.addCookie(usermsg);
    }
}

打印cookie和设置cookie

响应头中有set-cookie
设置cookie后

你可能感兴趣的:(后端框架学习--servlet (1))