整体结构就是使用ServerSocket监听一个地址,当有接受到请求之后,解析请求中的资源路径。服务器资源路径存放在项目下的一个目录中,服务器会到这个目录中根据请求的路径去寻找相应的资源。如果找到了则返回该文件内容,否则提示找不到文件。

功能主要分为三块,一块是监听IP和端口号;一块是接受HTTP请求报文,并解析报文;最后是处理和返回响应。

HttpServer.java

package com.oolong.webserver;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket; public class HttpServer { public static final String WEB_ROOT =
System.getProperty("user.dir") + File.separator + "webroot";
private final int port = 8080;
private boolean isShutdown = false; // 表示服务器是否关闭 public void waiting() {
ServerSocket serverSocket = null; try {
serverSocket = new ServerSocket(port, 1,
InetAddress.getByName("127.0.0.1"));
} catch (IOException ex) {
ex.printStackTrace();
System.exit(1);
} // 循环等待请求
while(!isShutdown) {
Socket socket = null;
InputStream input = null;
OutputStream output = null; try {
// 从服务器等待队列中获取一个连接
socket = serverSocket.accept();
input = socket.getInputStream();
output = socket.getOutputStream(); // 从输入中解析请求字符串,生成一个请求对象
HttpRequest request = new HttpRequest(input);
request.parse(); // 创建一个响应对象返回内容
HttpResponse response = new HttpResponse(output);
response.setRequest(request);
response.sendStaticResource(); // 关闭socket
socket.close(); } catch (Exception ex) {
ex.printStackTrace();
}
}
} public static void main(String[] args) {
HttpServer server = new HttpServer();
server.waiting();
}
}

这里利用ServerSocket接受请求的Socket:

socket = serverSocket.accept();

然后从请求的Socket中获取到输入流和输出流:

input = socket.getInputStream();
output = socket.getOutputStream();

将输入流交给一个HttpRequest对象处理,进行解析请求:

HttpRequest request = new HttpRequest(input);
request.parse();

将解析后的请求对象和输出流交给HttpResponse对象,用来返回响应:

HttpResponse response = new HttpResponse(output);
response.setRequest(request);
response.sendStaticResource();

下面分别看看请求对象和响应对象的实现。

HttpRequest.java

package com.oolong.webserver;

import java.io.IOException;
import java.io.InputStream; public class HttpRequest { private InputStream input;
private String uri; public HttpRequest(InputStream input) {
this.input = input;
} public void parse() {
// 从socket中读取字符流
StringBuffer requestStr = new StringBuffer(2048);
int i;
byte[] buffer = new byte[2048]; try {
i = input.read(buffer);
} catch (IOException ex) {
ex.printStackTrace();
i = -1;
} for (int j = 0; j < i; j++) {
requestStr.append((char) buffer[j]);
} System.out.println(requestStr.toString());
uri = parseUri(requestStr.toString());
System.out.println(uri);
} private String parseUri(String requestStr) {
int index1, index2;
index1 = requestStr.indexOf(' '); if (index1 != -1) {
index2 = requestStr.indexOf(' ', index1 + 1); if (index2 > index1) {
return requestStr.substring(index1 + 1, index2);
}
} return null;
} public String getUri() {
return uri;
}
}

可以看到这个请求对象中主要就是parse()这个解析请求字符的方法,以及parseUri()这个解析URI的方法。

首先看parse(),它创建了一个缓冲区,然后从输入流中读取请求字符串。然后调用parseUri()解析请求路径。

HttpResponse.java

这个是处理响应的类,看起来稍显复杂。其实只是根据请求解析的URI到资源目录中去寻找对应的文件,然后将文件写入Socket的输出流中。

如果没有找到,则输出一段错误信息即可。

package com.oolong.webserver;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.OutputStream; public class HttpResponse { private static final int BUFFER_SIZE = 1024;
private HttpRequest request;
private OutputStream output; public HttpResponse(OutputStream output) {
this.output = output;
} public void setRequest(HttpRequest request) {
this.request = request;
} public void sendStaticResource() throws IOException {
byte[] bytes = new byte[BUFFER_SIZE];
FileInputStream fis = null;
String filePath = request.getUri() == null ? "" : request.getUri().trim(); // 处理根路径
if (filePath.equals("/")) {
filePath = "/index.html";
} try {
String page = null;
File file = new File(HttpServer.WEB_ROOT, filePath);
Long fileLength = file.length();
byte[] fileContent = new byte[fileLength.intValue()]; if (file.exists()) {
fis = new FileInputStream(file);
fis.read(fileContent);
fis.close(); page = new String(fileContent);
page = warpMessage("200", page);
output.write(page.getBytes()); } else {
String errorMessage = warpMessage("404", "404 File Not Found!");
output.write(errorMessage.getBytes());
}
} catch(Exception ex) {
ex.printStackTrace();
}
} private String warpMessage(String statusCode, String message) {
return "HTTP/1.1 " + statusCode + "\r\n" +
"Content-Type: text/html\r\n" +
"Content-Length: " + message.length() + "\r\n" +
"\r\n" + message;
} public static void main(String[] args) {
System.out.println(HttpServer.WEB_ROOT);
}
}

