Servlet的多线程并发问题

一、Servlet : 用java语言编写的动态资源开发技术。
二、Servlet 特色:
1)普通的java类,继承HttpServlet类,覆盖doGet、doPost等方法。
2)Servlet类只能交给tomcat服务器运行。
三、怎样使用Eclipse开发Servlet?
1)编写一个servlet类,继承HttpServlet
public class Servlet extends HttpServlet{br/>@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("This is Servlet!");
resp.getWriter().write("This is Servlet!");
}
}
2)配置web.xml文件
<servlet>
<servlet-name>Servlet</servlet-name>
<servlet-class>zzuli.edu.cn.Servlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Servlet</servlet-name>
<url-pattern>/Servlet</url-pattern>
</servlet-mapping>
3)访问  http://localhost:8080/FirstServlet/Servlet
四、在web.xml中配置 <servlet>和<servlet-mapping>的做用?
servlet容器对url的匹配过程:
当一个请求发送到servlet容器的时候,容器先会将请求的url减去当前应用上下文的路径做为servlet的映射url,好比我访问的是http://localhost:8080/FirstServlet/Servlet,个人应用上下文是FirstServlet,容器会将http://localhost:8080/FirstServlet去掉,剩下的/Servlet部分拿来作servlet的映射匹配。
映射匹配步骤:
1)首先在web.xml文件中查找是否有匹配的url-pattern的内容(/Servlet)
2)若是找到匹配的url-pattern,则使用当前servlet-name的名称到web.xml文件中查询是否相同名称的servlet配置
3)若是找到相同名称的servlet配置,则取出对应的servlet配置信息中的servlet-class内容(zzuli.edu.cn.Servlet),而后经过servlet-class里的内容,反射构造Servlet的对象,调用Servlet对象里面的方法。
五、 Myeclipse和Eclipse中的Tomcat怎样部署项目?
1)Myeclipse默认将项目部署到tomcat安装目录下的webapps中
2)eclipse并不像MyEclipse默认将项目部署到tomcat安装目录下的webapps中,而是默认部署到工做目录(workspace)下的.metadata/.plugins/org.eclipse.wst.server.core/tmp0/wtpwebapps中
六、Servlet 注解br/>Servlet3.0以上可使用注解@WebServlet自动映射,不用在web.xml中配置<servlet>和<servlet-mapping>
使用方法:编写一个servlet类,继承HttpServlet,而后在servlet上面加上@webServlet便可。br/>@WebServlet("/FirstServlet2")
public class FirstServlet2 extends HttpServlet{br/>@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("This is Servlet2!");
resp.getWriter().write("This is Servlet2!");br/>}
}
注意点:@WebServlet注解括号里面必须加映射路径
七、Sevlet的生命周期(重点)
Servlet重要的四个生命周期方法:
1)构造方法:建立servlet对象的时候调用。默认状况下,第一次访问servlet的时候建立servlet对象。只调用1次,证实servlet对象在tomcat是单实例的。
2)init方法:建立完servlet对象的时候调用,只调用1次。
3)service方法:每次发出请求时调用,调用n次。
4)destroy方法:销毁servlet对象的时候调用。中止服务器或者从新部署web应用时销毁servlet对象。只调用1次。
八、伪代码演示servlet的生命周期
Tomtcat内部代码运行:
一、经过URL映射找到到servlet-class的内容(zzuli.edu.cn.FirstServlet)
二、经过反射构造FirstServlet对象
     2.1 获得字节码对象
           Class clazz = class.forName("zzuli.edu.cn.FirstServlet");
     2.2 调用无参数的构造方法来构造对象
          Object obj = clazz.newInstance();  ---1.servlet的构造方法被调用
三、建立ServletConfig对象,经过反射调用init方法
     3.1 获得方法对象
           Method m = clazz.getDeclareMethod("init",ServletConfig.class);
      3.2 调用方法
            m.invoke(obj,config);     --2.servlet的init方法被调用
四、建立request,response对象,经过反射调用service方法
      4.1 获得方法对象
            Methodm m=clazz.getDeclareMethod("service",HttpServletRequest.class,HttpServlet Response.class);
       4.2 调用方法
             m.invoke(obj,request,response);      --3.servlet的service方法被调用
