OSCHINA 软件库有一个分类——Web框架,该分类中包含多种编程语言的将近500个项目。html
Web框架是开发者在使用某种语言编写Web应用服务端时关于架构的最佳实践。不少Web框架是从实际的Web项目抽取出来的,仅和Web的请求和响应处理有关,造成一个基础,在开发别的应用项目的时候则能够从这个剥离出来的基础作起,让开发者更关注更具体的业务问题,而不是Web的请求和响应的控制。java
框架不少,但套路基本相似,帮你隐藏不少关于 HTTP 协议细节内容,专一功能开发。web
但对一个初学者来讲,过早的接触框架每每是事倍功半!一样一个问题,换一种框架你可能须要从头开始研究。express
下面是针对初学 Java 开发 Web 过程一些我的看法和思路,高手可略过。apache
1. 基本要求:Java 编程基础编程
有良好的 Java 语言编程基础,这是必须的,在讨论 Web 开发技术时提了一个 Java 编程基础的问题会被鄙视的。api
2. 环境准备 (Eclipse + Tomcat)浏览器
选择一个你喜好的Servlet容器,或者说大一点就是应用服务器,推荐 Tomcat 、Resin 或者 Jetty 这些轻量级的产品。这三个产品下载 zip 包解压后就能够用了。若是你不熟悉 Tomcat 的话请不要使用 exe 版本的 Tomcat,那会徒增不少烦恼。也不建议在 Eclipse 等一些开发环境中集成 Tomcat 的作法,也会徒增烦恼。tomcat
把应用服务器启动起来并能访问到其默认的页面为准。服务器
关于开发工具
不推荐使用 MyEclipse 和 Eclipse 的 JEE 版本,徒增烦恼、运行缓慢并且还让你没法了解 Web 项目的结构。普通的 Eclipse 或者你喜欢的开发工具就足够了,能支持普通 Java 项目开发便可。
为了方便,我作了一个最基本的Java 项目 —— ServletDemo.zip ,你可将它导入到 Eclipse 里就是一个完整的、最简单的 Web 项目。
而后将下面 XML 内容替换 Tomcat 下的 conf/server.xml 文件:
1 <?xml version='1.0' encoding='utf-8'?> 2 <Server port="8005" shutdown="SHUTDOWN"> 3 <Service name="Catalina"> 4 <Connector port="8080" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443" URIEncoding="UTF-8"/> 5 <Engine name="Catalina" defaultHost="localhost"> 6 <Host name="localhost"> 7 <Context path="" docBase="D:\WORKDIR\ServletDemo\webapp" reloadable="true"/> 8 </Host> 9 </Engine> 10 </Service> 11 </Server>
其中 D:\WORKDIR\ServletDemo 替换为你导入的项目路径,再次启动 Tomcat 后在浏览器打开 http://localhost:8080/hello 即可看到 Hello World 的输出信息。
3. 了解 Servlet 和 Filter
好了,我已经把环境搭起来了,接下来该干吗呢?
前面的步骤为的是搭建一个测试的环境,而后让你了解一个最基本的 Java Web 项目的结构。
一个最基本的 Java Web 项目所需的 jar 包只须要一个 servlet-api.jar ,这个 jar 包中的类大部分都是接口,还有一些工具类,共有 2 个包,分别是 javax.servlet 和 javax.servlet.http。我把这个jar包放到了 webapp 目录外的一个独立 packages 文件夹里,这是由于全部的 Servlet 容器都带有这个包,你无需再放到Web项目里,咱们放到这里只不过是编译的须要,运行是不须要的。若是你硬是把 servlet-api.jar 放到 webapp/WEB-INF/lib 目录下,那么 Tomcat 启动时还会报一个警告信息。
Java Web 项目还须要一个很是重要的配置文件 web.xml ,在这个项目中已经被我最小化了,只保留有用的信息:
1 <?xml version="1.0" encoding="UTF-8"?> 2 <!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" 3 "http://java.sun.com/dtd/web-app_2_3.dtd"> 4 <web-app> 5 6 <servlet> 7 <servlet-name>hello_world</servlet-name> 8 <servlet-class>demo.HelloServlet</servlet-class> 9 <load-on-startup>1</load-on-startup> 10 </servlet> 11 12 <servlet-mapping> 13 <servlet-name>hello_world</servlet-name> 14 <url-pattern>/hello</url-pattern> 15 </servlet-mapping> 16 17 </web-app>
每一个 servlet 都必须在 web.xml 中定义并进行 URL 映射配置,早期 Java 开发 Web 在没有框架满天飞的时候,这个文件会定义了大量的 servlet,或者有人为了省事干脆来一个 /servlet/* 来经过类名直接调用 Servlet。
Servlet 规范里还有另一个很是重要并且很是有用的接口那就是 Filter 过滤器。
下面是一个最简单的 Filter 类以及相应的定义方法:
1 package demo; 2 3 import java.io.IOException; 4 5 import javax.servlet.Filter; 6 import javax.servlet.FilterChain; 7 import javax.servlet.FilterConfig; 8 import javax.servlet.ServletException; 9 import javax.servlet.ServletRequest; 10 import javax.servlet.ServletResponse; 11 import javax.servlet.http.HttpServletRequest; 12 13 public class HelloFilter implements Filter { 14 15 @Override 16 public void init(FilterConfig arg0) throws ServletException { 17 System.out.println("Filter 初始化"); 18 } 19 20 @Override 21 public void doFilter(ServletRequest req, ServletResponse res, 22 FilterChain chain) throws IOException, ServletException { 23 HttpServletRequest request = (HttpServletRequest)req; 24 System.out.println("拦截 URI="+request.getRequestURI()); 25 chain.doFilter(req, res); 26 } 27 28 @Override 29 public void destroy() { 30 System.out.println("Filter 结束"); 31 } 32 }
1 <?xml version="1.0" encoding="UTF-8"?> 2 <!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" 3 "http://java.sun.com/dtd/web-app_2_3.dtd"> 4 <web-app> 5 6 <filter> 7 <filter-name>helloFilter</filter-name> 8 <filter-class>demo.HelloFilter</filter-class> 9 </filter> 10 11 <filter-mapping> 12 <filter-name>helloFilter</filter-name> 13 <url-pattern>/*</url-pattern> 14 </filter-mapping> 15 16 <servlet> 17 <servlet-name>hello_world</servlet-name> 18 <servlet-class>demo.HelloServlet</servlet-class> 19 <load-on-startup>1</load-on-startup> 20 </servlet> 21 22 <servlet-mapping> 23 <servlet-name>hello_world</servlet-name> 24 <url-pattern>/hello</url-pattern> 25 </servlet-mapping> 26 27 </web-app>
访问 http://localhost:8080/hello 时看看 Tomcat 控制台有何输出信息。
4. Servlet 和 HTTP 的对应关系
Servlet 是 J2EE 最重要的一部分,有了 Servlet 你就是 J2EE 了,J2EE 的其余方面的内容择需采用。而 Servlet 规范你须要掌握的就是 servlet 和 filter 这两项技术。绝大多数框架不是基于 servlet 就是基于 filter,若是它要在 Servlet 容器上运行,就永远也脱离不开这个模型。
为何 Servlet 规范会有两个包,javax.servlet 和 javax.servlet.http ,早先设计该规范的人认为 Servlet 是一种服务模型,不必定是依赖某种网络协议之上,所以就抽象出了一个 javax.servlet ,同时在提供一个基于 HTTP 协议上的接口扩展。可是从实际运行这么多年来看,彷佛没有发现有在其余协议上实现的 Servlet 技术。
javax.servlet 和 javax.servlet.http 这两个包总共加起来也不过是三十四个接口和类。你须要经过 J2EE 的 JavaDoc 文档 熟知每一个类和接口的具体意思。特别是下面几个接口必须熟知每一个方法的意思和用途:
再次强调 HttpServletRequest 和 HttpServletResponse 这两个接口更应该是烂熟于心。
若是你从字面上没法理解某个方法的意思,你能够在前面那个项目的基础上作实验看看其输出,再不行你能够到讨论区提问,这样的提问很是明确,不少人均可以帮到你。
为何我这么强调 HttpServletRequest 和 HttpServletResponse 这两个接口,由于 Web 开发是离不开 HTTP 协议的,而 Servlet 规范其实就是对 HTTP 协议作面向对象的封装,HTTP协议中的请求和响应就是对应了 HttpServletRequest 和 HttpServletResponse 这两个接口。
你能够经过 HttpServletRequest 来获取全部请求相关的信息,包括 URI、Cookie、Header、请求参数等等,别无它路。所以当你使用某个框架时,你想获取HTTP请求的相关信息,只要拿到 HttpServletRequest 实例便可。
而 HttpServletResponse接口是用来生产 HTTP 回应,包含 Cookie、Header 以及回应的内容等等。
5. 再谈谈 Session
HTTP 协议里是没有关于 Session 会话的定义,Session 是各类编程语言根据 HTTP 协议的无状态这种特色而产生的。其实现无非就是服务器端的一个哈希表,哈希表的Key就是传递给浏览器的名为 jsessionid 的 Cookie 值。
当须要将某个值保存到 session 时,容器会执行以下几步:
a. 获取 jsessionid 值,没有的话就生成一个,也就是 request.getSession() 这个方法
b. 拿到的 HttpSession 对象实例就至关于一个哈希表,你能够往哈希表里存放数据(setAttribute)
c. 你也能够经过 getAttribute 来获取某个值
而这个名为 jsessionid 的 Cookie 在浏览器关闭时会自动删除。把 Cookie 的 MaxAge 值设为 -1 就能达到浏览器关闭自动删除的效果。
6. 关于 JSP
首先我已经不用 JSP 不少年了,如今一直是使用 Velocity 模板引擎。
任何一个 JSP 页面在执行的时候都会编译成一个 Servlet 类文件,若是是 Tomcat 的话,这些生成的 java 文件会放置在 {TOMCAT}/work 目录下对应项目的子目录中,例如 Tomcat 生成的类文件以下:
1 package org.apache.jsp; 2 3 import javax.servlet.*; 4 import javax.servlet.http.*; 5 import javax.servlet.jsp.*; 6 import java.util.*; 7 8 public final class test_jsp extends org.apache.jasper.runtime.HttpJspBase 9 implements org.apache.jasper.runtime.JspSourceDependent { 10 11 private static final JspFactory _jspxFactory = JspFactory.getDefaultFactory(); 12 13 private static java.util.List<String> _jspx_dependants; 14 15 private javax.el.ExpressionFactory _el_expressionfactory; 16 private org.apache.tomcat.InstanceManager _jsp_instancemanager; 17 18 public java.util.List<String> getDependants() { 19 return _jspx_dependants; 20 } 21 22 public void _jspInit() { 23 _el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory(); 24 _jsp_instancemanager = org.apache.jasper.runtime.InstanceManagerFactory.getInstanceManager(getServletConfig()); 25 } 26 27 public void _jspDestroy() { 28 } 29 30 public void _jspService(final HttpServletRequest request, final HttpServletResponse response) 31 throws java.io.IOException, ServletException { 32 33 final PageContext pageContext; 34 HttpSession session = null; 35 final ServletContext application; 36 final ServletConfig config; 37 JspWriter out = null; 38 final Object page = this; 39 JspWriter _jspx_out = null; 40 PageContext _jspx_page_context = null; 41 42 43 try { 44 response.setContentType("text/html;charset=utf-8"); 45 pageContext = _jspxFactory.getPageContext(this, request, response, 46 null, true, 8192, true); 47 _jspx_page_context = pageContext; 48 application = pageContext.getServletContext(); 49 config = pageContext.getServletConfig(); 50 session = pageContext.getSession(); 51 out = pageContext.getOut(); 52 _jspx_out = out; 53 54 out.write("\r\n"); 55 out.write("<html>\r\n"); 56 out.write(" <title>Test</title>\r\n"); 57 out.write(" <style>\r\n"); 58 out.write(" </style> \r\n"); 59 out.write(" <body>\r\n"); 60 out.write("<h1>Test Demo (oschina)</h1>\r\n"); 61 out.write("<table cellspacing=\"1\" cellpadding=\"5\">\r\n"); 62 63 Enumeration Names=request.getHeaderNames(); 64 while(Names.hasMoreElements()) 65 {String name=(String)Names.nextElement(); 66 String value=request.getHeader(name); 67 68 out.write("\r\n"); 69 out.write(" <tr>\r\n"); 70 out.write(" <td>"); 71 out.print(name); 72 out.write("</td>\r\n"); 73 out.write(" <td>"); 74 out.print(value); 75 out.write("</td>\r\n"); 76 out.write(" \r\n"); 77 out.write(" </tr>\r\n"); 78 out.write(" "); 79 80 } 81 82 out.write("\r\n"); 83 out.write("</table>\r\n"); 84 out.write(" </body>\r\n"); 85 out.write("</html>"); 86 } catch (Throwable t) { 87 if (!(t instanceof SkipPageException)){ 88 out = _jspx_out; 89 if (out != null && out.getBufferSize() != 0) 90 try { out.clearBuffer(); } catch (java.io.IOException e) {} 91 if (_jspx_page_context != null) _jspx_page_context.handlePageException(t); 92 } 93 } finally { 94 _jspxFactory.releasePageContext(_jspx_page_context); 95 } 96 } 97 }
在 servlet 中有一个包 javax.servlet.jsp 是跟 JSP 相关的一些接口规范定义。JSP 比 Servlet 方便的地方在于可直接修改当即生效,不像 Servlet 修改后必须重启容器才能生效。
所以 JSP 适合用来作视图,而 Servlet 则适合作控制层。
7. 总结
罗哩罗嗦一大堆,概括一下就是下面几点:
等你真的掌握了 Servlet 规范再去看框架,便会以为一些都小菜。总之一点:不要被框架牵着鼻子走,框架是你的工具,它应该听你的!
红薯乱弹,随时准备挨喷。
原文地址连接:http://www.oschina.net/question/12_52027#tags_nav