测试

在项目的根目录下的webroot目录中创建一个简单的index.html页面。

<!DOCTYPE html>
<html>
<head>
<title></title>
</head>
<body>
Hello World!
</body>
</html>

运行HttpServer。然后在浏览器中输入http://127.0.0.1:8080/ 或者http://127.0.0.1:8080/index.html

可以看到:

如果输入其他地址,如:http://127.0.0.1:8080/home.html

注意

你可能在调试的时候会发现,在浏览器发起一次请求的时候,

socket = serverSocket.accept();

可能会执行两次,这是因为浏览器会自动发起一次对icon的请求。这个是浏览器的特性,与代码无关,不是bug。

Socket——实现一个简单的静态网页服务器的更多相关文章

  1. socket编程——一个简单的样例

    从一个简单的使用TCP样例開始socket编程,其基本过程例如以下: server                                                  client ++ ...

  2. 使用Socket模拟一个简单的Webservice调用

    webservice是对socket的一个封装,让远程调用调用变得更加简单,那么使用socket究竟有多么麻烦呢?来看看. 做一个简单的天气查询: 服务端: public class SocketSe ...

  3. socket计划——一个简单的例子

    从一个简单易用TCP样品开始socket计划,的基本过程例如下列: server                                                  client +++ ...

  4. socket编程——一个简单的例子

    从一个简单的使用TCP例子开始socket编程,其基本步骤如下: server                                                  client ++++ ...

  5. 初始nginx(启动运行) 使用nginx做一个简单的静态资源服务器

    第一次接触nginx的时候,那时候公司还是用的一些不知名的小技术,后来公司发展问题,重新招了人,然后接触到nginx,公司 使用nginx用来做代理服务器,所有请求 都先经过nginx服务器,然后交由 ...

  6. Socket实现一个简单的半双工通信

    Socket是client进行在网络与server进行数据交互的一种基本通信方式.通信有三种通信.即单工.半双工,和全双工. 所谓单工,就是仅仅可以进行单向通信,如bb机. 而半双工就是一来一回的通信 ...

  7. Css打造一个简单的静态七巧板

    偶然在微博上看到用css写一个七巧板,正好也有一些源代码,于是就试着敲了敲. 主要是利用了css3的transform,实现平移,旋转,变形,直接用看到的代码敲出来之后有些问题,因为宽度上下面绿色的三 ...

  8. socket实现一个简单的echo服务

    服务端的实现: public class EchoServer{ //创建一个serverSocket private final ServerSocket serverSocket; //创建一个构 ...

  9. 用socket写一个简单的服务器

    import socketsk=socket.socket()sk.bind(("127.0.0.1",7001))sk.listen()def login(url): with ...

随机推荐

  1. string字符串长度和字节长度问题

    string str = "abcdef 安安安"; int i = str.Length; byte[] bt = System.Text.Encoding.Default.Ge ...

  2. Linux工具- Sysdig

    Sysdig 是一个超级系统工具,比 strace.tcpdump.lsof 加起来还强大.可用来捕获系统状态信息,保存数据并进行过滤和分析.使用 Lua 开发,提供命令行接口以及强大的交互界面. 使 ...

  3. 深入学习Mybatis框架(二)- 进阶

    1.动态SQL 1.1 什么是动态SQL? 动态SQL就是通过传入的参数不一样,可以组成不同结构的SQL语句. 这种可以根据参数的条件而改变SQL结构的SQL语句,我们称为动态SQL语句.使用动态SQ ...

  4. 辨析 const指针 和 指向常量的指针

    辨析以下几种指针p的定义. ; int *p = &tmp; const int *p = &tmp; int const* p = &tmp; int * const p = ...

  5. 写Java也得了解CPU–CPU缓存

    CPU,一般认为写C/C++的才需要了解,写高级语言的(Java/C#/pathon…)并不需要了解那么底层的东西.我一开始也是这么想的,但直到碰到LMAX的Disruptor,以及马丁的博文,才发现 ...

  6. 手机 简易浏览器 WebView的基本使用 返回 缓存 进度条

    public class MainActivity extends AppCompatActivity { private WebView webView; private String url = ...

  7. c++字符和字符串转整数类型及大小端

    在网络传输中,很多数据都是按字节传递而不是字符串.最近就遇到了这个问题,在刚开始学c语言时都没有问题,可能太久不用了,记录一下 在报中文,用2个字节hex码来表示报文正文长度,什么是hex码呢 就是1 ...

  8. LeetCode02 - 两数相加(Java 实现)

    LeetCode02 - 两数相加(Java 实现) 来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/add-two-numbers 题目描述 ...

  9. linux 远程配置docker加速器

    https://www.jianshu.com/p/dca49964af04 curl -sSL https://get.daocloud.io/daotools/set_mirror.sh | sh ...

  10. 在MariaDB导入sql文件出现乱码解决方案

    第一方式: 命令: mysql -u(用户名) -p --default-character-set=utf8(设置编码) data_name(数据库名)<文件路径 示例: 这时,会有一个假死的 ...