servlet创建
- 编写servlet:写一个普通类,然后去实现或者继承javaEE规范中所提供的指定接口或者父类,然后再重写指定方法.
- 这个指定接口或者父类都在tomcat的lib目录里面的servlet-api.jar中
- 实现Servlet接口,重写里面的五种抽象方法,主要实现service方法
- 继承HttpServlet类,HttpServlet继承GenericServlet类重写doGet或者doPost方法
- 继承GenericServlet抽象类,实现service方法
- web.xml文件中配置自己编写好的servlet
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.briup.test.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
- servlet对象的创建,以及对象中指定方法的调用,都是由tomcat服务器来处理的
- web.xml配置文件又tomcat服务器在期间必须要读取的web项目的中的一个文件,所有要把自己写好的servlet相关信息配置到web.xml中,以便让tomcat及时的知道这些信息,然后就可以创建对象和调用方法了.
- 访问方式:浏览器访问指定uri:http://127.0.0.1:8989/web_test/hello
Servlet实现原理
- 什么是servlet:就是Servlet applet,Java EE中的组件;本质是一个java类,这种java类可以提供访问web形式的访问方法(接口规范,主要是servlet接口的service方法)
- Servlet的优点:
- 较好的移植性。Java语言具有跨平台性和可移植性强的特点,使用Servlet具有良好的移植性,无需修改代码可以部署到多种服务器平台上面。
- 执行效率高。Servlet针对每个请求都会创建一个线程来执行,而创建线程比创建CGI每次创建一个进程的开销要小,因此Servlet更具有响应时间短,效率高。
- 功能强大。可以和Web服务器进行交互。
- 使用方便。Servlet技术提供了多种接口和方法,处理请求和响应,以及cookie和会话跟踪技术。
- 可拓展性强。因为使用了Java进行编程。
- 按照Java EE的规范编写一个servlet
- a、实现Servlet接口:实现接口中的五个抽象方法,最主要是实现service方法,处理请求及响应.
- tomcat服务器main主函数入口调用service方法
- b、继承GenericServlet类,实现service抽象方法
- GenericServlet类中不但实现了Servlet接口中的init方法,而且还重载了一个无参的init()方法;
- tomcat默认调用Servlet中的init方法,最终调用的是GenericServlet中的实现的init方法;
- GenericServlet实现了ServletConfig接口,通过实现了servlet中的init方法,初始化ServletConfig接口引用;
@Override
public void init(ServletConfig config) throws ServletException {
this.config = config;
this.init();
}
public void init() throws ServletException {
// NOOP by default
}
- ServletConfig接口:规范了获取Servlet相关信息的方法(Servlet名字,初始化参数,application对象)。
- c、继承HttpServlet:重写doXxx方法,处理对应请求方法的请求
- HttpServlet继承了GenericServlet类;
- HttpServlet中有两个service方法,一个是实现了Servlet中,一个是重载的;两者的参数类型不一样;
- tomcat调用Servlet接口的service方法,其实是调用GenericServlet中的service方法,HttpServlet中的doGet方法和doPost方法都会接受请求和响应,所以两个方法都尽量必须实现,然后再根据请求行中的请求方法调用具体的方法。
源代码中俩个service方法的实现:
//实现Servlet接口中的service方法
@Override
public void service(ServletRequest req, ServletResponse res)
throws ServletException, IOException {
HttpServletRequest request;
HttpServletResponse response;
try {
request = (HttpServletRequest) req;
response = (HttpServletResponse) res;
} catch (ClassCastException e) {
throw new ServletException("non-HTTP request or response");
}
//调用重载之后的service方法
service(request, response);
}
//HttpServlet类中重载的service方法
protected void service(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
String method = req.getMethod();
if (method.equals(METHOD_GET)) {
long lastModified = getLastModified(req);
if (lastModified == -1) {
// servlet doesn't support if-modified-since, no reason
// to go through further expensive logic
doGet(req, resp);
} else {
long ifModifiedSince;
try {
ifModifiedSince = req.getDateHeader(HEADER_IFMODSINCE);
} catch (IllegalArgumentException iae) {
// Invalid date header - proceed as if none was set
ifModifiedSince = -1;
}
if (ifModifiedSince < (lastModified / 1000 * 1000)) {
// If the servlet mod time is later, call doGet()
// Round down to the nearest second for a proper compare
// A ifModifiedSince of -1 will always be less
maybeSetLastModified(resp, lastModified);
doGet(req, resp);
} else {
resp.setStatus(HttpServletResponse.SC_NOT_MODIFIED);
}
}
} else if (method.equals(METHOD_HEAD)) {
long lastModified = getLastModified(req);
maybeSetLastModified(resp, lastModified);
doHead(req, resp);
} else if (method.equals(METHOD_POST)) {
doPost(req, resp);
} else if (method.equals(METHOD_PUT)) {
doPut(req, resp);
} else if (method.equals(METHOD_DELETE)) {
doDelete(req, resp);
} else if (method.equals(METHOD_OPTIONS)) {
doOptions(req,resp);
} else if (method.equals(METHOD_TRACE)) {
doTrace(req,resp);
} else {
//
// Note that this means NO servlet supports whatever
// method was requested, anywhere on this server.
//
String errMsg = lStrings.getString("http.method_not_implemented");
Object[] errArgs = new Object[1];
errArgs[0] = method;
errMsg = MessageFormat.format(errMsg, errArgs);
resp.sendError(HttpServletResponse.SC_NOT_IMPLEMENTED, errMsg);
}