php的socket

来源:互联网 发布:mysql可视化工具下载 编辑:程序博客网 时间:2024/05/13 01:19

建立Socket连接至少需要一对套接字,其中一个运行于客户端,称为ClientSocket ,另一个运行于服务器端,称为ServerSocket 。
套接字之间的连接过程分为三个步骤:服务器监听,客户端请求,连接确认。
服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
客户端请求:指客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。
连接确认:当服务器端套接字监听到或者说接收到客户端套接字的连接请求时,就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户端,一旦客户端确认了此描述,双方就正式建立连接。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。

认识socket相关的PHP函数

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类型和一个公共协议类型。产生一个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

socket通讯演示

05172951-a955fce4e5d04082828e717fe0e102f9.jpg-26kB

服务器端: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端口已经处于被监听状态;接下来我们只要运行客户端程序即可连接上。

QQ截图20151010134926.png-6.1kB

这样就完成第一步服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。

接下来就第二步客户端请求:

<?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配置有问题~)

QQ截图20151010140519.png-29.4kB

QQ截图20151010140341.png-10.5kB

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

具体参考: http://www.cnblogs.com/lchb/articles/3078169.html

-------------------------END------------------------- 
原创粉丝点击