@a5635268
2015-10-10T10:31:14.000000Z
字数 5127
阅读 2097
PHP
建立Socket连接至少需要一对套接字,其中一个运行于客户端,称为ClientSocket ,另一个运行于服务器端,称为ServerSocket 。
套接字之间的连接过程分为三个步骤:服务器监听,客户端请求,连接确认。
服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
客户端请求:指客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。
连接确认:当服务器端套接字监听到或者说接收到客户端套接字的连接请求时,就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户端,一旦客户端确认了此描述,双方就正式建立连接。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。
socket_accept() # 接受一个Socket连接socket_bind() # 把socket绑定在一个IP地址和端口上socket_clear_error() # 清除socket的错误或者最后的错误代码socket_close() # 关闭一个socket资源socket_connect() # 开始一个socket连接socket_create_listen() # 在指定端口打开一个socket监听socket_create_pair() # 产生一对没有区别的socket到一个数组里socket_create() # 产生一个socket,相当于产生一个socket的数据结构socket_get_option() # 获取socket选项socket_getpeername() # 获取远程类似主机的ip地址socket_getsockname() # 获取本地socket的ip地址socket_iovec_add() # 添加一个新的向量到一个分散/聚合的数组socket_iovec_alloc() # 这个函数创建一个能够发送接收读写的iovec数据结构socket_iovec_delete() # 删除一个已经分配的iovecsocket_iovec_fetch() # 返回指定的iovec资源的数据socket_iovec_free() # 释放一个iovec资源socket_iovec_set() # 设置iovec的数据新值socket_last_error() # 获取当前socket的最后错误代码socket_listen() # 监听由指定socket的所有连接socket_read() # 读取指定长度的数据socket_readv() # 读取从分散/聚合数组过来的数据socket_recv() # 从socket里结束数据到缓存socket_recvfrom() # 接受数据从指定的socket,如果没有指定则默认当前socketsocket_recvmsg() # 从iovec里接受消息socket_select() # 多路选择socket_send() # 这个函数发送数据到已连接的socketsocket_sendmsg() # 发送消息到socketsocket_sendto() # 发送消息到指定地址的socketsocket_set_block() # 在socket里设置为块模式socket_set_nonblock() # socket里设置为非块模式socket_set_option() # 设置socket选项socket_shutdown() # 这个函数允许你关闭读、写、或者指定的socketsocket_strerror() # 返回指定错误号的详细错误socket_write() # 写数据到socket缓存socket_writev() # 写数据到分散/聚合数组
产生一个Socket,你需要三个变量:一个协议、一个socket类型和一个公共协议类型。产生一个socket有三种协议供选择,继续看下面的内容来获取详细的协议内容。
定义一个公共的协议类型是进行连接一个必不可少的元素。下面的表我们看看有那些公共的协议类型。
表一:协议
| 名字/常量 | 描述 |
|---|---|
| AF_INET | 这是大多数用来产生socket的协议,使用TCP或UDP来传输,用在IPv4的地址 |
| AF_INET6 | 与上面类似,不过是来用在IPv6的地址 |
| AF_UNIX | 本地协议,使用在Unix和Linux系统上,它很少使用,一般都是当客户端和服务器在同一台及其上的时候使用 |
表二:Socket类型
| 名字/常量 | 描述 |
|---|---|
| SOCK_STREAM | 这个协议是按照顺序的、可靠的、数据完整的基于字节流的连接。这是一个使用最多的socket类型,这个socket是使用TCP来进行传输。 |
| SOCK_DGRAM | 这个协议是无连接的、固定长度的传输调用。该协议是不可靠的,使用UDP来进行它的连接。 |
| SOCK_SEQPACKET | 这个协议是双线路的、可靠的连接,发送固定长度的数据包进行传输。必须把这个包完整的接受才能进行读取。 |
| SOCK_RAW | 这个socket类型提供单一的网络访问,这个socket类型使用ICMP公共协议。(ping、traceroute使用该协议) |
| SOCK_RDM | 这个类型是很少使用的,在大部分的操作系统上没有实现,它是提供给数据链路层使用,不保证数据包的顺序 |
表三:公共协议
| 名字/常量 | 描述 |
|---|---|
| ICMP | 互联网控制消息协议,主要使用在网关和主机上,用来检查网络状况和报告错误信息 |
| UDP | 用户数据报文协议,它是一个无连接,不可靠的传输协议 |
| TCP | 传输控制协议,这是一个使用最多的可靠的公共协议,它能保证数据包能够到达接受者那儿,如果在传输过程中发生错误,那么它将重新发送出错数据包。 |
现在你知道了产生一个socket的三个元素,那么我们就在php中使用socket_create()函数来产生一个socket。这个socket_create()函数需要三个参数:一个协议、一个socket类型、一个公共协议。socket_create()函数运行成功返回一个包含socket的资源类型,如果没有成功则返回false

