创建 TCP 服务端
程序代码:
<?php
//创建Server对象,监听 127.0.0.1:9501端口
$serv = new Swoole\Server("127.0.0.1", 9501);
//监听连接进入事件
$serv->on('Connect', function ($serv, $fd) {
echo "Client: Connect.\n";
});
//监听数据接收事件
$serv->on('Receive', function ($serv, $fd, $from_id, $data) {
echo "Client $fd send $data";
$serv->send($fd, "Server: ".$data);
});
//监听连接关闭事件
$serv->on('Close', function ($serv, $fd) {
echo "Client: Close.\n";
});
//启动服务器
$serv->start();
这里就创建了一个TCP
服务器,监听本机9501
端口。它的逻辑很简单,当客户端Socket
通过网络发送一个 hello
字符串时,服务器会回复一个 Server: hello
字符串。
Server
是异步服务器,所以是通过监听事件的方式来编写程序的。当对应的事件发生时底层会主动回调指定的函数。如当有新的TCP
连接进入时会执行onConnect
事件回调,当某个连接向服务器发送数据时会回调onReceive
函数。
- 服务器可以同时被成千上万个客户端连接,
$fd
就是客户端连接的唯一标识符 - 调用
$server->send()
方法向客户端连接发送数据,参数就是$fd
客户端标识符 - 调用
$server->close()
方法可以强制关闭某个客户端连接 - 客户端可能会主动断开连接,此时会触发
onClose
事件回调
执行程序
php server.php
在命令行下运行server.php
程序,启动成功后可以使用 netstat
工具看到,已经在监听9501
端口。这时就可以使用telnet/netcat
工具连接服务器。
当客户端Socket
通过网络发送一个 hello
字符串时,服务器会显示客户端的操作,并回复一个 Server: hello
字符串。
创建同步 TCP 客户端
程序代码
<?php
$client = new swoole_client(SWOOLE_SOCK_TCP);
//连接到服务器
if (!$client->connect('127.0.0.1', 9501, 0.5))
{
die("connect failed.");
}
//向服务器发送数据
if (!$client->send("hello world"))
{
die("send failed.");
}
//从服务器接收数据
$data = $client->recv();
if (!$data)
{
die("recv failed.");
}
echo $data;
//关闭连接
$client->close();
创建一个TCP的同步客户端,此客户端可以用于连接到我们第一个示例的TCP服务器。向服务器端发送一个hello world
字符串,服务器会返回一个 Server: hello world
字符串。
这个客户端是同步阻塞的,connect/send/recv 会等待IO完成后再返回。同步阻塞操作并不消耗CPU资源,IO操作未完成当前进程会自动转入sleep
模式,当IO完成后操作系统会唤醒当前进程,继续向下执行代码。
TCP
需要进行3
次握手,所以connect
至少需要3
次网络传输过程- 在发送少量数据时
$client->send
都是可以立即返回的。发送大量数据时,socket
缓存区可能会塞满,send
操作会阻塞。 recv
操作会阻塞等待服务器返回数据,recv
耗时等于服务器处理时间+网络传输耗时之和。
执行程序
php client.php
创建异步 TCP 客户端
异步客户端只能用于cli环境
程序代码
<?php
$client = new swoole_client(SWOOLE_SOCK_TCP, SWOOLE_SOCK_ASYNC);
//注册连接成功回调
$client->on("connect", function($cli) {
$cli->send("hello world\n");
});
//注册数据接收回调
$client->on("receive", function($cli, $data){
echo "Received: ".$data."\n";
});
//注册连接失败回调
$client->on("error", function($cli){
echo "Connect failed\n";
});
//注册连接关闭回调
$client->on("close", function($cli){
echo "Connection close\n";
});
//发起连接
$client->connect('127.0.0.1', 9501, 0.5);
异步客户端与上一个同步TCP客户端不同,异步客户端是非阻塞的。可以用于编写高并发的程序。swoole官方提供的redis-async
、mysql-async
都是基于异步swoole_client实现的。
异步客户端需要设置回调函数,有4个事件回调必须设置onConnect
、onError
、onReceive
、onClose
。分别在客户端连接成功、连接失败、收到数据、连接关闭时触发。
$client->connect()
发起连接的操作会立即返回,不存在任何等待。当对应的IO事件完成后,swoole底层会自动调用设置好的回调函数。
php tcp_async_client.php 报错
PHP Fatal error: Swoole\Client::__construct(): please install the ext-async extension, using Swoole\Async\Client in /home/www/php_demo/network/tcp_async_client.php on line 2
Fatal error: Swoole\Client::__construct(): please install the ext-async extension, using Swoole\Async\Client in /home/www/php_demo/network/tcp_async_client.php on line 2
解决办法,安装 ext-async 扩展 https://github.com/swoole/ext-async
必须先安装好 swoole 扩展再安装 ext-async,因为 ext-async 是依赖 swoole 扩展的