编写一个简易的 HTTP 服务器程序

转载:http://blog.jobbole.com/107279/#article-comment

好久没输出了,知识还是要写下总结才能让思路更加清晰。最近在学习计算机网络相关的知识,来聊聊如何编写一个建议的HTTP服务器。

HTTP 服务器

HTTP服务器,就是一个运行在主机上的程序。程序启动了之后,会一直在等待其他所有客户端的请求,接收到请求之后,处理请求,然后发送响应给客户端。客户端和服务器之间使用HTTP协议进行通信,所有遵循HTTP协议的程序都可以作为客户端。

先直接上代码,然后再详细说明实现细节。

测试运行

代码写好之后,运行测试一下,将上面代码保存到server.c,然后编译程序:

./server运行

runserver

服务器运行,监听9001端口。再用netstat命令查看:server_netstat

server程序在监听9001端口,运行正确。接着用浏览器访问http://localhost:9001

browser_server

成功输出了Hello World

再尝试用telnet去模拟HTTP请求:

telnet_http

  • 1、成功连接
  • 2、发送HTTP请求
  • 3、HTTP响应结果

上面是一个最简单的server程序,代码比较简单,省去一些细节,下面通过代码来学习一下socket的编程细节。

启动server的流程

server流程

socket 函数

创建一个套接字,通过各参数指定套接字的类型。

  • family:协议族。AF_INET:IPV4协议;AF_INET6:IPv6协议;AF_LOCAL:Unix域协议;AF_ROUTE:路由套接字;AF_KEY:密钥套接字
  • type:套接字类型。SOCK_STREAM : 字节流套接字;SOCK_DGRAM:数据包套接字;SOCK_SEGPACKET:有序分组套接字;SOCK_RAW:原始套接字
  • protocol:某个协议类型常量。TCP:0,UDP :1, SCTP :2

套接字地址结构

在socket编程中,大部分函数都用到一个指向套接字地址结构的指针作为参数。针对不同的协议类型,会有不同的结构体定义格式,对于ipv4,结构如下所示:

ip地址结构

套接字地址结构的作用是为了将ip地址和端口号传递到socket函数,写成结构体的方式是为了抽象。当作为一个参数传递进任何套接字函数时,套接字地址结构总是以引用方式传递。然而,协议族有很多,因此以这样的指针作为参数之一的任何套接字函数必须处理来自所有支持的任何协议族的套接字地址结构。使用void *作为通用的指针类型,因此,套接字函数被定义为以指向某个通用套接字结构的一个指针作为其参数之一,正如下面的bind函数原型一样。

这就要求,对这些函数的任何调用都必须要将指向特定于协议的套接字地址结构的指针进行强制类型转换,变成某个通用套接字地址结构的指针。例如:

对于所有socket函数而言,sockaddr的唯一用途就是对指向特定协议的套接字地址结构的指针执行强制类型转换,指向要绑定给sockfd的协议地址。

bind函数

将套接字地址结构绑定到套接字

绑定了socket之后,就可以使用该socket开始监听请求了。

listen函数

将sockfd从未连接的套接字转换成一个被动套接字,指示内核应接受指向该套接字的连接请求。

关于backlog参数,内核为任何一个给定的监听套接字维护两个队列: > * 1、未完成连接队列,在队列里面的套接字处于SYN_RCVD状态 > * 2、已完成队列,处于ESTABLISHED状态

两个队列之和不超过backlog的大小。

listen完成之后,socket就处于LISTEN状态,此时的socket调用accept函数就可以接受客户端发来的请求了。

accept函数

第一个参数sockfd是客户端的套接字描述符,第二个是客户端的套接字地址结构,第三个是套接字地址结构的长度。

如果accept成功,那么返回值是由内核自动生成的全新描述符,代表所返回的客户端的TCP连接。

对于accept函数,第一个参数称为监听套接字描述符,返回值称为已连接套接字。服务器仅创建监听套接字,它一直存在。已连接套接字由服务器进程接受的客户连接创建,当服务器完成某个连接的响应后,相应的已连接套接字就被关闭了。

accept函数返回时,会返回套接字描述符或出错指示的整数,以及引用参数中的套接字地址和该地址的大小。如果对返回值不感兴趣,可以把两个引用参数设为空。

accept之后,一个TCP连接就建立起来了,接着,服务器就接受客户端的请求信息,然后做出响应。

recv和send函数

分别用于从客户端读取信息和发送信息到客户端。在此不做过多的解释。

套接字地址结构大小和值-结果参数

可以看到,在bind函数和accept函数里面,都有一个套接字地址结构长度的参数,区别在于一个是值形式,另一个是引用形式。套接字地址结构的传递方式取决于该结构的传递方向:是从进程到内核,还是从内核到进程。

