亚洲国产AV一区二区三区久久_乱人妻中文字幕视频_91麻豆精品国产一级_精品国产欧美另类一区

您的當(dāng)前位置: 首頁(yè)>>赤壁新聞中心>>行業(yè)資訊

tomcat之一個(gè)簡(jiǎn)單的web服務(wù)器

瀏覽量(106687) 時(shí)間:2020-09-09

HTTP 是一種協(xié)議,允許 web 赤壁服務(wù)器和瀏覽器通過(guò)互聯(lián)網(wǎng)進(jìn)行來(lái)發(fā)送和接受數(shù)據(jù)。它是一種 請(qǐng)求和響應(yīng)協(xié)議??蛻舳苏?qǐng)求一個(gè)文件而赤壁服務(wù)器響應(yīng)請(qǐng)求。HTTP 使用可靠的 TCP 連接--TCP 默認(rèn) 使用 80 端口。

第一個(gè) HTTP 版是 HTTP/0.9,然后被 HTTP/1.0 所替代。正在取代 HTTP/1.0 的是 當(dāng)前版本 HTTP/1.1。

在 HTTP 中,始終都是客戶端通過(guò)建立連接和發(fā)送一個(gè) HTTP 請(qǐng)求從而開(kāi)啟一個(gè)事務(wù)。web 服 務(wù)器不需要聯(lián)系客戶端或者對(duì)客戶端做一個(gè)回調(diào)連接。無(wú)論是客戶端或者赤壁服務(wù)器都可以提前終止 連接。舉例來(lái)說(shuō),當(dāng)你正在使用一個(gè) web 瀏覽器的時(shí)候,可以通過(guò)點(diǎn)擊瀏覽器上的停止按鈕來(lái)停 止一個(gè)文件的下載進(jìn)程,從而有效的關(guān)閉與 web 服務(wù)器的 HTTP 連接。
HTTP 請(qǐng)求

一個(gè) HTTP 請(qǐng)求包括三個(gè)組成部分:

    方法—統(tǒng)一資源標(biāo)識(shí)符(URI)—協(xié)議/版本
    請(qǐng)求的頭部
    主體內(nèi)容

下面是一個(gè) HTTP 請(qǐng)求的例子:

POST /examples/default.jsp HTTP/1.1
Accept: text/plain; text/html
Accept-Language: en-gb
Connection: Keep-Alive
Host: localhost
User-Agent: Mozilla/4.0 (compatible; MSIE 4.01; Windows 98) Content-Length: 33
Content-Type: application/x-www-form-urlencoded Accept-Encoding: gzip, deflate

lastName=Franks&firstName=Michael

方法—統(tǒng)一資源標(biāo)識(shí)符(URI)—協(xié)議/版本出現(xiàn)在請(qǐng)求的第一行。
POST /examples/default.jsp HTTP/1.1

這里 POST 是請(qǐng)求方法,/examples/default.jsp 是 URI,而 HTTP/1.1 是協(xié)議/版本部分。

每個(gè) HTTP 請(qǐng)求可以使用 HTTP 標(biāo)準(zhǔn)里邊提到的多種方法之一。HTTP 1.1 支持 7 種類型的請(qǐng)求:GET, POST,HEAD, OPTIONS, PUT, DELETE 和 TRACE。GET 和 POST 在互聯(lián)網(wǎng)應(yīng)用里邊最普遍使用的。

URI 完全指明了一個(gè)互聯(lián)網(wǎng)資源。URI 通常是相對(duì)服務(wù)器的根目錄解釋的。因此,始終一斜 線/開(kāi)頭。

請(qǐng)求的頭部包含了關(guān)于客戶端環(huán)境和請(qǐng)求的主體內(nèi)容的有用信息。例如它可能包括瀏覽器設(shè)置的語(yǔ)言,主體內(nèi)容的長(zhǎng)度等等。每個(gè)頭部通過(guò)一個(gè)回車(chē)換行符(CRLF)來(lái)分隔的。

對(duì)于 HTTP 請(qǐng)求格式來(lái)說(shuō),頭部和主體內(nèi)容之間有一個(gè)回車(chē)換行符(CRLF)是相當(dāng)重要的。CRLF 告訴 HTTP 服務(wù)器主體內(nèi)容是在什么地方開(kāi)始的

在前面一個(gè) HTTP 請(qǐng)求中,主體內(nèi)容只不過(guò)是下面一行:

lastName=Franks&firstName=Michael
HTTP 響應(yīng)

類似于 HTTP 請(qǐng)求,一個(gè) HTTP 響應(yīng)也包括三個(gè)組成部分:

    方法—統(tǒng)一資源標(biāo)識(shí)符(URI)—協(xié)議/版本
    響應(yīng)的頭部
    主體內(nèi)容

