设为首页 收藏本站
查看: 441|回复: 0

[经验分享] 自己动手写Tomcat

[复制链接]

尚未签到

发表于 2017-1-21 11:10:03 | 显示全部楼层 |阅读模式
  最近研究一方socket编程,由于想动手写关于socket方面的东西。然而我们知道通过URL去访问某网址,其实其底层用的就是socket,于是我就写了一个很简单的tomcat服务器,主要目地在于学习,在此分享给大家。同时提供下载源工程。
  我写的工程用Maven管理的,但是我没有引入其它的JAR包,为此我就不列出pom.xml文件了。
  在此简要地说明每个类的作用:
  Server.java
  该类的作用就是将服务提起来的,并且利用线程池。

package com.cloud.tomcat.server;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class Server {
private static ServerSocket serverSocket;
private static ExecutorService executorService;
private final static int POOL_SIZE = 15;
public static void main(String[] args) throws Exception {
serverSocket = new ServerSocket(8080);
Socket socket = null;
executorService = Executors.newFixedThreadPool(POOL_SIZE);
while (true) {
socket = serverSocket.accept();
PrintWriter writer = new PrintWriter(new OutputStreamWriter(socket.getOutputStream(), "UTF-8"));
writer.println("HTTP/1.1 200 OK");
writer.println("Content-Type: text/html;charset=UTF-8");
writer.println();
executorService.execute(new Handler(socket, writer));
}
}
}
  Handler.java
  该类的作用是根据浏览器传过来信息做出相应的处理,同时实现Runnable接口。

