博客 |
---|
JavaEE |
Servlet概念
概念:Servlet 运行在服务端的Java小程序,是sun公司提供一套规范(接口),用来处理客户端请求、响应给浏览器的动态资源。但servlet的实质就是java代码,通过java的API,动态的向客户端输出内容。
servlet规范:包含三个技术点
1)servlet技术
2)filter技术—过滤器
3)listener技术—监听器
Servlet快速入门
步骤:
1. 创建JavaEE项目
2. 定义一个类,实现Servlet接口
3. 实现接口中的抽象方法
4. 在web.xml配置Servlet
- 创建JavaEE项目
- 定义一个类,实现Servlet接口和3. 实现接口中的抽象方法
public class ServletImpl implements Servlet {
public void init(ServletConfig servletConfig) throws ServletException {
}
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("servlet.....");
}
public void destroy() {
}
public String getServletInfo() {
return null;
}
public ServletConfig getServletConfig() {
return null;
}
}
- 在web.xml配置Servlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
version="3.1">
<servlet>
<servlet-name>demo</servlet-name>
<servlet-class>com.ginger.servlet.ServletImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>demo</servlet-name>
<url-pattern>/welcome</url-pattern>
</servlet-mapping>
</web-app>
结果:
servlet.....
Servlet执行原理
- 当服务器接受到客户端浏览器的请求后,会解析请求URL路径,获取访问的Servlet的资源路径
- 查找web.xml文件,是否有对应的标签体内容。
- 如果有,则在找到对应的全类名
- tomcat会将字节码文件加载进内存,并且创建其对象
- 调用其方法
servlet执行原理图解
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被销毁之前执行,一般用于释放资源。
Servlet3.0
好处:
支持注解配置。可以不需要web.xml了。
步骤:
1. 创建JavaEE项目,选择Servlet的版本3.0以上,可以不创建web.xml
2. 定义一个类,实现Servlet接口
3. 复写方法
4. 在类上使用@WebServlet注解,进行配置
@WebServlet(“资源路径”)
代码演示
@WebServlet(urlPatterns="/welcome")//请求时资源路径
public class ServletImpl implements Servlet {
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("servlet3.0 执行了.....");
}
public void init(ServletConfig servletConfig) throws ServletException {}
public ServletConfig getServletConfig() {return null;}
public String getServletInfo() {return null;}
public void destroy() {}
}
访问Servlet
控制台结果
servlet3.0 执行了.....
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 "";
}
Servlet的体系结构
Servlet -- 接口
|
GenericServlet -- 抽象类
|
HttpServlet -- 抽象类
GenericServlet:将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象
将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可
HttpServlet:对http协议的一种封装,简化操作
1. 定义类继承HttpServlet
2. 复写doGet/doPost方法
Servlet相关配置
-
urlpartten:Servlet3.0访问资源路径
1. 一个Servlet可以定义多个访问路径 : @WebServlet({"/d4","/dd4","/ddd4"})
2. 路径定义规则:
1. /xxx:路径匹配
2. /xxx/xxx:多层路径,目录结构
3. *.do:扩展名匹配urlpartten:Servlet2.5访问资源路径
1. 其中url-pattern的配置方式:
1)完全匹配 访问的资源与配置的资源完全相同才能访问到
2)目录匹配 格式:/虚拟的目录…/* 代表任意
3 ) 扩展名匹配 格式:.扩展名
注意:第二种与第三种不要混用 /aaa/bbb/*.abcd(错误的)
2.缺省Servlet
我认为缺省的Servlet对了解Servlet有很大重用。
如果将web.xml文件中的url-pattern配置一个/,代表该servlet是缺省的servlet。
什么是缺省的servlet?
当你访问资源地址所有的servlet都不匹配时 , 缺省的servlet负责处理。
其实,web应用中所有的资源的响应都是servlet负责,包括静态资源。
缺省的Servlet演示
1.定义一个Servlet
向页面输出response…
public class ServletImpl implements Servlet {
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
HttpServletResponse hsr =(HttpServletResponse) servletResponse;
hsr.getWriter().write("response.......");
}
public void init(ServletConfig servletConfig) throws ServletException {
}
public ServletConfig getServletConfig() {
return null;
}
public String getServletInfo() {
return null;
}
public void destroy() {
}
}
2.首先在web.xml文件中配置缺省Servlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
version="2.5">
<servlet>
<servlet-name>name</servlet-name>
<servlet-class>com.ginger.servlet.ServletImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>name</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
3.在写一个简单的HTML页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<h1>11111111111111111111</h1>
</body>
</html>
4.访问1.html页面
出现问题了,我想访问的是1.html页面,但是访问的是com.ginger.servlet.ServletImpl页面输出response…。
分析原因:首先是tomcat解析请求地址它会,会拿着1.html到web.xml去找/1.html但是web.xml根本没有,没有的话然后就去找缺省/也就找到了com.ginger.servlet.ServletImpl然后页面输出response…。
图解
5.那么问题来了,如果我没有在web.xml文件配置缺省的Servlet那么tomcat怎么找到1.html呢?
去掉缺省的Servlet 中web.xml内容
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5">
<servlet>
<servlet-name>name</servlet-name>
<servlet-class>com.ginger.servlet.ServletImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>name</servlet-name>
<url-pattern>/demo</url-pattern>
</servlet-mapping>
</web-app>
6.访问1.thml
首先是tomcat解析请求地址它会,会拿着1.html到web.xml去找/1.html动态资源但是没有。我自己项目中的web.xml缺省我已经去掉了,那tomcat是怎么去找到1.html呢?
其实是通过tomcat的自带的web.xml文件中也有缺省,也是通过该缺省找到1.html静态资源。(自己项目中的web.xml会继承tomcat中web.xml)
tomcat中自带的web.xm中的缺省,我只复制了一部分,也正是org.apache.catalina.servlets.DefaultServlet通过1.html名称去找web下的静态资源。
<servlet>
<servlet-name>default</servlet-name>
<servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>
<init-param>
<param-name>debug</param-name>
<param-value>0</param-value>
</init-param>
<init-param>
<param-name>listings</param-name>
<param-value>false</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
注意:如果是自己项目中web.xml文件中有缺省的Servlet会把tomca自带web.xml文件中的缺省会覆盖掉(就近原则)。也正是因为这样,最早上面其实使用的是自己项目中web.xml的缺省。响应的就是response…就没有使用tomcat自带的缺省,也就没有去找静态资源1.thml。
也就更加印证了web应用中所有的资源的响应都是servlet负责,包括静态资源。
3.欢迎页面
index.jsp页面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>$Title$</title>
</head>
<body>
来喽。。。。。。。。。。。。。。。。。。。。。
</body>
</html>
那么问题来了,为什么没有输入资源名称会自动访问index.jsp页面。
还是因为tomcat自带的web.xml文件中配置了欢迎页面,当然自己项目中的web.xml也可以配置欢迎页面,会自动覆盖tomcat自带的web.xml文件。
我只复制了tomcat自带web.xml文件一部分,在最底部。
</web-app>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.htm</welcome-file>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
</web-app>
IDEA与tomcat的相关配置
-
IDEA会为每一个tomcat部署的项目单独建立一份配置文件
查看控制台的log:Using CATALINA_BASE: “C:\Users\Administrator.IntelliJIdea2017.3\system\tomcat_basic-code” -
工作空间项目 和 tomcat部署的web项目
tomcat真正访问的是“tomcat部署的web项目”,“tomcat部署的web项目"对应着"工作空间项目” 的web目录下的所有资源
工作空间项目路径
tomcat部署的web项目路径
WEB-INF目录下的资源不能被浏览器直接访问。
3. 断点调试:使用"小虫子"启动 dubug 启动