下面是一個(gè) HTTP 響應(yīng)的例子:

HTTP/1.1 200 OK
Server: Microsoft-IIS/4.0
Date: Mon, 5 Jan 2004 13:13:33 GMT Content-Type: text/html
Last-Modified: Mon, 5 Jan 2004 13:13:12 GMT Content-Length: 112

<html>
<head>
<title>HTTP Response Example</title> </head>
<body>
Welcome to Brainy Software
</body>
</html>

響應(yīng)頭部的第一行類似于請(qǐng)求頭部的第一行。第一行告訴你該協(xié)議使用 HTTP 1.1,請(qǐng)求成 功(200=成功),表示一切都運(yùn)行良好。

響應(yīng)頭部和請(qǐng)求頭部類似,也包括很多有用的信息。響應(yīng)的主體內(nèi)容是響應(yīng)本身的 HTML 內(nèi) 容。頭部和主體內(nèi)容通過(guò) CRLF 分隔開(kāi)來(lái)。
Socket 類

套接字是網(wǎng)絡(luò)連接的一個(gè)端點(diǎn)。套接字使得一個(gè)應(yīng)用可以從網(wǎng)絡(luò)中讀取和寫(xiě)入數(shù)據(jù)。放在兩 個(gè)不同計(jì)算機(jī)上的兩個(gè)應(yīng)用可以通過(guò)連接發(fā)送和接受字節(jié)流。為了從你的應(yīng)用發(fā)送一條信息到另 一個(gè)應(yīng)用,你需要知道另一個(gè)應(yīng)用的 IP 地址和套接字端口。在 Java 里邊,套接字指的是java.net.Socket 類。

要?jiǎng)?chuàng)建一個(gè)套接字,你可以使用 Socket 類眾多構(gòu)造方法中的一個(gè)。其中一個(gè)接收主機(jī)名稱和端口號(hào):

public Socket (java.lang.String host, int port)
在這里主機(jī)是指遠(yuǎn)程機(jī)器名稱或者 IP 地址,端口是指遠(yuǎn)程應(yīng)用的端口號(hào)。例如,要連接yahoo.com 的 80 端口,你需要構(gòu)造以下的 Socket 對(duì)象:

new Socket ("yahoo.com", 80);

一旦你成功創(chuàng)建了一個(gè) Socket 類的實(shí)例,你可以使用它來(lái)發(fā)送和接受字節(jié)流。要發(fā)送字節(jié)流,你首先必須調(diào)用 Socket 類的 getOutputStream 方法來(lái)獲取一個(gè) java.io.OutputStream 對(duì)象。
要發(fā)送文本到一個(gè)遠(yuǎn)程應(yīng)用,你經(jīng)常要從返回的 OutputStream 對(duì)象中構(gòu)造一個(gè) java.io.PrintWriter 對(duì)象。要從連接的另一端接受字節(jié)流,你可以調(diào)用 Socket 類的 getInputStream 方法用來(lái)返回一個(gè) java.io.InputStream 對(duì)象。
ServerSocket 類

Socket 類代表一個(gè)客戶端套接字,即任何時(shí)候你想連接到一個(gè)遠(yuǎn)程服務(wù)器應(yīng)用的時(shí)候你構(gòu)造的套接字,現(xiàn)在,假如你想實(shí)施一個(gè)服務(wù)器應(yīng)用,例如一個(gè) HTTP 服務(wù)器或者 FTP 服務(wù)器,你需要一種不同的做法。這是因?yàn)槟愕姆?wù)器必須隨時(shí)待命,因?yàn)樗恢酪粋€(gè)客戶端應(yīng)用什么時(shí)候會(huì)嘗試去連接它。為了讓你的應(yīng)用能隨時(shí)待命,你需要使用 java.net.ServerSocket 類。這是 服務(wù)器套接字的實(shí)現(xiàn).

ServerSocket 和 Socket 不同,服務(wù)器套接字的角色是等待來(lái)自客戶端的連接請(qǐng)求。一旦服 務(wù)器套接字獲得一個(gè)連接請(qǐng)求,它創(chuàng)建一個(gè) Socket 實(shí)例來(lái)與客戶端進(jìn)行通信。

要?jiǎng)?chuàng)建一個(gè)服務(wù)器套接字,你需要使用 ServerSocket 類提供的四個(gè)構(gòu)造方法中的一個(gè)。你 需要指定 IP 地址和服務(wù)器套接字將要進(jìn)行監(jiān)聽(tīng)的端口號(hào)。通常,IP 地址將會(huì)是 127.0.0.1,也 就是說(shuō),服務(wù)器套接字將會(huì)監(jiān)聽(tīng)本地機(jī)器。服務(wù)器套接字正在監(jiān)聽(tīng)的 IP 地址被稱為是綁定地址。 服務(wù)器套接字的另一個(gè)重要的屬性是 backlog,這是服務(wù)器套接字開(kāi)始拒絕傳入的請(qǐng)求之前,傳 入的連接請(qǐng)求的最大隊(duì)列長(zhǎng)度。