package com.cloud.tomcat.server;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import com.cloud.tomcat.servlet.HttpServlet;
public class Handler implements Runnable {
private Socket socket;
private PrintWriter writer;
public Handler(Socket socket, PrintWriter writer) {
this.socket = socket;
this.writer = writer;
}
@Override
public void run() {
try {
InputStream inputStream = socket.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
String path = "";
String method = "";
while (true) {
String msg = reader.readLine();
if (null == msg || "".equals(msg.trim())) {
break;
}
String[] msgs = msg.split(" ");
if (3 == msgs.length && "HTTP/1.1".equalsIgnoreCase(msgs[2])) {
method = msgs[0];
path = msgs[1];
break;
}
}
if (path.endsWith("ico")) {
return;
}
HttpServlet httpServlet = ServletContainer.getHttpServlet(path);
String html = "";
if ("GET".equals(method)) {
html = httpServlet.doGet();
} else if ("POST".equals(method)) {
html = httpServlet.doGet();
}
writer.write(html);
writer.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
writer.close();
socket.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
  ServletContainer.java
  该类首先会解析web.xml文件,然后根据url的信息,拿到相应的servlet。

package com.cloud.tomcat.server;
import java.util.HashMap;
import java.util.Map;
import com.cloud.tomcat.model.Servlet;
import com.cloud.tomcat.model.ServletMapping;
import com.cloud.tomcat.servlet.HttpServlet;
import com.cloud.tomcat.util.XMLUtil;
public class ServletContainer {
private static Map<String, Object> servletMaps = new HashMap<String, Object>();
private static Map<String, Object> servletMappingMaps = new HashMap<String, Object>();
private static Map<String, HttpServlet> servletContainer = new HashMap<String, HttpServlet>();
static {
try {
Map<Integer, Map<String, Object>> maps = XMLUtil.parseWebXML();
if (null != maps && 2 == maps.size()) {
servletMaps = maps.get(0);
servletMappingMaps = maps.get(1);
}
} catch (Exception e) {
e.printStackTrace();
}
}
public static HttpServlet getHttpServlet(String path) {
if (null == path || "".equals(path.trim()) || "/".equals(path)) {
path = "/index";
}
if (servletContainer.containsKey(path)) {
return servletContainer.get(path);
}
if (!servletMappingMaps.containsKey(path)) {
return null;
}
ServletMapping servletMapping = (ServletMapping) servletMappingMaps.get(path);
String name = servletMapping.getName();
if (!servletMaps.containsKey(name)) {
return null;
}
Servlet servlet = (Servlet) servletMaps.get(name);
String clazz = servlet.getClazz();
if (null == clazz || "".equals(clazz.trim())) {
return null;
}
HttpServlet httpServlet = null;
try {
httpServlet = (HttpServlet) Class.forName(clazz).newInstance();
servletContainer.put(path, httpServlet);
} catch (Exception e) {
e.printStackTrace();
}
return httpServlet;
}
}
  HttpServlet.java
  为了实现起来简单方便,我自己定义了一个HttpServlet。

package com.cloud.tomcat.servlet;
public interface HttpServlet {
public String doGet();
public String doPost();
}
  CloudServlet.java
  HttpServlet的具体实现类。

package com.cloud.tomcat.servlet;
public class CloudServlet implements HttpServlet {
@Override
public String doGet() {
return this.doPost();
}
@Override
public String doPost() {
return "<h1>Chicago at Cloud!!!</h1>";
}
}
  下面一一列出解析web.xml用到的类,由于我没有引入第三JAR包,可能这部分有点麻烦。
  Servlet.java

package com.cloud.tomcat.model;
public class Servlet {
private String name;
private String clazz;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getClazz() {
return clazz;
}
public void setClazz(String clazz) {
this.clazz = clazz;
}
}
  ServletMapping.java

package com.cloud.tomcat.model;
public class ServletMapping {
private String name;
private String url;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getUrl() {
return url;
}
public void setUrl(String url) {
this.url = url;
}
}
  XMLUtil.java

package com.cloud.tomcat.util;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import com.cloud.tomcat.model.Servlet;
import com.cloud.tomcat.model.ServletMapping;
public class XMLUtil {
public static Map<Integer, Map<String, Object>> parseWebXML() throws Exception {
Map<Integer, Map<String, Object>> result = new HashMap<Integer, Map<String,Object>>();
DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
DocumentBuilder db = dbf.newDocumentBuilder();
InputStream in = XMLUtil.class.getClassLoader().getResourceAsStream("web.xml");
Document document = db.parse(in);
Element root = document.getDocumentElement();
NodeList xmlNodes = root.getChildNodes();
for (int i = 0; i < xmlNodes.getLength(); i++) {
Node config = xmlNodes.item(i);
if (null != config && config.getNodeType() == Node.ELEMENT_NODE) {
String nodeName1 = config.getNodeName();
if ("servlet".equals(nodeName1)) {
Map<String, Object> servletMaps = null;
if (result.containsKey(0)) {
servletMaps = result.get(0);
} else {
servletMaps = new HashMap<String, Object>();
}
NodeList childNodes = config.getChildNodes();
Servlet servlet = new Servlet();
for (int j = 0; j < childNodes.getLength(); j++) {
Node node = childNodes.item(j);
if (null != node && node.getNodeType() == Node.ELEMENT_NODE) {
String nodeName2 = node.getNodeName();
String textContent = node.getTextContent();
if ("servlet-name".equals(nodeName2)) {
servlet.setName(textContent);
} else if ("servlet-class".equals(nodeName2)) {
servlet.setClazz(textContent);
}
}
}
servletMaps.put(servlet.getName(), servlet);
result.put(0, servletMaps);
} else if ("servlet-mapping".equals(nodeName1)) {
Map<String, Object> servletMappingMaps = null;
if (result.containsKey(1)) {
servletMappingMaps = result.get(1);
} else {
servletMappingMaps = new HashMap<String, Object>();
}
NodeList childNodes = config.getChildNodes();
ServletMapping servletMapping = new ServletMapping();
for (int j = 0; j < childNodes.getLength(); j++) {
Node node = childNodes.item(j);
if (null != node && node.getNodeType() == Node.ELEMENT_NODE) {
String nodeName2 = node.getNodeName();
String textContent = node.getTextContent();
if ("servlet-name".equals(nodeName2)) {
servletMapping.setName(textContent);
} else if ("url-pattern".equals(nodeName2)) {
servletMapping.setUrl(textContent);
}
}
}
servletMappingMaps.put(servletMapping.getUrl(), servletMapping);
result.put(1, servletMappingMaps);
}
}
}
return result;
}
public static void main(String[] args) throws Exception {
System.out.println(parseWebXML());
}
}
  web.xml

<?xml version="1.0" encoding="UTF-8"?>
<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_2_5.xsd"
version="2.5">
<servlet>
<servlet-name>cloud</servlet-name>
<servlet-class>com.cloud.tomcat.servlet.CloudServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>cloud</servlet-name>
<url-pattern>/index</url-pattern>
</servlet-mapping>
</web-app>

  运行结果:
  将Server类运行起来,然后用浏览器输入:
  http://localhost:8080/index或http://localhost:8080
  得到如下结果:
DSC0000.png
 

  
DSC0001.png
 

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-331533-1-1.html 上篇帖子: 增大tomcat内存2 下篇帖子: Tomcat参数及性能优化
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表