swoole 解决tcp粘包问题

Tcp粘包问题

tcp在发送数据的时候因为存在数据缓存的关系,对于数据在发送的时候在 短时间内 如果连续发送很多小的数据的时候就会有可能一次性一起发送,还有就是对于大的数据就会分开连续发送多次

Tcp粘包处理方案

1、特殊字符

根据客户端与服务端相互约定的特殊的符号,对接收的数据进行分割处理

2、固定包头+包体协议(主流)

通过与在数据传输之后会在tcp的数据包中携带上数据的长度,然后呢服务端就可以根据这个长度,对于数据进行截取

 

第一种处理方案——特殊字符

服务端

//原生处理方法
<?php
$server = new Swoole\Server("0.0.0.0", 9503);


$server->on('connect', function ($server, $fd){
   echo "connection open: {$fd}\n";
});

$server->on('receive', function ($server, $fd, $reactor_id, $data) {
   // echo "接收到数据:".$data."\n";
   var_dump(explode("\r\n", $data));//原生处理方法,使用explode去分割数据
   echo "接收到数据:".$fd."\n";
   $server->send($fd, "Swoole: ok");
});

$server->on('close', function ($server, $fd) {
   echo "connection close: {$fd}\n";
});

$server->start();


//swoole处理方法
<?php
$server = new Swoole\Server("0.0.0.0", 9503);
//swoole处理方法
$server->set([
    'open_eof_check' => true,   //打开EOF检测
    'package_eof'    => "\r\n", //设置EOF
]);

$server->on('connect', function ($server, $fd){
   echo "connection open: {$fd}\n";
});

$server->on('receive', function ($server, $fd, $reactor_id, $data) {
   echo "接收到数据:".$data."\n";
   // echo "接收到数据:".$fd."\n";
   $server->send($fd, "Swoole: ok");
});

$server->on('close', function ($server, $fd) {
   echo "connection close: {$fd}\n";
});

$server->start();

客户端

<?php
$client = new Swoole\Client(SWOOLE_SOCK_TCP );

if (!$client->connect('127.0.0.1', 9503, -1)) {
    exit("connect failed. Error: {$client->errCode}\n");
}

//数据分隔符
$end = "\r\n";

for ($i=0; $i < 100; $i++) {
    //发送消息
    $client->send("hello_world_".$end);
}
//接收消息
echo $client->recv();

第二种处理方法——固定包头

服务端 

//原生处理方法
<?php
$server = new Swoole\Server("0.0.0.0", 9503);

$server->on('connect', function ($server, $fd){
   echo "connection open: {$fd}\n";
});

$server->on('receive', function ($server, $fd, $reactor_id, $data) {
   $fooLen = unpack("n", substr($data, 0, 2))[1];
   // 得到真正的数据
   $context = substr($data, 2, $fooLen);
   $server->send($fd, "Swoole: ok");
});

$server->on('close', function ($server, $fd) {
   echo "connection close: {$fd}\n";
});

$server->start();

//swoole处理方法
<?php
$server = new Swoole\Server("0.0.0.0", 9503);

$server->set([
  'open_length_check'     => true,// 打开包长检测特性
  'package_max_length'    => 2 * 1024 * 1024,//最大允许的包长度。因为在一个请求包完整接收前,需要将所有数据保存在内存中,所以需要做保护。避免内存占用过大。
  'package_length_type'   => 'n',//长度字段的类型,固定包头中用一个4字节或2字节表示包体长度。
  'package_length_offset' => 0,//从第几个字节开始是长度,比如包头长度为120字节,第10个字节为长度值,这里填入9(从0开始计数
  'package_body_offset'   => 2,//从第几个字节开始计算长度,比如包头为长度为120字节,第10个字节为长度值,包体长度为1000。如果长度包含包头,这里填入0,如果不包含包头,这里填入120
]);

$server->on('connect', function ($server, $fd){
   echo "connection open: {$fd}\n";
});

$server->on('receive', function ($server, $fd, $reactor_id, $data) {
   echo "接收到数据:".$data."\n";
   $server->send($fd, "Swoole: ok");
});

$server->on('close', function ($server, $fd) {
   echo "connection close: {$fd}\n";
});

$server->start();

客户端

<?php
$client = new Swoole\Client(SWOOLE_SOCK_TCP );

if (!$client->connect('127.0.0.1', 9503, -1)) {
    exit("connect failed. Error: {$client->errCode}\n");
}

// pack(); // 将数据打包成二进制字符串
// unpack(); // 解析二进制字符串

for ($i=0; $i < 10; $i++) {
    $context = "123";
    // 利用pack打包长度
    $len = pack("n", strlen($context));
    // 组包
    $send = $len . $context;
    // 发送
    $client->send($send);
}

echo $client->recv();

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值