其中一個(gè) ServerSocket 類的構(gòu)造方法如下所示:

public ServerSocket(int port, int backLog, InetAddress bindingAddress);

對(duì)于這個(gè)構(gòu)造方法,綁定地址必須是 java.net.InetAddress 的一個(gè)實(shí)例。一種構(gòu)造 InetAddress 對(duì)象的簡(jiǎn)單的方法是調(diào)用它的靜態(tài)方法 getByName,傳入一個(gè)包含主機(jī)名稱的字符
串,就像下面的代碼一樣。

InetAddress.getByName("127.0.0.1");

下面一行代碼構(gòu)造了一個(gè)監(jiān)聽(tīng)的本地機(jī)器 8080 端口的 ServerSocket,它的 backlog 為 1。

new ServerSocket(8080, 1, InetAddress.getByName("127.0.0.1"));

一旦你有一個(gè) ServerSocket 實(shí)例,你可以讓它在綁定地址和服務(wù)器套接字正在監(jiān)聽(tīng)的端口
上等待傳入的連接請(qǐng)求。你可以通過(guò)調(diào)用 ServerSocket 類的 accept 方法做到這點(diǎn)。這個(gè)方法只 會(huì)在有連接請(qǐng)求時(shí)才會(huì)返回,并且返回值是一個(gè) Socket 類的實(shí)例。Socket 對(duì)象接下去可以發(fā)送 字節(jié)流并從客戶端應(yīng)用中接受字節(jié)流.

HttpServer.java

package org.how.tomcat.works.ex01;
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;

/**
 * @author : Ares
 * @createTime : Aug 21, 2012 9:45:01 PM
 * @version : 1.0
 * @description :
 */
public class HttpServer {

    /**
     * WEB_ROOT is the directory where our HTML and other files reside. For this
     * package, WEB_ROOT is the "webroot" directory under the working directory.
     * The working directory is the location in the file system from where the
     * java command was invoked.
     */
    public static final String WEB_ROOT = System.getProperty("user.dir")
            + File.separator + "webroot";

    // shutdown command
    private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";

    // the shutdown command received
    private boolean shutdown = false;

    public static void main(String[] args) {
        HttpServer server = new HttpServer();
        server.await();
    }

    public void await() {
        ServerSocket serverSocket = null;
        int port = 8080;
        try {
            serverSocket = new ServerSocket(port, 1, InetAddress
                    .getByName("127.0.0.1"));
        } catch (IOException e) {
            e.printStackTrace();
            System.exit(1);
        }
        // Loop waiting for a request
        while (!shutdown) {
            Socket socket = null;
            InputStream input = null;
            OutputStream output = null;

            try {
                socket = serverSocket.accept();
                input = socket.getInputStream();
                output = socket.getOutputStream();
                // create Request object and parse
                Request request = new Request(input);
                request.parse();

                // create Response object
                Response response = new Response(output);
                response.setRequest(request);
                response.sendStaticResource();

                // Close the socket
                socket.close();
                // check if the previous URI is a shutdown command
                shutdown = request.getUri().equals(SHUTDOWN_COMMAND);
            } catch (Exception e) {
                e.printStackTrace();
                continue;
            }
        }
    }
}


Request

package org.how.tomcat.works.ex01;
import java.io.IOException;
import java.io.InputStream;

/**
 * @author : Ares
 * @createTime : Aug 21, 2012 9:48:45 PM
 * @version : 1.0
 * @description :
 */
public class Request {
    private InputStream input;
    private String uri;

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

    public void parse() {
        // Read a set of characters from the socket
        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());
    }

    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;
    }

    public String getUri() {
        return uri;
    }
}

Response

package org.how.tomcat.works.ex01;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;

/**
 * @author : Ares
 * @createTime : Aug 21, 2012 9:51:37 PM
 * @version : 1.0
 * @description :
 *
 * 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;

    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 {
            File file = new File(HttpServer.WEB_ROOT, request.getUri());
            if (file.exists()) {
                fis = new FileInputStream(file);
                int ch = fis.read(bytes, 0, BUFFER_SIZE);
                while (ch != -1) {
                    output.write("HTTP/1.0 200 OK\r\nContent-Type: text/html\r\n\r\n".getBytes());
                    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();
            }
        }
    }
}



最新文章