1、从进程到内核:bind、connect、sendto。 函数将指针和指针所指内容的大小都传给了内核,于是内核知道到底需要从进程复制多少数据进来。

2、从内核到进程: accept、recvfrom、getsockname、getperrname。 这四个函数的结构大小是以只引用的方式传递。 因为当函数被调用时,结构大小是一个值,它告诉内核该结构的大小,这样内核在写该结构时不至于越界;当函数返回时,结构大小又是一个结果,它告诉内核在该结构中究竟存储了多少信息。

HTTP响应报文

发送响应给客户端时,发送的报文要遵循HTTP协议,HTTP的响应报文格式如下:

第一行status-line,状态栏,格式:HTTP版本 状态码 状态码代表文字

headers是返回报文的类型,长度等信息,接着是一个空行,然后是响应报文的实体。

一个HTTP响应报文例子:

最后close函数关闭套接字,时刻保持关闭文件描述符是一个很好的编程习惯。

总结

虽然很多东西看起来很简单,但只有自己真正动手做一遍,才发现其中的简单,之后才能说这些基础是最简单的。要更好和更深入地理解系统的知识,你必须重新一点一点地重新构建一次。

### 回答1: 好的,我可以回答这个问题。使用Java可以通过Socket类实现简易的网络编程程序。可以通过创建ServerSocket对象来监听端口,然后通过Socket对象来建立连接并进行数据传输。具体实现可以参考Java官方文档或者相关教程。 ### 回答2: 使用Java的网络编程功能,可以轻松编写一个简单的网络应用程序。下面是一个使用Java编写简易网络编程程序示例: ```java import java.io.*; import java.net.*; public class SimpleNetworkingProgram { public static void main(String[] args) { try { // 创建服务器Socket并监听指定端口 ServerSocket serverSocket = new ServerSocket(8888); System.out.println("服务器正在监听端口 8888..."); // 接受客户端的连接请求 Socket clientSocket = serverSocket.accept(); System.out.println("连接成功!"); // 获取输入输出流 BufferedReader in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream())); PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true); // 读取客户端发送的消息 String clientMessage = in.readLine(); System.out.println("客户端发送的消息:" + clientMessage); // 向客户端发送回复消息 out.println("服务器已收到你的消息:" + clientMessage); // 关闭连接 clientSocket.close(); serverSocket.close(); } catch (IOException e) { e.printStackTrace(); } } } ``` 这是一个简单的服务器程序,它监听8888端口,并在接受到客户端连接请求后,读取客户端发送的消息并发送回复消息。你可以使用Telnet工具或编写一个客户端程序连接到这个服务器并发送消息。 ### 回答3: 使用Java编写一个简易网络编程程序可以通过Java的Socket和ServerSocket类来实现。这两个类分别用于客户端和服务器端的网络通信。 首先,我们需要创建一个服务器端程序。在服务器端,首先创建一个ServerSocket对象,并指定一个端口号,例如: ``` ServerSocket serverSocket = new ServerSocket(8888); ``` 然后,使用serverSocket的accept()方法等待客户端的连接请求,并返回一个Socket对象,例如: ``` Socket socket = serverSocket.accept(); ``` 接下来,可以通过获取socket的输入输出流来进行网络数据的读写操作。例如,可以使用BufferedReader来读取客户端发送的数据,使用PrintWriter向客户端发送数据,如下所示: ``` BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream())); PrintWriter out = new PrintWriter(socket.getOutputStream(), true); ``` 在服务器端,可以通过循环持续地接收和处理客户端发送的数据。可使用while循环进行处理,例如: ``` String inputLine; while ((inputLine = in.readLine()) != null) { // 对接收到的数据进行处理 // 发送响应数据给客户端 } ``` 在客户端,首先创建一个Socket对象,并指定连接的服务器的IP地址和端口号,例如: ``` Socket socket = new Socket("127.0.0.1", 8888); ``` 然后,通过获取socket的输入输出流进行网络数据的读写操作,如下所示: ``` BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream())); PrintWriter out = new PrintWriter(socket.getOutputStream(), true); ``` 在客户端可以使用while循环来发送数据给服务器端,并读取服务器端的响应数据,例如: ``` String inputLine; while ((inputLine = in.readLine()) != null) { // 发送数据给服务器端 // 读取服务器端的响应数据 } ``` 以上就是一个简易的用Java实现的网络编程程序。通过使用Socket和ServerSocket类,我们可以实现客户端和服务器端之间的网络通信。同时,在实际开发中可能还需考虑多线程、异常处理等其他方面。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值