手写Tomcat服务器(二)

本文是手写Tomcat服务器系列的第二篇,重点介绍了如何实现web.xml的读取,包括使用SAX解析文件,提取servlet和servlet-mapping信息,并将这些信息映射到具体类。此外,还讲解了Servlet接口的设计,URL到Servlet的映射,以及服务器如何转发请求。最后,针对服务器响应请求后程序停止的问题,提出了通过多线程解决,以实现处理多个客户端请求的能力。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

上一篇文章我们实现了Http请求与响应,封装了Request与Response对象。这篇文章我们来实现Tomcat服务器的核心功能,web.xml读取,映射,页面跳转等功能。
web.xml文件定义如下:
在这里插入图片描述
一、读取web.xml文件
读取xml文件的方式有很多,有DOM,SAX,Stax等。这里采用的是SAX读取方式。
在这里插入图片描述
二、提取标签信息
将servlet标签和servlet-mapping标签以及子标签提取出来并存储到Mapping和Entity中
在这里插入图片描述
Entity源码:

package org.wrf.httpServer.server.core;
/**
 *  <servlet>
	<servlet-name>login</servlet-name>
	<servlet-class>com.shsxt.server.LoginServlet</servlet-class>
	</servlet>
 * Copyright © 2019 WRF. All rights reserved.
 * 功能描述:
 * @Package: org.wrf.httpServer.servlet 
 * @author: knight   
 * @date: 2019年3月26日 下午7:36:58
 */
public class Entity {
	private String name;
	private String clz;
	
	public Entity() {
	}
	
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getClz() {
		return clz;
	}
	public void setClz(String clz) {
		this.clz = clz;
	}

	@Override
	public String toString() {
		return "Entity [name=" + name + ", clz=" + clz + "]";
	}
	

}

Mapping源码:

package org.wrf.httpServer.server.core;

import java.util.HashSet;
import java.util.Set;

/**
 *<servlet-mapping>
  	<servlet-name>others</servlet-name>
  	<url-pattern>/o</url-pattern> 
 </servlet-mapping>
 * Copyright © 2019 WRF. All rights reserved.
 * 功能描述:
 * @Package: org.wrf.httpServer.servlet 
 * @author: knight   
 * @date: 2019年3月26日 下午7:37:42
 */
public class Mapping {
	private String name;
	private Set<String>patterns;
	
	public Mapping() {
		patterns=new HashSet<String>();
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public Set<String> getPatterns() {
		return patterns;
	}

	public void setPatterns(Set<String> patterns) {
		this.patterns = patterns;
	}
	
	public void addPattern(String pattern) {
		this.patterns.add(pattern);
	}
	
}

三、提取信息转化为Map并提供反射
转化为Map
在这里插入图片描述
提供反射:
在这里插入图片描述
四、定义Servlet接口以及其实现类
Servlet接口
在这里插入图片描述
LoginServlet
在这里插入图片描述
RegisterServlet
在这里插入图片描述

五、定义url至Servlet接口或其实现类的映射

在这里插入图片描述
六、重构Server,实现Url请求转发至指定类功能
在这里插入图片描述
至此,Tomcat基本框架已经搭建完毕。运行功能正常,但是笔者测试的时候发现一个小bug,那就是服务器响应浏览器请求后程序就会停止。这个并不符合我们日常的需求。该如何解决呢?答案是多线程。通过配置一个分发器,实现一台服务器可以响应多个客户端请求。

package org.wrf.httpServer.server.core;

import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;
/**
 * 实现404,500,主页面
 * Copyright © 2019 WRF. All rights reserved.
 * 功能描述:
 * @Package: org.wrf.httpServer.server 
 * @author: knight   
 * @date: 2019年3月27日 下午10:47:51
 */
public class Dispatcher implements Runnable {
	private Socket client;
	private Request request;
	private Response response;

	public Dispatcher(Socket client) {
		this.client = client;
		try {
			// 获取协议内容
			request = new Request(client);
			System.out.println(request.getParamtersMap());
			// 返回Response
			response = new Response(client);
		} catch (IOException e) {
			e.printStackTrace();
			release();
		}
	}

	@Override
	public void run() {
		// 关注状态码
		Servlet servlet = WebApp.getServletFromUrl(request.getUrl());
		try {
			if(request.getUrl()==null||request.getUrl().equals("")) {
				 InputStream is=Thread.currentThread().getContextClassLoader().getResourceAsStream("org/wrf/httpServer/server/index.html");
				 byte []buff=new byte[1024*1024];
				 int len=is.read(buff);
				 response.print(new String(buff,0,len));
				 response.pushToBrowser(200);
			}
			
			if (servlet != null) {
				servlet.service(request, response);
				response.pushToBrowser(200);
			} else {
				  InputStream is=Thread.currentThread().getContextClassLoader().getResourceAsStream("org/wrf/httpServer/server/error.html");
				  byte []buff=new byte[1024*1024];
				  int len=is.read(buff);
				  response.print(new String(buff,0,len));
				  response.pushToBrowser(404);
			}
		} catch (IOException e) {
			e.printStackTrace();
			try {
				response.println("服务器崩溃了。。。。。。");
				response.pushToBrowser(500);
			} catch (IOException e1) {
				e1.printStackTrace();
			}
			release();
		}
		release();
	}

	private void release() {
		try {
			client.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
}

最后,完善Server代码。

package org.wrf.httpServer.server.core;

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * HttpServer服务器
 * Copyright © 2019 WRF. All rights reserved.
 * 功能描述:
 * 1.通过ServerSocket与浏览器建立连接,
 * 2.获取协议内容 返回内容
 * 3.封装response实现动态添加内容,状态码
 * 4.封装Request
 * 5.解析参数
 * 6.引入Servlet
 * 7.整合web.xml
 * 8.加入多线程实现分发器
 * 9.实现404,500,主页等页面
 * @Package: org.wrf.httpServer.server 
 * @author: knight   
 * @date: 2019年3月26日 下午9:58:32
 */
public class Server {
	private ServerSocket serverSocket;
	private boolean isRunning;
	public static void main(String[] args) {
		Server server=new Server();
		server.start();
	}
	
	//启动服务器
	public void start() {
		try {
			serverSocket=new ServerSocket(8888);
			isRunning=true;
			receive();
		} catch (IOException e) {
			e.printStackTrace();
			System.out.println("服务器启动失败........");
			this.stop();
		}
	}
	
	//接收浏览器请求
	public void receive() {
		//建立连接
		while(isRunning) {
		try {
			Socket client = serverSocket.accept();
			System.out.println("一个客户端建立了连接");
			new Thread(new Dispatcher(client)).start();
			
		} catch (IOException e) {
			e.printStackTrace();
			System.out.println("客户端连接失败.......");
		}
		}
		
	}
	
	//停止服务器
	public void stop() {
		isRunning=false;
		System.out.println("服务器已停止.......");
		try {
			serverSocket.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
	
	
}

最终效果:
在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

需要项目源码的可以私信我哈!2823863294@qq.com

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值