1. Servlet: server applet

  1. 概念:运行在服务器端的小程序
    1. Servlet就是一个接口,<mark>定义了Java类被浏览器访问到(tomcat识别)的规则。</mark>
    2. 将来我们自定义一个类,实现Servlet接口,复写方法。

2. 快速入门

  1. 创建JavaEE项目
  2. 定义一个类,实现Servlet接口
    • public class ServletDemo1 implements Servlet
  3. 实现接口中的抽象方法
  4. 配置Servlet
    在web.xml中配置:
<!--配置Servlet -->
	    <servlet>
	        <servlet-name>demo1</servlet-name>
	        <servlet-class>cn.itcast.web.servlet.ServletDemo1</servlet-class>
	    </servlet>
	
	    <servlet-mapping>
	        <servlet-name>demo1</servlet-name>
	        <url-pattern>/demo1</url-pattern>
	    </servlet-mapping>

2.1 执行原理

1. 当服务器接受到客户端浏览器的请求后,会解析请求URL路径,获取访问的Servlet的资源路径
2. 查找web.xml文件,是否有对应的< url-pattern>标签体内容。
3. 如果有,则会通过url-pattern找到servlet-name,最后再通过servlet-name在找到对应的< servlet-class>全类名
4. tomcat会将字节码文件加载进内存,通过反射,并且创建其对象
5. 调用其方法

2.2 Servlet中的生命周期方法

1. 被创建:执行init方法,只执行一次
	* Servlet什么时候被创建(可修改)?
		* 默认情况下,第一次被访问时,Servlet被创建
		* 可以配置执行Servlet的创建时机。
			* 在< servlet>标签下配置
				1. 第一次被访问时,创建
            		* < load-on-startup>的值为负数
	            2. 在服务器启动时,创建
	                * < load-on-startup>的值为0或正整数
	* Servlet的init方法,只执行一次,说明一个Servlet在内存中只存在一个对象,Servlet是单例的
		* 多个用户同时访问时,可能存在线程安全问题,如果加锁将大大影响性能。
		* 解决:尽量不要在Servlet中定义成员变量。即使定义了成员变量,也不要对修改值

2. 提供服务:执行service方法,执行多次
	* 每次访问Servlet时,Service方法都会被调用一次。
3. 被销毁:执行destroy方法,只执行一次
	* Servlet被销毁时执行。服务器关闭时,Servlet被销毁
	* 只有服务器正常关闭时,才会执行destroy方法。
	* destroy方法在Servlet被销毁之前执行,一般用于释放资源

2.3 Servlet3.0

* 好处:
	* 支持注解配置。可以不需要web.xml了。

* 步骤:
	1. 创建JavaEE项目,选择Servlet的版本3.0以上,可以不创建web.xml
	2. 定义一个类,实现Servlet接口
	3. 复写方法
	4. 在类上使用@WebServlet注解,进行配置
		* @WebServlet("资源路径")
@WebServlet注解:
@Target({
   ElementType.TYPE})
			@Retention(RetentionPolicy.RUNTIME)
			@Documented
			public @interface WebServlet {
   
			    String name() default "";//相当于<Servlet-name>
			
			    String[] value() default {
   };//代表urlPatterns()属性配置
			
			    String[] urlPatterns() default {
   };//相当于<url-pattern>
			
			    int loadOnStartup() default -1;//相当于<load-on-startup>
			
			    WebInitParam[] initParams() default {
   };
			
			    boolean asyncSupported() default false;
			
			    String smallIcon() default "";
			
			    String largeIcon() default "";
			
			    String description() default "";
			
			    String displayName() default "";
			}

2.4 IDEA与tomcat相关配置

1. IDEA会为每一个tomcat部署的项目单独建立一份配置文件
	* 查看控制台的log:Using CATALINA_BASE:   "C:\Users\fqy\.IntelliJIdea2018.1\system\tomcat\_itcast"
2. 工作空间项目    和     tomcat部署的web项目
	* tomcat真正访问的是“tomcat部署的web项目”,"tomcat部署的web项目"对应着"工作空间项目" 的web目录下的所有资源
	* WEB-INF目录下的资源不能被浏览器直接访问。
3. 断点调试:使用"小虫子"启动 dubug 启动

2.5 Servlet的体系结构

