servlet的404错误困扰了两天,各种方法都试过了,翻书逛论坛终于把问题解决了,写此博客来纪念自己的第一个servlet经历。
下面我会将自己的编写第一个servlet的详细过程提供给初学者,大神们可以自动过滤掉。下面的步骤有这样的前提:你的eclipse嵌入了TomCat插件(我们会在Eclipse上操作servlet)或者本地主机已经有安装好某一个版本的TomCat,我这里用的是Tomcat7.0.69.
第一步:创建Dynamic Web Project
Eclipse下->new->other->web->Dynamic Web Project,名称我们就叫TestServlet2
TargetRuntime这个选项,如果你是第一次使用Tomcat的话应该还没有配置,选择右边的New Runtime会出现下面的页面:
我选择的是v7.0,因为我电脑上装的就是v7.0.69;
Configuration右面的选项Modify:我们暂时只需要Dynamic和Java两项就可以,把那个JavaScript前面的勾去掉,点击Ok;
点击next:
src是我们源代码的目录位置,下面的Default output folder是我们源代码编译之后的.class文件的输出位置,我们这里改成WEB-INF/classes(为了与我们把应用发布到TomCat的webapps下的目录一致,方便理解,此时我们的build目录就没啥用了,等下它自动生成的时候就可以删除了);
继续点击next:
直接默认就可以了,Context root是我们web应用的主目录,Context directory是我们存放JSP文件其他文件的,里面包括两个文件夹META-INF和WEB-INF,最后勾选自动配置web.xml文件;
现在的项目目录如下图所示:
第二步:写我们的servlet
在src目录下新建一个class,类名就叫做HelloWorldServlet,添加一个包:com.ysw.servlet
在HelloWorldServlet中添加如下代码:我们这里用extends HttpServlet的方式来编写,其他两种方式大家可以自己Google一下;
package com.ysw.servlet; import java.io.IOException; import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class HelloWorldServlet extends HttpServlet { private static final long serialVersionUID = 1L; @Override
protected void doGet(HttpServletRequest request,
HttpServletResponse response) throws ServletException, IOException {
System.out.println("doGet");
}
}
实现的简单功能是在我们Eclipse的控制台打印"doGet"字符串;
第三步:配置文件web.xml
<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_3_0.xsd"
version="3.0"> <servlet>
<description>This is the description of my J2EE component</description>
<display-name>This is the display name of my J2EE component</display-name>
<servlet-name>HW</servlet-name>
<servlet-class>com.ysw.servlet.HelloWorldServlet2</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HW</servlet-name>
<url-pattern>/abc</url-pattern>
</servlet-mapping>
<session-config>
<session-timeout>20</session-timeout>
</session-config>
</web-app>
在我们的WEB-INF目录下有一个web.xml文件,我们在其中添加如下内容:每一个标签代表啥意思大家自己查一下,学习的过程是痛并快乐着;
我还是说一下吧:<servlet-name>是给我们的servlet起一个名字,可以是任意的;<servlet-class>指向我们要注册的servlet的类地址,这里一定要注意:如果你的程序使用了包,比如我的程序中的包名是com.ysw.servlet
<servlet-mapping>中的servlet-name与我们上面定义的<servlet>中的那个<servelt-name>一定要一致;<url-pattern>意思是我们在浏览器中输入/abc这种形式就能访问到Tomcat上相应的资源
第四步:修改Project下的.class文件
在我的电脑上是这样的:
编辑.classpath文件,将<classpathentry kind="output" path="build/classes"/>这一项中的path改成我们前面已经说过的WEB-INF/classes目录,即:<classpathentry kind="output" path="WEB-INF/classes"/>保存退出;
【重要的一步】:右键工程(TestServlet2)->Build Project,eclipse一般默认都是自动编译代码,只要你的代码修改后保存,eclipse是会自动编译成class文件的。但如果你修改过配置,eclipse不是自动编译方式,那么就要点击这个build project选项手动把代码编译成class文件了。
第五步:在Tomcat上运行我们的servlet
点击以后出现下面的界面:选择我们先前配置的相应版本的Tomcat7.0.69,你的机器因人而异
点击->next,会让我们选择要运行的servlet,将要运行的servley移动到右面区域,点击->finashed这是就可以从控制台开到Tomcat的启动,
你看到的是这样的情况:
【没有消息就是最好的消息】:最下面切换到console这一视图,刷新一下我们上图的那个浏览器,我们可以看到:
可以看到我们的打印输出信息;
第六步:将我们的web应用部署到Tomcat服务器
在Tomcat的webapps目录下新建一个名称为myWebSite的文件夹,这个文件夹就是我们web应用的根目录。在myWebSite内新建一个名称叫做WEB-INF(这个名称是固定的,不能用其他名称),然后在WEB-INF文件夹内新建一个名称为classes的文件夹,用于存放我们编译好的.class文件,在WEB-INF文件夹内还有一个文件,就是我们前面提到的web.xml文件,直接放在这个位置就可以了.
在class文件夹下放我们编译好的class文件,其实只要把我们在的工程TestServlet2下的WEB-INF下的class文件夹拷贝过来就行(这个时候我们在前面的操作:<classpathentry kind="output" path="WEB-INF/classes"/>),
在这里就方便我们的应用发布了,因为我程序中用到了包com.ysw.servlet,因此class文件下也包括了各级目录;
接下来就是要用浏览器访问我们的servlet:首先要关闭掉Eclipse中我们之前一直运行的TomCat服务,否则我们在cmd下无法启动TomCat;
启动服务以后可以清楚的看到我们的应用得到了配置和部署:
在浏览器下我们输入:http://localhost:8080/myWebSite/abc
可以看到:doGet得以输出,至此我的第一个Servlet完成!
cookie,session,application的区别:这部分转载自:http://blog.csdn.net/u010214269/article/details/44801567
Cookie: ①存在于客户端(可被阻止) ②只能是文本文档
③如果设置了期限值,则写入客户端的文件;
如果没有,它只对本窗口或其子窗口有效,其它窗口不能访问该Cookie
④在Servlet/JSP中设置的Cookie可以被同路径下或其子路径的Servlet/JSP访问,父路径不可以
○注:这里的路径是指URL,而不是Web文件的目录
Session: ①存在于服务器端 ②每个Session对应一个窗口,用SessionID标识,这个Session为该窗口及其子窗口共享
③有两种实现方式:a.使用Cookie(在Cookie可用时)b.URL重写(在Cookie被禁用时)
④没有访问路径的问题。同一个WebApplication下的Servlet/JSP设置的Session可以被互相访问
前提是:同一个浏览器窗口或其子窗口
Application: 与一个Web应用程序相对应,该应用程序下所有的Servlet/JSP共用该Application中的信息(ServletContext) 1、session保存在服务器,客户端不知道其中的信息;cookie保存在客户端,服务器能够知道其中的信息。 2、session中保存的是对象,cookie中保存的是字符串。 3、session不能区分路径,同一个用户在访问一个网站期间,所有的session在任何一个地方都可以访问到。而cookie中如果设置了路径参数,
那么同一个网站中不同路径下的cookie互相是访问不到的。 4、session需要借助cookie才能正常工作。如果客户端完全禁止cookie,session将失效。 但是如果服务器端启用了url编码,也就是用
URLEncoder.encode()把所有的url编码了,则会在url后面出现如下类似的东西
index.jsp:jsessionid=fdsaffjdlks;jaf;lkdjsf 服务器通过这个进行session的判断 5 session在用户会话结束后就会关闭了,但cookie因为保存在客户端,可以长期保存 Cookie存储的信息是放到客户端的,用户在访问服务器端页面时,必然在客户端和服务器端之间频繁交换信息,影响了程序的性能。而Session由于存储在服务器内存中,因此不存在这个问题。不过,Session存储的信息是临时的,用户一旦关闭浏览器,状态即失去。而Cookie则相反。COOKIE 是本地文件,是 40 大盗在阿里巴巴家做的记号,或者是送牛奶的人在你家门口钉的箱子。 Application状态为应用程序提供了一个全局的状态。所有客户都可以使用该状态。从设计的角度来说,我们通常用Application来存储一些标准的数据。同时,我们在使用它时要注意避免性能的降低,存储的数据尽可能提供给客户只读的功能。 APPLICATION 是公共浴池。在这里能看见所有人,包括 ppmm 哦:)。 Session和客户端的Cookie是有关的,当客户关掉Cookie时,Session就失效了,SESSION 是服务器端内存,是你洗澡时浴池发给你的钥匙。自己专用,可以开自己的好多箱子。 Application: application是应用级别的,同一个应用中的所有用户将共享此对象。因此,我们利用application对象来存储一些坏境信息。 Request: Request也可以像Session那样使用,但是它第二次请求服务器页面时就失去存储功能了 request的范围只在一jsp页发出请求到另一页之间,随后这个属性失效;
session范围是用户和服务器连接的那段时间,用户与服务器断开属性就失效;
application作用范围最大,慎用,在服务器一开始执行服务到服务器关闭为止。可能造成服务器负载过重。
Servlet的生命周期
* API中的生命周期:一个servlet只有一个对象(实例)
* 1.加载(classLoader)将类加载进来
* 2.实例化,new一个对象
* 3.初始化:调用init(ServletConfig)方法
* 4.处理请求:service doGet doPost(http协议)
* 5.退出服务:destory()
* init():只执行一次
* destory():
* doGet():
* service():
验证servlet的生命周期的代码:
package com.ysw.servlet; import java.io.IOException; import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; /**
*
* @author Administrator
*
*/
public class TestLifeCycleServlet extends HttpServlet { /**
* API中的生命周期:一个servlet只有一个对象(实例)
* 1.加载(classLoader)将类加载进来
* 2.实例化,new一个对象
* 3.初始化:调用init(ServletConfig)方法
* 4.处理请求:service doGet doPost(http协议)
* 5.退出服务:destory()
* init():只执行一次
* destory():
* doGet():
* service():
*/
private static final long serialVersionUID = 1L;
@Override
public void destroy() { System.out.println("destory");
} @Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException { System.out.println("doGet");
} @Override
public void init(ServletConfig config) throws ServletException { System.out.println("init");
} public TestLifeCycleServlet() {
System.out.println("constructor!");
} }