前言
在上一篇《网络编程-从TCP连接的建立说起》中简单介绍了TCP连接的建立,本文暂时先抛开TCP更加详细的介绍,来看看如何实现一个简单的网络程序。
一个简单的echo程序
本文以及后续文章都将会围绕该程序进行介绍。程序大体流程如下:
首先启动服务端,客户端通过TCP的三次握手与服务端建立连接;而后,客户端发送一段字符串,服务端收到字符串后,原封不动的发回给客户端。
我们先将代码呈现,后面再进行更加详细的解释。
客户端代码client.c如下:
//client.c
服务端代码server.c如下:
//server.c
编译运行
编译客户端服务端代码:
$ gcc -o client client.c
$ gcc -o server server.c
在两个终端分别运行server和client。
$ ./server
start server at 0.0.0.0:1234
运行客户端,并输入内容:
$ ./client 127.0.0.1 1234
hello 编程珠玑
服务端最终打印:
start server at 0.0.0.0:1234
connect from 127.0.0.1 47536
recv msg:hello 编程珠玑
客户端最终打印:
hello 编程珠玑
recv msg:hello 编程珠玑
从运行结果可以看到,客户端连接到服务端后,发送一段字符串“hello 编程珠玑”后,服务端返回同样的字符串,达到了我们想要的目的。当然代码里有很多地方还需要完善,但这不影响我们对网络编程的学习。
整体流程说明
整体流程可结合下图来理解:
TCP的三次握手,我们在《网络编程-从TCP连接的建立说起》中就已经介绍了。在图中,标示了在调用某些接口后的状态。例如,服务端在调用socket,bind,listen等函数后,处于LISTEN状态;客户端调用connect函数并返回后,完成三次握手,客户端与服务端都处于ESTABLISHED状态。
这些状态我们是可以观察到的,首先在一个终端启动服务器:
$ ./server
start server at 0.0.0.0:1234
在另外一个终端使用netstat命令(或使用ss命令)观察:
$ netstat -anp |grep :1234
tcp 0 0 0.0.0.0:1234 0.0.0.0:* LISTEN 17730/server
netstat命令的使用可参考netstat命令详解,可以看到server程序当前处于LISTEN状态。
而如果客户端进行连接后再观察会发现:
$ netstat -anp |grep :1234
tcp 0 0 0.0.0.0:1234 0.0.0.0:* LISTEN 17730/server
tcp 0 0 127.0.0.1:48094 127.0.0.1:1234 ESTABLISHED 17957/client
tcp 0 0 127.0.0.1:1234 127.0.0.1:48094 ESTABLISHED 17730/server
从结果中看到,客户端此时处于ESTABLISHED状态,而服务端有一条连接处于ESTABLISHED,还有一条处于LISTEN状态,这是为何呢?我们后面再解释。
由于三次握手的过程非常快,其他的状态我们不是很方便能观察到。
那么结合代码,整个流程又是怎样的呢?请看下图:
在弄清楚图中的接口含义之前,实际上你可以认为客户端连接服务器的整个过程你可以看成是这样的:
- 服务端准备(socket,bind,listen,accept等待客户端)
- 客户端准备(socket)
- 客户端连接(connect)
- 服务端收到客户端的连接(accept返回),客户端连接成功,connect返回
- 客户端发送数据(write)
- 服务端接收数据(read),随后又将原数据发回(write)
- 客户端收到来自服务端的数据(read)
当然了,我们需要注意到的是:
- 服务端在accept阻塞的过程中,处于LISTEN状态
- 客户端在connect返回之后完成TCP的三次握手
- 三次握手完成后,客户端与服务端处于ESTABLISHED状态
- 服务端始终有一个处于LISTEN状态
不要着急,对于图中所提到的接口和数据结构的介绍和使用说明都会在后面进行详细介绍。
小结
看到这里,想必你对我们的echo程序的整体已经有了大致的了解。在对这些接口和数据结构进行详细介绍之前,你可以将代码复制并进行编译运行,观察文中提到的内容。