服务器端:server.php
<?php//确保在连接客户端时不会超时set_time_limit(0);$ip = '127.0.0.1';$port = 1935;/*+-------------------------------* @socket通信整个过程+-------------------------------* @socket_create* @socket_bind* @socket_listen* @socket_accept* @socket_read* @socket_write* @socket_close+--------------------------------*//*---------------- 以下操作都是手册上的 -------------------*/if (($sock = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)) < 0) { // 创建一个Socket链接echo "socket_create() 失败的原因是:" . socket_strerror($sock) . "\n";}if (($ret = socket_bind($sock, $ip, $port)) < 0) { //绑定Socket到端口echo "socket_bind() 失败的原因是:" . socket_strerror($ret) . "\n";}if (($ret = socket_listen($sock, 4)) < 0) { // 开始监听链接链接echo "socket_listen() 失败的原因是:" . socket_strerror($ret) . "\n";}$count = 0;do {if (($msgsock = socket_accept($sock)) < 0) { //堵塞等待另一个Socket来处理通信echo "socket_accept() failed: reason: " . socket_strerror($msgsock) . "\n";break;} else {//发送消息到客户端$msg = "测试成功!\n";socket_write($msgsock, $msg, strlen($msg));//接收客户端消息echo "测试成功了啊\n";$buf = socket_read($msgsock, 8192); // 获得客户端的输入$talkback = "收到的信息:$buf\n";echo $talkback;if (++$count >= 5) {break;};}//echo $buf;socket_close($msgsock);} while (true);socket_close($sock);?>
然后php server.php,发现1935端口已经处于被监听状态;接下来我们只要运行客户端程序即可连接上。

这样就完成第一步服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
接下来就第二步客户端请求:
<?phperror_reporting(E_ALL);set_time_limit(0);echo "<h2>TCP/IP Connection</h2>\n";$port = 1935;$ip = "127.0.0.1";/*+-------------------------------* @socket连接整个过程+-------------------------------* @socket_create* @socket_connect* @socket_write* @socket_read* @socket_close+--------------------------------*/$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);// 第一个参数”AF_INET”用来指定域名;// 第二个参数”SOCK_STREM”告诉函数将创建一个什么类型的Socket(在这个例子中是TCP类型),UDP是SOCK_DGRAMif ($socket < 0) {echo "socket_create() failed: reason: " . socket_strerror($socket) . "\n";} else {echo "OK.\n";}echo "试图连接 '$ip' 端口 '$port'...\n";$result = socket_connect($socket, $ip, $port);if ($result < 0) {echo "socket_connect() failed.\nReason: ($result) " . socket_strerror($result) . "\n";} else {echo "连接OK\n";}$in = "Ho\r\n";$in.= "first blood\r\n";$out = '';if (!socket_write($socket, $in, strlen($in))) {echo "socket_write() failed: reason: " . socket_strerror($socket) . "\n";} else {echo "发送到服务器信息成功!\n";echo "发送的内容为:<font color='red'>$in</font> <br>";}while ($out = socket_read($socket, 8192)) {echo "接收服务器回传信息成功!\n";echo "接受的内容为:", $out;}echo "关闭SOCKET...\n";socket_close($socket);echo "关闭OK\n";?>
这时我们来看看各自的链接(先不管图中的错误,这是我php配置有问题~)


然后服务器端接着处于监听状态,每次client请求都会接到反馈,注意该列使用的socket通讯方式其实是很落后的同步阻塞 IO 模型,其上还有同步非阻塞 IO 模型(select/poll 的同步模型)以及使用 epoll/kqueue 的异步模型:属于异步阻塞/非阻塞 IO 模型;(大多数都是epoll/kqueue模型)