2018-03-07

servlet是什么?

emmm...servlet主要功能在于交互式地浏览和修改数据,生成动态Web内容。狭义的Servlet是指Java语言实现的一个接口,广义的Servlet是指任何实现了这个Servlet接口的类,一般情况下,人们将Servlet理解为后者。

Servlet工作流程分为三个阶段:init(初始化),service(运行),destroy(销毁)

Servlet没有main方法,所有行为由Container控制。Container是web容器中的servlet容器,常见的web容器有tomcat等。

在加载Servlet的.class后,Servlet会由构造函数生成一个实例,然后Container调用init()方法完成参数的初始化,接着调用service()方法,service会根据网页的请求,调用doGet或者doPost方法,最后调用销毁方法。整个流程如下图:

使用springboot编写一个servlet,show me the code,dont bb...

maven配置

<dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-tomcat</artifactId>
            <!--<scope>provided</scope>-->
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>4.0.0</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>

servlet代码:

package com.juliet.testServlet.servlet;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

/**
 * @Author: Juliet
 * @Date: 2018/3/7
 * @Usage: Juliet's graduation design
 * Description:
 */
public class servletDemo extends HttpServlet {

    private static final long serialVersionUID = -9200488852821156029L;

    // 该函数用于初始化该servlet, 类似于我们的类的构造函数
    // 该函数只是会被调用一次, 当用户第一次访问该servlet的时候被调用
    public void init(ServletConfig parm1) throws ServletException
    {
        System.out.println("init servlet demo !");
    }

    //提供service
    @Override
    protected void service(HttpServletRequest req, HttpServletResponse resp)
            throws ServletException, IOException {
        System.out.println("service......");
        PrintWriter out = resp.getWriter();
        out.println("<html>");
        out.println("<head>");
        out.println("<title>Hello World</title>");
        out.println("</head>");
        out.println("<body>");
        out.println("<h1>this is a servlet demo</h1>");
        out.println("</body>");
        out.println("</html>");
    }

    // 销毁servlet实例(释放内存)
    // 1 reload 该servlet(webApp)
    // 2 关闭Tomcat 或者说 关机之后 都会调用这个函数
    public void destroy()
    {
        System.out.println("destory servlet demo");
    }
}

springboot Main函数代码:

package com.juliet.testServlet;

import com.juliet.testServlet.servlet.servletDemo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.ServletComponentScan;
import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.annotation.Bean;

//自动扫描servlet
@ServletComponentScan
@SpringBootApplication
public class TestServletApplication {

    public static void main(String[] args) {
        SpringApplication.run(TestServletApplication.class, args);
    }

    //servlet注册
    @Bean
    public ServletRegistrationBean servletRegistrationBean() {
        // ServletName默认值为首字母小写,即myServlet
        return new ServletRegistrationBean(new servletDemo(),"/juliet/*");
    }

}

测试结果展示:

[图片上传失败...(image-b0ba0e-1520406726860)]
刷新网页后,控制台显示如下:
[图片上传失败...(image-1f9353-1520406726860)]

servlet是否线程安全?

不是,因为每个servlet在tomcat中只有一个实例,当多个请求过来时,tomcat会开启多个线程去调用servlet,当servlet中具有静态变量或者实例变量的时候,就不能保证线程安全,做个实验验证下:

servlet代码:

package com.juliet.testServlet.servlet;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

/**
 * @Author: Juliet
 * @Date: 2018/3/7
 * @Usage: Juliet's graduation design
 * Description:
 */
public class servletThreadsafeDemo extends HttpServlet{
    private static final long serialVersionUID = -8284164610778425578L;

    private static int staticInt = 0;
    int instanceInt = 100;

    @Override
    protected void service(HttpServletRequest req, HttpServletResponse resp)
            throws ServletException, IOException {
        System.out.println("service......");
        staticInt++;
        instanceInt++;
        try {
            Thread.sleep(1000);
        }catch (Exception e){
            e.printStackTrace();
        }
        System.out.println("thread name is: "+Thread.currentThread().getName()+" instanceInt is: "+instanceInt);
        System.out.println("thread name is: "+Thread.currentThread().getName()+" staticInt is: "+staticInt);
    }

}

springboot main函数代码:

package com.juliet.testServlet;

import com.juliet.testServlet.servlet.servletDemo;
import com.juliet.testServlet.servlet.servletThreadsafeDemo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.ServletComponentScan;
import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.annotation.Bean;

//自动扫描servlet
@ServletComponentScan
@SpringBootApplication
public class TestServletApplication {

    public static void main(String[] args) {
        SpringApplication.run(TestServletApplication.class, args);
    }

    //servlet注册
    @Bean
    public ServletRegistrationBean servletDemoBean() {
        return new ServletRegistrationBean(new servletDemo(),"/servletDemo/*");
    }

    //servlet注册
    @Bean
    public ServletRegistrationBean servletThreadSafeBean() {
        return new ServletRegistrationBean(new servletThreadsafeDemo(),"/servletThreadSafe/*");
    }

}

刷新几次网页后,测试结果如下:


image.png

可以看出tomcat容器是调用了多线程来并发处理servlet请求的,普通的servlet并不支持线程安全

如何使servlet支持线程安全呢?

具体有三种方法:

1.简单粗暴的使用synchronized 关键字,就是给代码块上锁,该关键字的使用方法详细自己去百度哦~

使用方法如下:

Public class XXXXXX extends HttpServlet {
    synchronized (this){XXXX}
}

使用该方法后,测试结果如下:


这种方法使得一个servlet在一段时间只能处理一个请求,使得处理请求的吞吐量降低,很多请求将处于阻塞状态。

2.实现 SingleThreadModel 接口

public class XXXXX extends HttpServlet implements SingleThreadModel {
…………
}

如果一个Servlet实现了SingleThreadModel接口,Servlet引擎将为每个新的请求创建一个单独的Servlet实例,这将引起大量的系统开销。

3.避免使用实例变量和静态变量

解决一个问题最佛系的办法就是避免问题发生...

servlet3.0的异步

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,394评论 11 349
  • 0 系列目录# WEB请求处理 WEB请求处理一:浏览器请求发起处理 WEB请求处理二:Nginx请求反向代理 本...
    七寸知架构阅读 14,079评论 22 190
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,107评论 19 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,805评论 18 399
  • 终于,我又复活了,恢复到了当初没有你的状态,整个人无忧无虑无所牵挂,每天都有很多空余时间,现在开始努力练习...
    小可爱是阿瑶阅读 297评论 1 1