JavaWeb核心之Servlet
1. Servlet: server applet
- 概念:运行在服务器端的小程序
- Servlet就是一个接口,<mark>定义了Java类被浏览器访问到(tomcat识别)的规则。</mark>
- 将来我们自定义一个类,实现Servlet接口,复写方法。
2. 快速入门
- 创建JavaEE项目
- 定义一个类,实现Servlet接口
- public class ServletDemo1 implements Servlet
- 实现接口中的抽象方法
- 配置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对象
- 概念:代表整个web应用,可以和程序的容器(服务器)进行通信,要和装区分开来。
- 获取:
- 通过request对象获取
request.getServletContext(); - 通过HttpServlet类获取
this.getServletContext();
- 通过request对象获取
/**
* 获取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);
}
}
-
功能
- 获取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); }
-
域对象:可以共享数据,区别于request的请求转发
请求转发:- 状态栏地址不变
- 运行servlet自动转发到新的servlet并运行新的servlet
ServletContext:
- 使用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); } }
- 获取文件的真实(服务器)路径
* 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 }