Servlet -- 接口
 	|
GenericServlet -- 抽象类
	|
HttpServlet  -- 抽象类

* GenericServlet:将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象
	* 将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可
* HttpServlet:对http协议的一种封装,简化操作
	1. 定义类继承HttpServlet
	2. 复写doGet/doPost方法

2.6 Servlet相关配置

1. urlpartten:Servlet访问路径
	1. 一个Servlet可以定义多个访问路径 : @WebServlet({"/d4","/dd4","/ddd4"})
	2. 路径定义规则:
		1. /xxx:路径匹配,特例:/*,星号表示输入任意字符都行
		2. /xxx/xxx:多层路径,目录结构
		3. *.do:扩展名匹配,注意 星号前面不能有/

2.7 ServletContext对象

  1. 概念:代表整个web应用,可以和程序的容器(服务器)进行通信,要和装区分开来。
  2. 获取:
    1. 通过request对象获取
      request.getServletContext();
    2. 通过HttpServlet类获取
      this.getServletContext();
/**
 * 获取ServletContext对象
 */
@WebServlet("/servletContextDemo1")
public class ServletContextDemo1  extends HttpServlet{
   
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
   
        //两种获取ServletContext对象的方法
        ServletContext context1 = req.getServletContext();
        ServletContext context2 = this.getServletContext();
        //输出
        System.out.println(context1);
        System.out.println(context2);
        System.out.println(context1 == context2);  //true
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
   
        this.doGet(req,resp);
    }
}
  1. 功能

    1. 获取MIME类型
    在这里插入代码片    @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            ServletContext servletContext = this.getServletContext();
            String file = "a.jpg";
            String mimeType = servletContext.getMimeType(file);
            System.out.println(mimeType);
        }
    
    1. 域对象:可以共享数据,区别于request的请求转发
      请求转发

      1. 状态栏地址不变
      2. 运行servlet自动转发到新的servlet并运行新的servlet

      ServletContext:

      1. 使用ServletContext对象存储数据,运行servlet共享数据后,所以部署在tomcat上的servlet都可以收到数据,需要是就可以取出使用。
    /**
     * ServletContext共享数据
     */
    @WebServlet("/servletContextDemo3")
    public class ServletContextDemo3 extends HttpServlet{
         
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            ServletContext servletContext = this.getServletContext();
            servletContext.setAttribute("msg","Hello I am servletContextDemo3...");
        }
        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            this.doGet(req,resp);
        }
    }
    
    **
     * 获取ServletContextDemo3共享的数据
     */
    @WebServlet("/servletContextDemo4")
    public class ServletContextDemo4 extends HttpServlet{
         
    
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            ServletContext servletContext = this.getServletContext();
            Object msg = servletContext.getAttribute("msg");
            System.out.println(msg);
        }
        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            this.doGet(req,resp);
        }
    }
    
    1. 获取文件的真实(服务器)路径
     * tomcat工程名
     *      src
     *          a.txt
     *      web
     *          WEB-INF
     *              c.txt
     *          b.txt
    

    获取以上配置文件a.txt,b.txt,c.txt的资源路径

    //获取b.txt
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            ServletContext servletContext = this.getServletContext();
            String realPath = servletContext.getRealPath("/b.txt");  //打印出来的路径是被发布到服务器的真实路径 并被服务器所访问的
            System.out.println(realPath); // C:\Users\lzy\IdeaProjects\helloTomcat\out\artifacts\day05_war_exploded\a.txt
        }
    
    //获取c.txt
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            ServletContext servletContext = this.getServletContext();
            String realPath = servletContext.getRealPath("/WEB-INF/c.txt");
            System.out.println(realPath); //输出:C:\Users\lzy\IdeaProjects\helloTomcat\out\artifacts\day05_war_exploded\WEB-INF\c.txt
            //其中 /代表的就是C:\Users\lzy\IdeaProjects\helloTomcat\out\artifacts\day05_war_exploded
    
        }
    
    //获取a.txt
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         
            ServletContext servletContext = this.getServletContext();
            String realPath = servletContext.getRealPath("/WEB-INF/classes/a.txt");
            System.out.println(realPath); //输出:C:\Users\lzy\IdeaProjects\helloTomcat\out\artifacts\day05_war_exploded\WEB-INF\classes\a.txt
        }