【PHPsocket编程专题(实战篇①)】php-socket通信演示
2015-10-23 14:41
597 查看
建立Socket连接至少需要一对套接字,其中一个运行于客户端,称为ClientSocket ,另一个运行于服务器端,称为ServerSocket 。
套接字之间的连接过程分为三个步骤:服务器监听,客户端请求,连接确认。
服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
客户端请求:指客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。
连接确认:当服务器端套接字监听到或者说接收到客户端套接字的连接请求时,就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户端,一旦客户端确认了此描述,双方就正式建立连接。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。
定义一个公共的协议类型是进行连接一个必不可少的元素。下面的表我们看看有那些公共的协议类型。
表一:协议
| 名字/常量 | 描述 |
|---|----|
| AF_INET | 这是大多数用来产生socket的协议,使用TCP或UDP来传输,用在IPv4的地址 |
| AF_INET6 | 与上面类似,不过是来用在IPv6的地址 |
| AF_UNIX | 本地协议,使用在Unix和Linux系统上,它很少使用,一般都是当客户端和服务器在同一台及其上的时候使用 |
表二:Socket类型
表三:公共协议
现在你知道了产生一个socket的三个元素,那么我们就在php中使用socket_create()函数来产生一个socket。这个socket_create()函数需要三个参数:一个协议、一个socket类型、一个公共协议。socket_create()函数运行成功返回一个包含socket的资源类型,如果没有成功则返回false
服务器端:server.php
然后
这样就完成第一步服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
接下来就第二步客户端请求:
这时我们来看看各自的链接(先不管图中的错误,这是我php配置有问题~)
然后服务器端接着处于监听状态,每次client请求都会接到反馈,注意该列使用的socket通讯方式其实是很落后的同步阻塞 IO 模型,其上还有同步非阻塞 IO 模型(select/poll 的同步模型)以及使用 epoll/kqueue 的异步模型:属于异步阻塞/非阻塞 IO 模型;(大多数都是epoll/kqueue模型)
具体参考: http://www.cnblogs.com/lchb/articles/3078169.html
套接字之间的连接过程分为三个步骤:服务器监听,客户端请求,连接确认。
服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
客户端请求:指客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。
连接确认:当服务器端套接字监听到或者说接收到客户端套接字的连接请求时,就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户端,一旦客户端确认了此描述,双方就正式建立连接。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。
认识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() # 删除一个已经分配的iovec socket_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,如果没有指定则默认当前socket socket_recvmsg() # 从iovec里接受消息 socket_select() # 多路选择 socket_send() # 这个函数发送数据到已连接的socket socket_sendmsg() # 发送消息到socket socket_sendto() # 发送消息到指定地址的socket socket_set_block() # 在socket里设置为块模式 socket_set_nonblock() # socket里设置为非块模式 socket_set_option() # 设置socket选项 socket_shutdown() # 这个函数允许你关闭读、写、或者指定的socket socket_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通讯演示
服务器端: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端口已经处于被监听状态;接下来我们只要运行客户端程序即可连接上。
这样就完成第一步服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
接下来就第二步客户端请求:
<?php error_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_DGRAM if ($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模型)
具体参考: http://www.cnblogs.com/lchb/articles/3078169.html
相关文章推荐
- php的命令行用法
- 10款实用的PHP开源工具
- PHP实现一致性hash
- PHP高级编程之守护进程,实现优雅重启
- PHP获取上周、本周、上月、本月、本季度、上季度时间方法大全
- winform布局 FlowLayoutPanel的控件
- php通过会话控制实现身份验证
- ThinkPHP登录
- php生成代金券码
- PHP 教程
- laravel 5.1 登录后跳转到指定地址
- PHP设计模式三:原型设计模式
- PHP文件下载实例
- JQ+Php实现抽奖 大转盘
- PHP中的错误和异常处理
- PHP中的unset方法究竟能否释放内存
- PHP常用函数整理
- 将远程图片抓取到本地
- YII2笔记之三
- STP选举根桥、根端口、指定端口