深入理解Servlet

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接: https://blog.csdn.net/Mr_wxc/article/details/102730312

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
	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

4、在web.xml中配置 <servlet>和<servlet-mapping>的作用?

servlet容器对url的匹配过程:

当一个请求发送到servlet容器的时候,容器先会将请求的url减去当前应用上下文的路径作为servlet的映射url,比如我访问的是http://localhost:8080/FirstServlet/Servlet,我的应用上下文是FirstServlet,容器会将http://localhost:8080/FirstServlet去掉,剩下的/Servlet部分拿来做servlet的映射匹配。

扫描二维码关注公众号,回复: 7590286 查看本文章

映射匹配步骤:

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>

使用方法:编写一个servlet类,继承HttpServlet,然后在servlet上面加上@webServlet即可。

@WebServlet("/FirstServlet2")
public class FirstServlet2 extends HttpServlet{
	@Override
	protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
		System.out.println("This is Servlet2!");
		resp.getWriter().write("This is Servlet2!");
	}
}

注意点:@WebServlet注解括号里面必须加映射路径

7、Sevlet的生命周期(重点)

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类中尽量不要使用成员变量。如果确实要使用成员,必须同步,而且尽量缩小同步代码块的范围(哪里使用到了成员变量,就同步哪里!!),以避免因为同步而导致并发效率降低。

    3)只要在servlet类中不使用共享数据,就不影响servlet的正常使用。

线程不安全代码演示:

/**
 * @classDesc: 功能描述:(线程不安全演示)
 */
@WebServlet("/ServletlDemo")
public class ServletlDemo extends HttpServlet {
	private int i = 1;

	@Override
	public void init() throws ServletException {
		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 + "次访问...");
			i++;
	}

	@Override
	public void destroy() {
		System.out.println("ServletlDemo...destroy()");

	}
}

当用两个浏览器同时访问时,会出现线程不安全问题,如下图所示

 

线程安全代码:(使用synchronized)

@WebServlet("/ServletDemo")
public class ServletDemo extends HttpServlet {
	private int i = 1;

	@Override
	public void init() throws ServletException {
		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) {
			}
			i++;
		}
	}

	@Override
	public void destroy() {
		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)

猜你喜欢

转载自blog.csdn.net/Mr_wxc/article/details/102730312