`
uule
  • 浏览: 6358730 次
  • 性别: Icon_minigender_1
  • 来自: 一片神奇的土地
社区版块
存档分类
最新评论

(一)Http请求、Http响应、 Socket

 
阅读更多

第一章:一个简单的Web服务器

    本章说明java web服务器是如何工作的。Web服务器也成为超文本传输协议(HTTP)服务器,因为它使用HTTP来跟客户端进行通信的,这通常是个web浏览器。

一个基于java的web服务器使用两个重要的类:java.net.Socket和java.net.ServerSocket,并通过HTTP消息进行通信。因此这章就自然是从HTTP和这两个类的讨论开始的。接下去,解释这章附带的一个简单的web服务器。

 

1.1超文本传输协议(HTTP)

    HTTP是一种协议,允许web服务器和浏览器通过互联网进行来发送和接受数据。它是一种请求和响应协议。客户端请求一个文件而服务器响应请求。HTTP使用可靠的TCP连接--TCP默认使用80端口。第一个HTTP版是HTTP/0.9,然后被HTTP/1.0所替代。正在取代HTTP/1.0的是当前版本HTTP/1.1,它定义于征求意见文档(RFC) 2616,可以从http://www.w3.org/Protocols/HTTP/1.1/rfc2616.pdf下载。

    注意:本节涵盖的HTTP 1.1只是简略的帮助你理解web服务器应用发送的消息。假如你对更多详细信息感兴趣,请阅读RFC 2616。

    在HTTP中,始终都是客户端通过建立连接和发送一个HTTP请求从而开启一个事务。web服务器不需要联系客户端或者对客户端做一个回调连接。无论是客户端或者服务器都可以提前终止连接。举例来说,当你正在使用一个web浏览器的时候,可以通过点击浏览器上的停止按钮来停止一个文件的下载进程,从而有效的关闭与web服务器的HTTP连接。

 

1.2HTTP请求                                     

    一个HTTP请求包括三个组成部分

·  方法 — 统一资源标识符(URI) — 协议/版本

·  请求的头部

·  主体内容

 

    下面是一个HTTP请求的例子:

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

 

    方法—统一资源标识符(URI)—协议/版本出现在请求的第一行。

POST /examples/default.jsp HTTP/1.1

    这里POST是请求方法,/examples/default.jsp是URI,而HTTP/1.1是协议/版本部分。

 

每个HTTP请求可以使用HTTP标准里边提到的多种方法之一。HTTP 1.1支持7种类型的请求:GET, POST,HEAD, OPTIONS, PUT, DELETE和TRACE。

GET和POST在互联网应用里边最普遍使用的。

URI完全指明了一个互联网资源。URI通常是相对服务器的根目录解释的。因此,始终一斜线/开头。统一资源定位器(URL)其实是一种URI(查看http://www.ietf.org/rfc/rfc2396.txt)来的。该协议版本代表了正在使用的HTTP协议的版本。

请求的头部包含了关于客户端环境和请求的主体内容的有用信息。例如它可能包括浏览器设置的语言,主体内容的长度等等。每个头部通过一个回车换行符(CRLF)来分隔的。

对于HTTP请求格式来说,头部和主体内容之间有一个回车换行符(CRLF)是相当重要的。CRLF告诉HTTP服务器主体内容是在什么地方开始的。在一些互联网编程书籍中,CRLF还被认为是HTTP请求的第四部分。

 

在前面一个HTTP请求中,主体内容只不过是下面一行:

lastName=Franks&firstName=Michael

    实体内容在一个典型的HTTP请求中可以很容易的变得更长。

 

1.3 HTTP Response

 

与HTTP Request类似,HTTP Response也由三部分组成:

    协议-状态-描述

Response 头

Response 体

 

举例如下: 

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> 

   

注意响应实体(entity)与响应头(header)之间有一个空白行(CRLF)。

 

1.4   Socket 类

套接字socket代表客户端与服务器连接,你可以通过他与服务器建立连接,可以指定host和port,Java中用Socket类来建立,有多个构造函数。

 

可以通过ServerSocket建立http服务器或者ftp服务器。

 

socket 通信的实例代码如下:

Socket socket = new Socket("127.0.0.1", "8080"); 	//建立连接
  OutputStream os = socket.getOutputStream(); 	//获取输出流
  boolean autoflush = true; 
  PrintWriter out = new PrintWriter( 
		socket.getOutputStream(), autoflush); 	//设置自动flush
  BufferedReader in = new BufferedReader( 
		new InputStreamReader( socket.getInputstream() )); 
 
  // send an HTTP request to the web server 
  out.println("GET /index.jsp HTTP/1.1"); 		//拼装HTTP请求信息
  out.println("Host: localhost:8080"); 
 
  out.println("Connection: Close"); 
  out.println(); 
 
  // read the response 
  boolean loop = true; 
  StringBuffer sb = new StringBuffer(8096); 
  while (loop) { 
    if ( in.ready() ) { 
      int i=0;
	   while (i!=-1) { 
        i = in.read(); 
        sb.append((char) i); 
      } 
      loop = false; 
    } 
    Thread.currentThread().sleep(50); 	//由于是阻塞写入,暂停50ms,保证可以写入。
  } 
 
  // display the response to the out console 
  System.out.println(sb.toString()); 
  socket.close();

   

1.5   ServerSocket 类

Socket类表示一个客户端socket,相应的ServerSocket类表示了一个服务器端应用。服务器端socket需要等待来自客户端的连接请求。一旦ServerSocket接收到来自客户端的连接请求,它会实例化一个Socket类的对象来处理与客户端的通信。

 

1.6  应用举例

该程序包括三个部分,HttpServer、Request和Response。该程序只能发送静态资源,如HTML文件,图片文件,但不会发送响应头信息。

代码文件如下:

 

 服务器:

socket = serverSocket.accept();   

input = socket.getInputStream();   

output = socket.getOutputStream(); 

 

服务器Request将inputStream请求解析为字符串,从中拿到URI

服务器Response拼接服务器路径和URI,得到文件路径,判断是否存在,存在则输出到outputStream请求中,不存在则输出错误字符串到outputstream中

package ex01.pyrmont; 
 
import java.net.Socket; 
import java.net.ServerSocket; 
import java.net.InetAddress; 
import java.io.InputStream; 
import java.io.OutputStream; 
import java.io.IOException; 
import java.io.File; 
 
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; 
        } 
      } 
    }   
} 

 

package ex01.pyrmont; 
 
import java.io.InputStream; 
import java.io.IOException; 
 
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; 
   } 
}

 

package ex01.pyrmont; 
 
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; 
 
   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(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(); 
     } 
   } 
} 

 ...

 Servlet容器处理Servlet请求的简单流程图

 

 

分享到:
评论

相关推荐

    http.rar_http请求响应_socket http 请求

    本文将详细解析"HTTP.RAR_HTTP请求响应_SOCKET HTTP请求"这一主题,帮助你理解如何通过Socket编程实现HTTP请求并接收响应。 首先,让我们了解HTTP的基本概念。HTTP是一种应用层协议,用于在Web上交换各种类型的数据...

    c#使用Socket发送HTTP/HTTPS请求的实现代码

    C#中使用Socket发送HTTP/HTTPS请求是一种高效的方法,特别是在需要自定义HTTP封包或对HTTP请求进行深入控制时。本文主要介绍了如何使用C#的Socket类来发送HTTP/HTTPS请求,并对相关的知识点进行了详细的解释。 ...

    HttpSocket.rar_HTTP_c http请求_httpsocket_http请求类_socket

    本文将深入探讨标题中的"HttpSocket.rar",这是一个基于Windows Socket实现的HTTP请求封装类,它提供了便捷的方式来发送和接收HTTP请求。 首先,让我们来理解一下HTTP协议。HTTP是一种无状态、基于TCP/IP的应用层...

    c语言编写 基于SOCKET HTTP Json 请求

    本文将深入探讨如何使用C语言通过SOCKET实现HTTP协议,并且重点讲解如何构造和发送JSON格式的数据进行POST请求。我们将以Visual Studio 2013作为开发环境,通过分析“scoket_http”这个压缩包中的文件,来揭示这一...

    tomcat 分配请求之——socket获取请求

    在Web服务器中,Tomcat是一个广泛使用的开源应用服务器,它负责解析并响应来自客户端(如浏览器)的HTTP请求。这个过程涉及到多个关键步骤,其中第一步就是通过Socket来接收和处理网络数据。 首先,我们需要理解...

    java socket 发送http请求webservice.

    要发送HTTP请求到Web Service,我们需要创建一个Socket对象,连接到目标服务器的指定端口,然后通过输入流写入HTTP请求,通过输出流读取响应。 下面是一段简单的Java代码示例,演示如何使用Socket发送GET类型的HTTP...

    ZMQ 请求,响应模式deom

    这种模式确保了每个请求都有对应的响应,且请求和响应之间存在一对一的关联关系。 在ZMQ的请求/响应模式中,有几个关键的概念: 1. **REQ socket(请求套接字)**:客户端使用REQ socket发送请求,并等待来自...

    QT实现HTTP请求回应程序

    在本文中,我们将深入探讨如何使用QT来实现HTTP请求和响应的程序,这将涵盖QT的基本概念、HTTP协议原理以及实际的代码实现。 首先,我们需要了解QT中的网络模块。QT提供了一个强大的网络编程接口,它允许开发者方便...

    利用C# Socket 实现HTTP WEB服务器

    在C#中实现HTTP服务器,我们需要创建一个Socket实例来监听特定端口上的连接请求。首先,我们导入System.Net和System.Net.Sockets命名空间,这两个命名空间提供了Socket类和相关的网络功能。以下是一个简单的服务器...

    利用java socket实现发送http请求

    基于java socket发送http请求。这种原生的发送请求的方式能适合一些特定场景,比如: A向B发送请求,A不考虑B是否处理成功,即A不关注B的响应结果,那么A就不用阻塞等B的回应。 传统的Httpclient请求方式都是阻塞...

    VC Socket GET_POST发送HTTP请求示例

    本文将深入探讨使用VC(Visual C++)进行Socket编程,特别是如何通过GET和POST方法发送HTTP请求。HTTP(超文本传输协议)是互联网上应用最广泛的一种网络协议,用于从Web服务器获取或提交数据。 首先,理解Socket...

    vc 用socket 发送http post 请求

    在C++编程环境中,尤其是使用Microsoft Foundation Class (MFC) 库时,向服务器发送HTTP POST请求是一项常见的任务。这通常涉及到网络编程和套接字(Socket)的使用。本教程将详细介绍如何在Visual C++ (VC++) 中...

    MTK平台用Socket实现HTTP请求总结

    在MTK(MediaTek)平台上,使用Socket实现HTTP请求是一项常见的任务,特别是在开发嵌入式设备或物联网应用时。本文将深入探讨如何在MTK平台上通过Socket编程来发送HTTP请求,涉及的知识点包括Socket基本原理、HTTP...

    MTK中使用socket实现http的get请求

    `GRZHttpUtil.c`很可能包含了HTTP GET请求的具体实现,包括构造请求头、建立socket连接、发送请求、接收响应的过程。而`GRZHttpUtilProt.h`可能是对应的头文件,包含了函数声明、常量定义和数据结构等。 在`...

    C++编写的Linux下Http请求

    它是一个基于请求与响应模式的、无状态的、应用层的协议。C++是一种通用、面向对象的编程语言,它提供了丰富的库支持,使得在Linux环境下编写HTTP客户端程序成为可能。 首先,要进行HTTP请求,我们需要理解HTTP请求...

    使用socket实现的请求代理

    本文将深入探讨如何使用Socket实现一个请求代理,该代理能够转发客户端的HTTP请求到目标服务器,并将响应返回给客户端。首先,我们需要理解Socket的基本概念。 Socket,又称为套接字,是操作系统提供的一种进程间...

    SOCKET转HTTP

    这通常涉及到将Socket通信中的数据封装成HTTP请求或响应格式。 1. **HTTP请求格式**: HTTP请求由请求行、请求头、空行和请求体四部分组成。 2. **HTTP响应格式**: HTTP响应同样由状态行、响应头、空行和响应...

    oc语言做的socket网络请求链接

    当我们谈论“oc语言做的socket网络请求链接”时,我们实际上是在讨论使用Objective-C来实现基于Socket的网络通信。Socket编程允许应用直接与服务器进行低级别、高性能的数据交换,是网络编程的基础。在这个场景中,...

    Http.rar_C http_http请求_j2me socket cmnet_s40 cmnet socket_visual

    对于HTTP请求,通常先建立Socket连接,然后发送HTTP头和请求体,最后接收服务器响应。 描述中提到的“包括cmwap和cmnet的请求方式”涉及到中国移动的两种GPRS接入方式。CMWAP(China Mobile Wireless Application ...

    php 利用socket发送HTTP请求(GET,POST)

    今天给大家带来的是如何利用socket发送GET,POST请求。我借用燕十八老师封装好的一个Http类给进行说明。...服务器根据请求信息返回一个响应信息。连接断开。  HTTP请求的格式如下所示: []  

Global site tag (gtag.js) - Google Analytics