引入课程和Maven
1.Maven
maven中央仓库:Maven Repository: Search/Browse/Explore (mvnrepository.com)
maven仓库是国外的一个网站,由于网络问题,我们也常使用maven仓库的镜像
maven的原理和java程序操作数据库,然后数据库返回相关数据类似。
1.1创建maven-web项目
-
先创建一个Maven的Web项目
选择file-new-project
-
在弹出的窗口中选择Maven,点击选择右边的create from archetype
-
在下面选择maven-archetype-webapp,然后点击next
-
输入相关数据,点击next
-
指定项目使用的maven程序(第一次使用可以先不指定2和3,稍后再指定),点击finish
-
默认创建的项目结构如下:
-
点击项目左上角的file-setting,在弹出的窗口中点击build,execution,deployment,再选择maven
-
在右边的maven home path框中,选择默认的或者你自己下载的maven程序(我这里选择默认)
-
在资源管理器中打开该上面的目录,将conf下的setting.xml文件复制到c盘用户目录的m2.目录下
-
打开复制后的setting.xml文件,找到mirror标签,注释掉原来的mirror镜像
添加新镜像如下:
<mirror> <id>alimaven</id> <name>aliyun</name> <url>https://maven.aliyun.com/nexus/content/groups/public</url> <mirrorOf>central</mirrorOf> </mirror>
-
返回idea的窗口中,选择刚才配置的xml文件和要存储下载的jar包的目录,点击override,点击ok
这样就完成了maven的镜像配置
-
在项目的pom.xml文件中添加配置
在dependency元素中添加配置如下:
<!--引入Servlet.jar--> <!--1.为了开发servlet,需要引入其jar包 2.dependency 表示引入一个包 3.groupId 表示包的开发公司/组织/个人 的信息 javax.servlet 4.artifactId 表示包的项目名称 javax.servlet-api 补充:groupId+artifactId是以目录形式体现 5.version 表示该包当前的版本 6.scope 表示引入的包的作用范围 7.provided 表示 这里你引入的jar包,在编译,测试有效,但是在打包发布的时候,不要带上这个jar包 (因为Tomcat本身有jar包) 8.下载的包在你指定的目录中 9.可以去修改我们要下载的目录位置 10.可以去指定maven仓库,即配置maven镜像 --> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>3.1.0</version> <scope>provided</scope> </dependency>
-
点击右边的Maven,点击循环按钮,开始下载依赖
-
下载完毕后,可以在配置的下载目录中看到该依赖:
下一次如果要使用依赖,就会先到下载目录下去找该依赖。
2.测试小案例
编写如下的测试案例
创建java目录,存放源文件
创建Servlet(如果选项中没有Servlet,点击右边的maven,重新导一下包)
在web.xml文件中配置Servlet(使用注解也可以)
<servlet> <servlet-name>CalServlet</servlet-name> <servlet-class>com.li.servlet.CalServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>CalServlet</servlet-name> <url-pattern>/calServlet</url-pattern> </servlet-mapping>
编写程序
WebUtils:
package com.li.utils; public class WebUtils { /** * 将一个字符串数字转成int,如果转换失败,就返回传入的defaultVal * * @param strNum * @param defaultVal * @return */ public static int parseInt(String strNum, int defaultVal) { try { return Integer.parseInt(strNum); } catch (NumberFormatException e) { System.out.println(strNum + "输入的格式错误"); } return defaultVal; } }
CalServlet:
package com.li.servlet; import com.li.utils.WebUtils; import javax.servlet.*; import javax.servlet.http.*; import java.io.IOException; import java.io.PrintWriter; public class CalServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //接收提交的数据进行计算 String strNum1 = request.getParameter("num1"); String strNum2 = request.getParameter("num2"); //把strNum1和strNum2转成int类型 int num1 = WebUtils.parseInt(strNum1, 0); int num2 = WebUtils.parseInt(strNum2, 0); int result = num1 + num2; response.setContentType("text/html;charset=utf-8"); PrintWriter writer = response.getWriter(); writer.print("<h1>" + num1 + "+" + num2 + "=" + result + "</h1>"); writer.flush(); writer.close(); } }
cal.html:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>计算器</title> </head> <body> <h1>计算器</h1> <!--这里的路径是浏览器要解析的,因此如果写成/calServlet的话,就会解析成http://主机名/calServlet 想要这样配置的话,tomcat的路径就不要写项目名了--> <form action="/calServlet" method="post"> num1:<input type="text" name="num1"/><br/> num2:<input type="text" name="num2"/><br/> <input type="submit" value="提交"> </form> </body> </html>
配置Tomcat(注意application context的配置)
点击运行tomcat,在浏览器地址栏中输入:http://localhost:8080/cal.html
,显示页面如下:
输入数据,点击提交:
2.Tomcat整体项目架构分析
我们创建web项目,都要依赖于Tomcat等web服务以及实现继承Servlet接口。
可以看出对Tomcat的底层机制和调用到Servlet的流程的理解是十分重要的。
现在我们来模拟Tomcat底层实现,并能调用我们自己设计的Servlet,完成相同的功能。
说明:Tomcat有三种运行模式(BIO,NIO,APR),因为核心讲解的是Tomcat如何接受客户端请求,解析请求,调用Servlet,并返回结果的流程机制,因此采用BIO线程模型来模拟。
整个流程如下:
- 浏览器在地址栏输入请求的URL,发送http请求
- http请求到达Tomcat后,底层的socket网络编程会接收这个请求,创建一个线程(每一个请求对应一个线程)
- 被创建的线程就会去解析这个URL是什么请求,如果是请求Servlet资源,就去调用Servlet(底层会用反射实例化);如果调用的是静态资源,就直接返回