温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

Servlet的多线程并发问题

发布时间:2020-05-30 12:21:34 来源:网络 阅读:318 作者:专注地一哥 栏目:编程语言

1、Servlet : 用java语言编写的动态资源开发技术。
2、Servlet 特点:
1)普通的java类,继承HttpServlet类,覆盖doGet、doPost等方法。
2)Servlet类只能交给tomcat服务器运行。
3、怎样使用Eclipse开发Servlet?
1)编写一个servlet类,继承HttpServlet
public class Servlet extends HttpServlet{@Override
br/>@Override
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
4、在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对象里面的方法。
5、 Myeclipse和Eclipse中的Tomcat怎样部署项目?
1)Myeclipse默认将项目部署到tomcat安装目录下的webapps中
2)eclipse并不像MyEclipse默认将项目部署到tomcat安装目录下的webapps中,而是默认部署到工作目录(workspace)下的.metadata/.plugins/org.eclipse.wst.server.core/tmp0/wtpwebapps中
6、Servlet 注解Servlet3.0以上可以使用注解@WebServlet自动映射,不用在web.xml中配置<servlet>和<servlet-mapping>
br/>Servlet3.0以上可以使用注解@WebServlet自动映射,不用在web.xml中配置<servlet>和<servlet-mapping>
br/>@WebServlet("/FirstServlet2")
br/>@Override
System.out.println("This is Servlet2!");
resp.getWriter().write("This is Servlet2!");}
}
注意点:@WebServlet注解括号里面必须加映射路径
br/>}
}
注意点:@WebServlet注解括号里面必须加映射路径
Servlet重要的四个生命周期方法:
1)构造方法:创建servlet对象的时候调用。默认情况下,第一次访问servlet的时候创建servlet对象。只调用1次,证明servlet对象在tomcat是单实例的。
2)init方法:创建完servlet对象的时候调用,只调用1次。
3)service方法:每次发出请求时调用,调用n次。
4)destroy方法:销毁servlet对象的时候调用。停止服务器或者重新部署web应用时销毁servlet对象。只调用1次。
8、伪代码演示servlet的生命周期
Tomtcat内部代码运行:
1、通过URL映射找到到servlet-class的内容(zzuli.edu.cn.FirstServlet)
2、通过反射构造FirstServlet对象
     2.1 得到字节码对象
           Class clazz = class.forName("zzuli.edu.cn.FirstServlet");
     2.2 调用无参数的构造方法来构造对象
          Object obj = clazz.newInstance();  ---1.servlet的构造方法被调用
3、创建ServletConfig对象,通过反射调用init方法
     3.1 得到方法对象
           Method m = clazz.getDeclareMethod("init",ServletConfig.class);
      3.2 调用方法
            m.invoke(obj,config);     --2.servlet的init方法被调用
4、创建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方法被调用
5、当tomcat服务器停止或web应用重新部署,通过反射调用destroy方法
      5.1 得到方法对象
            Method m = clazz.getDeclareMethod("destroy",null);
     5.2 调用方法
           m.invoke(obj,null);    --4.servlet的destroy方法被调用
9、怎样证明Servlet是单例的?
通过构造函数来证明,当多次请求(访问)servlet时,如果构造函数只被执行一次,说明servlet是单例的。
Servlet默认是单例的,是在第一次请求被执行时创建的(懒汉式)。
10、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>
11、Servlet的多线程并发问题(重点)
注意: servlet对象在tomcat服务器中是单实例多线程的。
因为servlet是多线程的,所以当多个servlet的线程同时访问了servlet的共享数据,如成员变量,可能会引发线程安全问题。
解决线程不安全问题办法:
    1)把使用到共享数据的代码块进行同步(使用synchronized关键字进行同步)
    2)在servlet类中尽量不要使用XM返佣www.kaifx.cn/broker/xm.html成员变量。如果确实要使用成员,必须同步,而且尽量缩小同步代码块的范围(哪里使用到了成员变量,就同步哪里!!),以避免因为同步而导致并发效率降低。
    3)只要在servlet类中不使用共享数据,就不影响servlet的正常使用。
线程不安全代码演示:
/**

  • @classDesc: 功能描述:(线程不安全演示)*/
    @WebServlet("/ServletlDemo")
    br/>*/
    @WebServlet("/ServletlDemo")
    private int i = 1;@Override
    br/>@Override
    }
    @Override
    br/>System.out.println("ServletlDemo...init()");
    }
    @Override
    resp.setCharacterEncoding("utf-8");// 内容编码,防止出现中文乱码
    resp.setContentType("text/html;charset=utf-8");
    // 向浏览器输出内容
    resp.getWriter().write("这是第" + i + "次访问...");i++;
    }
    @Override
    br/>i++;
    }
    @Override
    }
    }
    当用两个浏览器同时访问时,会出现线程不安全问题,如下图所示 
    线程安全代码:(使用synchronized)
    @WebServlet("/ServletDemo")
    br/>System.out.println("ServletlDemo...destroy()");
    }
    }
    当用两个浏览器同时访问时,会出现线程不安全问题,如下图所示 
    线程安全代码:(使用synchronized)
    @WebServlet("/ServletDemo")
    private int i = 1;@Override
    br/>@Override
    }
    @Override
    br/>System.out.println("ServletDemo...init()");
    }
    @Override
    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) {}
    i++;
    }
    }
    @Override
    br/>}
    i++;
    }
    }
    @Override
    System.out.println("ServletDemo...destroy()");
    }
    }
    此时用两个浏览器同时访问,不会出现线程不安全问题
    12、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)
向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI