简单JAVA多线程WEB服务器

  • 一个简单的JAVA多线程WEB服务器,有时间的话我会慢慢更新的。
  • 结构很简单,HttpServer serverThread Request Response Util
  • 直接上源码,有注释。

HttpServer.java

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

public class HttpServer {
  /*
   * WEB_ROOT是HTML和其它文件存放的目录. 这里的WEB_ROOT为工作目录下的webroot目录
   */
  public static final String WEB_ROOT = System.getProperty("user.dir") + File.separator + "webroot";
  public static final String SERVERIP = "127.0.0.1";
  public static final int PORT = 8088;

  // 关闭服务命令
  public static final String SHUTDOWN_COMMAND = "/SHUTDOWN";
  public static boolean closeServer;

  public static void main(String[] args) {
    HttpServer server = new HttpServer();
    //等待连接请求
//    Util.scanPort();
    server.swait();
  }

  public void swait() {
    ServerSocket serverSocket = null;
    Thread receiveThread = null;
    System.out.println("WEB_ROOT:"+ WEB_ROOT + "port:" + PORT);
    try {
      //服务器套接字对象
      serverSocket = new ServerSocket(PORT, 10, InetAddress.getByName(SERVERIP));
    } catch (IOException e) {
      e.printStackTrace();
      System.exit(1);
    }
    // 循环等待一个请求
    while (true) {
      Socket socket = null;
      try {
        //等待连接,连接成功后,返回一个Socket对象
        socket = serverSocket.accept();
        receiveThread =new Thread(new serverThread(socket));
        receiveThread.start();
        // 检查是否是关闭服务命令
        if (closeServer) {
          break;
        }
      } catch (Exception e) {
        e.printStackTrace();
        continue;
      }
    }
  }
}

serverThread

package example;

import java.io.*;
import java.net.Socket;

public class serverThread implements Runnable {
    InputStream input;
    OutputStream output;
    Socket clientRequest;

    public serverThread(Socket socket) {
        this.clientRequest = socket;
        try{
            input = clientRequest.getInputStream() ;
            output = clientRequest.getOutputStream() ;
        }catch (IOException e){
            e.printStackTrace();
        }
    }

