servlet
流程:
1.编写程序
2.发布到服务器
3.服务器加载到文件,根据web.xml中的servlet的配置创建servlet对象(反射)
4.客服端的请求由服务器分配给相应的servlet处理
注意:每一次请求都会产生一个新的request对象
Servlet:运行在服务器端多线程方式处理客户的请求
运行原理:启动服务器 加载web.xml文件的时候创建servlet对象,服务器根据客服的请求动态的生成HTML页面
socket:传输层传输数据
序列化对象:对象持久化 (servlet请求就是对象,传输的就是对象)
按一定格式把对象转成字符串
(1)路径:web.xml中的
访问时:地址:端口号/工程/配置的servlet的<URL-Pattern>
(2)程序的写法,初期建议
服务器端的请求(路径中方的是web.xml配置的地址)
request.getRequestDispatcher(“MyServlet2”).forward(request
,response);
客服端的请求(/工程名/web.xml文件配置的路径)
response.sendRedirect(“/servlet2_3/MyServlet2”);
<a href=”/servlet2_3/MyServlet2”></a>
<form action=”/servlet2_3/MyServlet2”></form>
Window.location.href=”/工程名/web.xml文件”;
Window.open(“/工程名/web.xml文件“);
form 指定的是提交的区域 form外面的表单元素不会被提交
Post请求:长度没有限制,是一种安全的提交方式,放在请求体中
Get请求:长度有限制,不安全的提交方式 链接,open,window等都是get
servlet连接数据库时:当连接到数据库编码出错时,可以在选项修改编码:如utf-8,编码格式,应该跟myEclipce一致,UTF-8:和unicode:编码方式一样,前者会截取掉后面空的
服务器tomcat 内部文件详情:
清除jsp缓存,删除work目录
服务器加载时,把xml文件加载进去,同时通过xml
文件反射来创建文件//反射机制
jsp文件:是一种servlet,默认存放在根目录下
服务器跳转:从客服端发送一个请求到服务器:遇到req.getRequestDispatcher
("/m2"),直接跳转到下一个servlet;
客服端跳转:
把状态码和路径发回客服端,然后再请求到服务器跳转到下一个servelet(server applet)
当客户端第一次请求某个Servlet时,Servlet容器将会根据web.xml配置文件实例化这个Servlet类。当有新的客户端请求该Servlet时,一般不会再实例化该Servlet类多个线程在使用这个实例。Servlet容器会自动使用线程池等技术来支持系统的运行.
servlet的含义:运行在服务器端 以多线程的方式处理用户请求的小程序
//2servlet的生命周期 服务器启动加载web.xml文件的时候产生 服务器关闭销毁
//3注意servlet是非线程安全的
//4解决线程安全的方式
private String name;
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
name=request.getParameter("username");
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
PrintWriter print = response.getWriter();
print.print("welcome-->"+name);
print.flush();
print.close();
//解决线程安全问题
//方式一 创建同步代码块
/*synchronized (this) {//this指的是当前的servlet对象
name=request.getParameter("username");
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
PrintWriter print = response.getWriter();
print.print("welcome-->"+name);
print.flush();
print.close();
}*/
//方式二 实现SingleThreadModel接口
//注意:尽量不要实现 SingleThreadModel 接口。这种实践将导致 Web 容器创建多个 servlet 实例;即为每个用户创建一个实例。对于任何大小的应用程序,这种实践都将导致严重的性能问题。
/* name=request.getParameter("username");
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
PrintWriter print = response.getWriter();
print.print("welcome-->"+name);
print.flush();
print.close();*/
/*//方式三 尽量不要让多个请求对象操作同一个属性值
String name=request.getParameter("username");//设置属性私有,可以避免线程非安全
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
PrintWriter print = response.getWriter();
print.print("welcome-->"+name);
print.flush();
print.close();*/
}
Servlet生命周期分为三个阶段:
1,初始化阶段 调用init()方法
2,响应客户请求阶段 调用service()方法
3,终止阶段 调用destroy()方法
Servlet初始化阶段:
在下列时刻Servlet容器装载Servlet:
1,Servlet容器启动时自动装载某些Servlet,实现它只需要在web.XML文件中的<Servlet></Servlet>之间添加如下代码:<loadon-startup>1</loadon-startup>
2,在Servlet容器启动后,客户首次向Servlet发送请求
3,Servlet类文件被更新后,重新装载Servlet
Servlet被装载后,Servlet容器创建一个Servlet实例并且调用Servlet的init()方法进行初始化。在Servlet的整个生命周期内,init()方法只被调用一次。
Servlet工作原理:
首先客户发送一个请求,Servlet是调用service()方法对请求进行响应的,通过源代码可见,service()方法中对请求的方式进行了匹配,选择调用doGet,doPost等这些方法,然后再进入对应的方法中调用逻辑层的方法,实现对客户的响应。在Servlet接口和GenericServlet中是没有doGet,doPost等等这些方法的,HttpServlet中定义了这些方法,但是都是返回error信息,所以,我们每次定义一个Servlet的时候,都必须实现doGet或doPost等这些方法。
每一个自定义的Servlet都必须实现Servlet的接口,Servlet接口中定义了五个方法,其中比较重要的三个方法涉及到Servlet的生命周期,分别是上文提到的init(),service(),destroy()方法。GenericServlet是一个通用的,不特定于任何协议的Servlet,它实现了Servlet接口。而HttpServlet继承于GenericServlet,因此HttpServlet也实现了Servlet接口。所以我们定义Servlet的时候只需要继承HttpServlet即可。
Servlet接口和GenericServlet是不特定于任何协议的,而HttpServlet是特定于HTTP协议的类,所以HttpServlet中实现了service()方法,并将请求ServletRequest,ServletResponse强转为HttpRequest和HttpResponse。
public void service(ServletRequest req,ServletResponse res)
throws ServletException,IOException
{
HttpRequest request;
HttpResponse response;
try
{
req = (HttpRequest)request;
res = (HttpResponse)response;
}catch(ClassCastException e)
{
throw new ServletException("non-HTTP request response");
}
service(request,response);
}
代码的最后调用了HTTPServlet自己的service(request,response)方法,然后根据请求去调用对应的doXXX方法,因为HttpServlet中的doXXX方法都是返回错误信息,
protected void doGet(HttpServletRequest res,HttpServletResponse resp)
throws ServletException,IOException
{
String protocol = req.getProtocol();
String msg = IStrings.getString("http.method_get_not_supported");
if(protocol.equals("1.1"))
{
resp.sendError(HttpServletResponse.SC.METHOD.NOT.ALLOWED,msg);
}
esle
{
resp.sendError(HttpServletResponse.SC_BAD_REQUEST,msg);
}
}
所以需要我们在自定义的Servlet中override这些方法!
Servlet响应请求阶段:
对于用户到达Servlet的请求,Servlet容器会创建特定于这个请求的ServletRequest对象和ServletResponse对象,然后调用Servlet的service方法。service方法从ServletRequest对象获得客户请求信息,处理该请求,并通过ServletResponse对象向客户返回响应信息。
对于Tomcat来说,它会将传递过来的参数放在一个Hashtable中,该Hashtable的定义是:
private Hashtable<String String[]> paramHashStringArray =new Hashtable<String String[]>();
这是一个String-->String[]的键值映射。
HashMap线程不安全的,Hashtable线程安全。
Servlet终止阶段:
当WEB应用被终止,或Servlet容器终止运行,或Servlet容器重新装载Servlet新实例时,Servlet容器会先调用Servlet的destroy()方法,在destroy()方法中可以释放掉Servlet所占用的资源。
Servlet何时被创建:
1,默认情况下,当WEB客户第一次请求访问某个Servlet的时候,WEB容器将创建这个Servlet的实例。
2,当web.xml文件中如果<servlet>元素中指定了<load-on-startup>子元素时,Servlet容器在启动web服务器时,将按照顺序创建并初始化Servlet对象。
注意:在web.xml文件中,某些Servlet只有<serlvet>元素,没有<servlet-mapping>元素,这样我们无法通过url的方式访问这些Servlet,这种Servlet通常会在<servlet>元素中配置一个<load-on-startup>子元素,让容器在启动的时候自动加载这些Servlet并调用init()方法,完成一些全局性的初始化工作。
Web应用何时被启动:
1,当Servlet容器启动的时候,所有的Web应用都会被启动
2,控制器启动web应用