swoole json不完整怎么办
发布时间 - 2020-04-09 00:00:00 点击率:次
swoole json不完整怎么办?
swoole客户端与服务端收发数据完整性问题解决
一、下面这个实例,启动swoole服务后,监听9501端口,接收从客户端发来的数据,原样返回。
class Server
{
private $serv;
public function __construct() {
$this->serv = new swoole_server("127.0.0.1", 9501);
$this-
>serv->set(array(
'worker_num' => 4, //一般设置为服务器CPU数的1-4倍
'daemonize' => 1, //以守护进程执行
'max_request' => 2000,
'dispatch_mode' => 2,//进程数据包分配模式 1平均分配,2按FD取摸固定分配,3抢占式分配
'task_worker_num' => 8, //task进程的数量
"task_ipc_mode " => 3 , //使用消息队列通信,并设置为争抢模式
"log_file" => "./log/taskqueueu.log" ,//日志
));
$this->serv->on('Receive', array($this,'onReceive'));//接收到数据时回调此函数
$this->serv->start();
}
public function onReceive(swoole_server $serv, $fd, $from_id, $data ) {
$serv->send($fd, $data);
usleep(500); //不加延时的话,经常两条数据被合并成一条返回了。
$serv->close($fd);
}
public function onClose(swoole_server $serv, $fd) {
$serv->send($fd, 'CLOSED');
}
}直接new Server启动服务端。
二、启动swoole客户端,向服务端发数据,并接收返回。
$client = new swoole_client(SWOOLE_SOCK_TCP);
if (!$client->connect('127.0.0.1', 9501, -1))
{
exit("connect failed. Error: {$client->errCode}\n");
}
$data=[
'type'=>1,
'data'=>array(
'PlatformCode'=>'...........很长的数据.',
)
];
$sender=$client->send(json_encode($data)."\r\n\r\n");
while($result = $client->recv()){
if($result=='CLOSED'){
echo "任务结束。byebye~\r\n";
break;
}else{
echo $result;
}
}
$client->close();当发送比较大的数据包时,会发现接收到的json包是不完整的,可以使用EOF协议处理的方式来解决,比如在服务端设置:
$serv->set(
array('open_eof_split' => TRUE, 'package_eof' => "\r\n\r\n")
);这样,'\r\n\r\n'就成为EOF协议结束符。
发送数据包时,在包结尾添加上'\r\n\r\n',接口数据时遇到这个字符就认为数据接收完毕,如此就保证的数据的完整性。
注:数据包中不能出现'\r\n\r\n'字符,否则会出现分包错误。
Swoole的Server和异步Client都是在onReceive回调函数中处理数据包,当设置了协议处理后,只有收到一个完整数据包时才会触发onReceive事件。
另外一种方式,也可以预先设定好要发送的包的长度,如:
$server->set(array(
'open_length_check' => true,
'package_max_length' => 81920,
'package_length_type' => 'n', //see php pack()
'package_length_offset' => 0,
'package_body_offset' => 2,
));可以固定包的长度来确保数据的完整性,官方注释如下:
固定包头的协议非常通用,在BAT的服务器程序中经常能看到。这种协议的特点是一个数据包总是由包头+包体2部分组成。包头由一个字段指定了包体或整个包的长度,长度一般是使用2字节/4字节整数来表示。服务器收到包头后,可以根据长度值来精确控制需要再接收多少数据就时完整的数据包。Swoole的配置可以很好的支持这种协议,可以灵活地设置4项参数应对所有情况。
Swoole的Server和异步Client都是在onReceive回调函数中处理数据包,当设置了协议处理后,只有收到一个完整数据包时才会触发onReceive事件。同步客户端在设置了协议处理后,调用 $client->recv() 不再需要传入长度,recv函数在收到完整数据包或发生错误后返回。
# swoole
# json
# EOF
# 回调函数
# 接口
# 事件
# 异步
# 数据包
# 服务端
# 客户端
# 是在
# 回调
# 设置为
# 时才
# 不完整
# 是一个
# 很好
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
作用域操作符会触发自动加载吗_php类自动加载机制与::调用【教程】
大连 网站制作,大连天途有线官网?
深入理解Android中的xmlns:tools属性
郑州企业网站制作公司,郑州招聘网站有哪些?
Laravel Seeder怎么填充数据_Laravel数据库填充器的使用方法与技巧
如何用已有域名快速搭建网站?
Laravel事件监听器怎么写_Laravel Event和Listener使用教程
制作企业网站建设方案,怎样建设一个公司网站?
Linux安全能力提升路径_长期防护思维说明【指导】
Android仿QQ列表左滑删除操作
如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)
Edge浏览器怎么启用睡眠标签页_节省电脑内存占用优化技巧
Laravel如何生成PDF或Excel文件_Laravel文档导出工具与使用教程
Win11怎么设置默认图片查看器_Windows11照片应用关联设置
Laravel storage目录权限问题_Laravel文件写入权限设置
Laravel中DTO是什么概念_在Laravel项目中使用数据传输对象(DTO)
Laravel如何连接多个数据库_Laravel多数据库连接配置与切换教程
Laravel Sail是什么_基于Docker的Laravel本地开发环境Sail入门
高端网站建设与定制开发一站式解决方案 中企动力
打开php文件提示内存不足_怎么调整php内存限制【解决方案】
手机怎么制作网站教程步骤,手机怎么做自己的网页链接?
长沙做网站要多少钱,长沙国安网络怎么样?
Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比
Python企业级消息系统教程_KafkaRabbitMQ高并发应用
Laravel如何操作JSON类型的数据库字段?(Eloquent示例)
html5如何实现懒加载图片_ intersectionobserver api用法【教程】
网站制作免费,什么网站能看正片电影?
简历没回改:利用AI润色让你的文字更专业
Laravel如何使用Eloquent进行子查询
智能起名网站制作软件有哪些,制作logo的软件?
网站建设要注意的标准 促进网站用户好感度!
javascript读取文本节点方法小结
如何在万网利用已有域名快速建站?
深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?
如何制作公司的网站链接,公司想做一个网站,一般需要花多少钱?
消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工
Win11怎么关闭专注助手 Win11关闭免打扰模式设置【操作】
百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏
如何在Tomcat中配置并部署网站项目?
专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?
Laravel怎么在Blade中安全地输出原始HTML内容
如何正确下载安装西数主机建站助手?
Laravel模型事件有哪些_Laravel Model Event生命周期详解
高端建站三要素:定制模板、企业官网与响应式设计优化
利用 Google AI 进行 YouTube 视频 SEO 描述优化
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
EditPlus中的正则表达式 实战(4)
Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践
PHP正则匹配日期和时间(时间戳转换)的实例代码
如何在建站宝盒中设置产品搜索功能?


>serv->set(array(
'worker_num' => 4, //一般设置为服务器CPU数的1-4倍
'daemonize' => 1, //以守护进程执行
'max_request' => 2000,
'dispatch_mode' => 2,//进程数据包分配模式 1平均分配,2按FD取摸固定分配,3抢占式分配
'task_worker_num' => 8, //task进程的数量
"task_ipc_mode " => 3 , //使用消息队列通信,并设置为争抢模式
"log_file" => "./log/taskqueueu.log" ,//日志
));
$this->serv->on('Receive', array($this,'onReceive'));//接收到数据时回调此函数
$this->serv->start();
}
public function onReceive(swoole_server $serv, $fd, $from_id, $data ) {
$serv->send($fd, $data);
usleep(500); //不加延时的话,经常两条数据被合并成一条返回了。
$serv->close($fd);
}
public function onClose(swoole_server $serv, $fd) {
$serv->send($fd, 'CLOSED');
}
}