    @Override
    public void run() {
        // 创建Request对象并解析
        Request request = new Request(input);
        request.parse();
        if (request.getUri().equals(HttpServer.SHUTDOWN_COMMAND)) {
            HttpServer.closeServer = true;//接受关闭服务器指令
            try {
                clientRequest.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        if (HttpServer.closeServer !=true){
            // 创建 Response 对象
            Response response = new Response(output);
            response.setRequest(request);
            try {
                response.sendStaticResource();
            } catch (IOException e) {
                e.printStackTrace();
            }
            try{
                // 关闭 socket 对象
                clientRequest.close();
            }catch (IOException e){
                System.out.println(e.getMessage());
            }
        }

    }

}

Request

package example;

import java.io.InputStream;
import java.io.IOException;

public class Request {
    private InputStream input;
    private String uri;

    public Request(InputStream input) {
        this.input = input;
    }

    //从InputStream中读取request信息,并从request中获取uri值
    public void parse() {
        StringBuffer request = new StringBuffer(2048);
        int i;
        byte[] buffer = new byte[2048];
        try {
            i = input.read(buffer);
        } catch (IOException e) {
            e.printStackTrace();
            i = -1;
        }
        for (int j = 0; j < i; j++) {
            request.append((char) buffer[j]);
        }
        System.out.print(request.toString());
        uri = parseUri(request.toString());
    }
    /*
     *
     * requestString形式如下:
     * GET /index.html HTTP/1.1
     * Host: localhost:8080
     * Connection: keep-alive
     * Cache-Control: max-age=0
     * ...
     * 该函数目的就是为了获取/index.html字符串
     */
    private String parseUri(String requestString) {
        int index1, index2;
        index1 = requestString.indexOf(' ');
        if (index1 != -1) {
            index2 = requestString.indexOf(' ', index1 + 1);
            if (index2 > index1)
                return requestString.substring(index1 + 1, index2);
        }
        return null;
    }

    private String parseMethod(String requestString) {//GET、POST、HEAD、PUT、DELETE、TRACE、CONNECT、OPTIONS
        int index1 = requestString.indexOf(' ');
        if (index1 != -1) {
                return requestString.substring(0, index1);
        }
        return null;
    }

    public String getUri() {
        return uri;
    }

}

Response

package example;

import java.io.OutputStream;
import java.io.IOException;
import java.io.FileInputStream;
import java.io.File;

/*
  HTTP Response = Status-Line
    *(( general-header | response-header | entity-header ) CRLF)
    CRLF
    [ message-body ]
    Status-Line = HTTP-Version SP Status-Code SP Reason-Phrase CRLF
*/

public class Response {

    private static final int BUFFER_SIZE = 1024;
    Request request;
    OutputStream output;
    int lenghtStream;

    public Response(OutputStream output) {
        this.output = output;
    }

    public void setRequest(Request request) {
        this.request = request;
    }

    public void sendStaticResource() throws IOException {
        byte[] bytes = new byte[BUFFER_SIZE];
        FileInputStream fis = null;
        try {
//            将web文件写入到OutputStream字节流中
            File file = new File(HttpServer.WEB_ROOT, request.getUri());
            if (file.exists()) { //如果访问路径存在
                fis = new FileInputStream(file);
                lenghtStream = fis.available();//最大获取2GB,File的length()方法可获取大于2GB大小,或者用java.nio
                int ch = fis.read(bytes, 0, BUFFER_SIZE);
                if (ch != -1){//添加header
                    String headerMessage = "HTTP/1.1 200 OK\r\n" + "Content-Type: text/html; charset=UTF-8\r\n"
                            + "Content-Length: "+ lenghtStream +"\r\n" + "Connection: keep-alive"+"\r\n" + "\r\n";
                    byte[] newBytes = Util.byteSum(headerMessage,bytes);
                    output.write(newBytes, 0, newBytes.length);
                    ch = fis.read(bytes, 0, BUFFER_SIZE);
                    while (ch != -1) {
                        output.write(bytes, 0, ch);
                        ch = fis.read(bytes, 0, BUFFER_SIZE);
                    }
                }
            } else {
                // file not found
                String errorMessage = "HTTP/1.1 404 File Not Found\r\n" + "Content-Type: text/html\r\n"
                        + "Content-Length: 23\r\n" + "\r\n" + "<h1>File Not Found</h1>";
                output.write(errorMessage.getBytes());
            }
        } catch (Exception e) {
            // thrown if cannot instantiate a File object
            System.out.println(e.toString());
        } finally {
            if (fis != null)
                fis.close();
        }
    }
}

Util

package example;

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

public class Util {
    public static byte[] byteSum (byte[] a,byte[] b){
        byte[] c= new byte[a.length+b.length];
        System.arraycopy(a,0,c,0,a.length);
        System.arraycopy(b,0,c,a.length,b.length);
        return c;
    }

    public static byte[] byteSum (String a_string,byte[] b){
        byte[] a = a_string.getBytes();
        byte[] c= new byte[a.length+b.length];
        System.arraycopy(a,0,c,0,a.length);
        System.arraycopy(b,0,c,a.length,b.length);
        return c;
    }

    public static void scanPort (){
        for(int port=1;port<=65535;port++){
            try{
                ServerSocket serverSocket=new ServerSocket(port);
                serverSocket.close();   //及时关闭ServerSocket
            }catch(IOException e){
                System.out.println("端口"+port+" 已经被其他服务器进程占用");
            }
        }
    }
}

源码下载

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,378评论 25 707
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,559评论 18 399
  • 1、摘录 作为人类,我们趋向于享受即时的快乐,避免即时的代价。如果感觉很好,我们就像立即享受;如果感到痛苦,我们就...
    wuyingty阅读 389评论 0 0
  • 1.比特币运行机理概述 账本(区块) 掌握核心概念是学习任何学科的关键。 区块,区块链,交易单,比特币,交易是比特...
    牛秦勇的区块链自留地阅读 1,445评论 3 3