五、当tomcat服务器中止或web应用从新部署,经过反射调用destroy方法
      5.1 获得方法对象
            Method m = clazz.getDeclareMethod("destroy",null);
     5.2 调用方法
           m.invoke(obj,null);    --4.servlet的destroy方法被调用
九、怎样证实Servlet是单例的?
经过构造函数来证实,当屡次请求(访问)servlet时,若是构造函数只被执行一次,说明servlet是单例的。
Servlet默认是单例的,是在第一次请求被执行时建立的(懒汉式)。
十、Servlet的自动加载
默认状况下,servlet对象是在第一次请求被执行时建立的。若是servlet的构造方法或init方法中执行了比较多的逻辑代码,那么致使用户第一次访问sevrlet的时候比较慢。
怎样解决这种问题呢?
改变servlet建立对象的时机:提早到加载web应用的时候建立!
方法:只须要在servlet的配置信息中,加上一个<load-on-startup>便可!
<servlet>
<servlet-name>ServletDemo</servlet-name>
<servlet-class>zzuli.edu.cn.ServletDemo</servlet-class>
<!-- 让servlet对象在启动时自动加载 -->
<load-on-startup>1</load-on-startup> 注意: 整数值越大,建立优先级越低!!
</servlet>
十一、Servlet的多线程并发问题(重点)
注意: servlet对象在tomcat服务器中是单实例多线程的。
由于servlet是多线程的,因此当多个servlet的线程同时访问了servlet的共享数据,如成员变量,可能会引起线程安全问题。
解决线程不安全问题办法:
    1)把使用到共享数据的代码块进行同步(使用synchronized关键字进行同步)
    2)在servlet类中尽可能不要使用XM返佣www.kaifx.cn/broker/xm.html成员变量。若是确实要使用成员,必须同步,并且尽可能缩小同步代码块的范围(哪里使用到了成员变量,就同步哪里!!),以免由于同步而致使并发效率下降。
    3)只要在servlet类中不使用共享数据,就不影响servlet的正常使用。
线程不安全代码演示:
/**html

  • @classDesc: 功能描述:(线程不安全演示)br/>*/
    @WebServlet("/ServletlDemo")
    public class ServletlDemo extends HttpServlet {
    private int i = 1;br/>@Override
    public void init() throws ServletException {}
    @Override
    br/>System.out.println("ServletlDemo...init()");
    }
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    resp.setCharacterEncoding("utf-8");// 内容编码,防止出现中文乱码
    resp.setContentType("text/html;charset=utf-8");
    // 向浏览器输出内容
    resp.getWriter().write("这是第" + i + "次访问...");br/>i++;
    }
    @Override
    public void destroy() {}
    }
    当用两个浏览器同时访问时,会出现线程不安全问题,以下图所示 
    线程安全代码:(使用synchronized)
    @WebServlet("/ServletDemo")
    br/>System.out.println("ServletlDemo...destroy()");
    }
    }
    当用两个浏览器同时访问时,会出现线程不安全问题,以下图所示 
    线程安全代码:(使用synchronized)
    @WebServlet("/ServletDemo")
    public class ServletDemo extends HttpServlet {
    private int i = 1;br/>@Override
    public void init() throws ServletException {}
    @Override
    br/>System.out.println("ServletDemo...init()");
    }
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    resp.setCharacterEncoding("utf-8");// 内容编码,防止出现中文乱码
    resp.setContentType("text/html;charset=utf-8");
    synchronized (ServetlDemo.class) {
    // 向浏览器输出内容
    resp.getWriter().write("这是第" + i + "次访问...");
    try {
    Thread.sleep(5000);
    } catch (Exception e) {br/>}i++;}}@Overridepublic void destroy() {System.out.println("ServletDemo...destroy()");}}此时用两个浏览器同时访问,不会出现线程不安全问题十二、ServletContext对象1)获得web应用上下文路径:ServletContext servletContext=this.getServletContext();String path=servletContext.getContextPath();2)ServletContext域对象:做用范围在整个web应用中有效!     经常使用方法:     保存数据:void setAttribute(java.lang.String name, java.lang.Object object)                                  获取数据:java.lang.Object getAttribute(java.lang.String name)       删除数据:void removeAttribute(java.lang.String name)
相关文章
相关标签/搜索