初学Java Web开发应该如何学习

web框架是开发者在使用某种语言编写web应用服务端项目时关于架构的最佳实践。很多web框架是从实际的web项目抽取出来的,仅和web的请求和响应处理有关,形成一个基础,在开发别的应用项目的时候则可以从这个剥离出来的基础做起,让开发者更关注更具体的业务问题,而不是web的请求和响应的控制。
框架很多,但套路基本类似,帮你隐藏很多关于 http 协议细节内容,专注功能开发。
但对一个初学者来说,过早的接触框架往往是事倍功半!同样一个问题,换一种框架你可能需要从头开始研究。
下面是针对初学 java 开发 web 过程一些个人见解和思路,高手可略过。
1. 基本要求:java 编程基础
有良好的 java 语言编程基础,这是必须的,在讨论 web 开发技术时提了一个 java 编程基础的问题会被鄙视的。
2. 环境准备 (eclipse + tomcat)
选择一个你喜爱的servlet容器,或者说大一点就是应用服务器,推荐 tomcat 、resin 或者 jetty 这些轻量级的产品。这三个产品下载 zip 包解压后就可以用了。如果你不熟悉 tomcat 的话请不要使用 exe 版本的 tomcat,那会徒增很多烦恼。
把应用服务器启动起来并能访问到其默认的页面为准。
关于开发工具:不推荐使用 myeclipse 和 eclipse 的 jee 版本,徒增烦恼、运行缓慢而且还让你无法了解 web 项目的结构。普通的 eclipse 或者你喜欢的开发工具就足够了,能支持普通 java 项目开发即可。
为了方便,我做了一个最基本的java 项目 ——servletdemo.zip(评论区见链接),你可将它导入到 eclipse 里就是一个完整的、最简单的 web 项目。
然后将下面 xml 内容替换 tomcat 下的 conf/server.xml 文件:
其中 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 ,在这个项目中已经被我最小化了,只保留有用的信息:
hello_worlddemo.helloservlet1hello_world/hello
每个 servlet 都必须在 web.xml 中定义并进行 url 映射配置,早期 java 开发 web 在没有框架满天飞的时候,这个文件会定义了大量的 servlet,或者有人为了省事干脆来一个 /servlet/* 来通过类名直接调用 servlet。
servlet 规范里还有另外一个非常重要而且非常有用的接口那就是 filter 过滤器。
下面是一个最简单的 filter 类以及相应的定义方法:
package demo;import java.io.ioexception;import javax.servlet.filter;import javax.servlet.filterchain;import javax.servlet.filterconfig;import javax.servlet.servletexception;import javax.servlet.servletrequest;import javax.servlet.servletresponse;import javax.servlet.http.httpservletrequest;public class hellofilter implements filter {@overridepublic void init(filterconfig arg0) throws servletexception {system.out.println(filter 初始化);}@overridepublic void dofilter(servletrequest req, servletresponse res,filterchain chain) throws ioexception, servletexception {httpservletrequest request = (httpservletrequest)req;system.out.println(拦截 uri=+request.getrequesturi());chain.dofilter(req, res);}@overridepublic void destroy() {system.out.println(filter 结束);}}
在 web.xml 中的配置必须放在 servlet 的前面:
访问 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 文档 熟知每个类和接口的具体意思。特别是下面几个接口必须熟知每个方法的意思和用途:
httpservlet
servetconfig
servletcontext
filter
filterconfig
filterchain
requestdispatcher
httpservletrequest
httpservletresponse
httpsession
一些 listenser 类
再次强调 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 生成的类文件如下:
package org.apache.jsp;import javax.servlet.*;import javax.servlet.http.*;import javax.servlet.jsp.*;import java.util.*;public final class test_jsp extends org.apache.jasper.runtime.httpjspbase implements org.apache.jasper.runtime.jspsourcedependent { private static final jspfactory _jspxfactory = jspfactory.getdefaultfactory(); private static java.util.list _jspx_dependants; private javax.el.expressionfactory _el_expressionfactory; private org.apache.tomcat.instancemanager _jsp_instancemanager; public java.util.list getdependants() { return _jspx_dependants; } public void _jspinit() { _el_expressionfactory = _jspxfactory.getjspapplicationcontext(getservletconfig().getservletcontext()).getexpressionfactory(); _jsp_instancemanager = org.apache.jasper.runtime.instancemanagerfactory.getinstancemanager(getservletconfig()); } public void _jspdestroy() { } public void _jspservice(final httpservletrequest request, final httpservletresponse response) throws java.io.ioexception, servletexception { final pagecontext pagecontext; httpsession session = null; final servletcontext application; final servletconfig config; jspwriter out = null; final object page = this; jspwriter _jspx_out = null; pagecontext _jspx_page_context = null; try { response.setcontenttype(text/html;charset=utf-8); pagecontext = _jspxfactory.getpagecontext(this, request, response, null, true, 8192, true); _jspx_page_context = pagecontext; application = pagecontext.getservletcontext(); config = pagecontext.getservletconfig(); session = pagecontext.getsession(); out = pagecontext.getout(); _jspx_out = out; out.write(\r\n); out.write(\r\n); out.write( test\r\n); out.write( \r\n); out.write( \r\n); out.write(test demo (oschina)\r\n); out.write(\r\n);enumeration names=request.getheadernames();while(names.hasmoreelements()){string name=(string)names.nextelement();string value=request.getheader(name); out.write(\r\n); out.write( \r\n); out.write( ); out.print(name); out.write( \r\n); out.write( ); out.print(value); out.write( \r\n); out.write( \r\n); out.write(
\r\n); out.write( ); } out.write(\r\n); out.write(
\r\n); out.write( \r\n); out.write(); } catch (throwable t) { if (!(t instanceof skippageexception)){ out = _jspx_out; if (out != null && out.getbuffersize() != 0) try { out.clearbuffer(); } catch (java.io.ioexception e) {} if (_jspx_page_context != null) _jspx_page_context.handlepageexception(t); } } finally { _jspxfactory.releasepagecontext(_jspx_page_context); } }}
在 servlet 中有一个包 javax.servlet.jsp 是跟 jsp 相关的一些接口规范定义。jsp 比 servlet 方便的地方在于可直接修改立即生效,不像 servlet 修改后必须重启容器才能生效。
因此 jsp 适合用来做视图,而 servlet 则适合做控制层。
7. 总结
罗哩罗嗦一大堆,归纳一下就是下面几点:
熟知 servlet 规范之前,请不要学习任何框架
使用最简单的工具,不要任何向导和可视化
熟知 http 协议
等你真的掌握了 servlet 规范再去看框架,便会觉得一些都小菜。总之一点:不要被框架牵着鼻子走,框架是你的工具,它应该听你的!

vivo官方宣布为旗下X系列手机X9推出新配色磨砂黑
Barefoot:可编程交换技术将在5G网络发展中发挥作用
德力西电气连续多年荣获CAIMRS奖项
微软:不会发布Surface Duo双屏手机12月补丁
Google Play显示过去两年内共有1000万以上安卓用户改用iPhone
初学Java Web开发应该如何学习
斗轮机主轴紧定套位磨损的修复方法
原电池供电远程患者监护仪的电源设计注意事项
关于英特尔数字标牌匿名视频分析技术的分析和应用
浅谈移动DR的应用及优势
了解DRAM存储芯片是怎么玩的
丰田固态电池今年发布:十分钟空电充至满电 续航500公里
中国移动加大网络设备端投资力度,2020年5G资本预算1000亿
MAX3803 直流耦合、UCSP封装、3.125Gbps均
关于视频放大器的一些知识点
计算机科学和互联网技术发展的产物:云计算革命
西门子博途:寻址全局变量/访问I/O设备
10万新日产劲客今上市,配置外观对得起这个价?剑指国产SUV
SMA连接器与BNC连接器用途有什么不同
浅谈行星减速器,它的精度该如何调整