php接口是8282,thinkphp5部署workerman长连接

我的博客文章网址:

由于最近做一个物联网项目,该项目需要远程将温度推送到服务器,并由服务器推送到web前台,硬件可以利用tcp协议将数据上传到到服务器,但是由于不固定ip的原因,服务器是找不到web前端的,而在这个时候我们就需要利用到长连接workerman,在之前我曾经利用workerman进行聊天软件的编写,实现多人在线聊天功能,而在这之前我没有将这个博客编写成功,本次趁着这个机会,我将长连接的知识进行了复习,正好博客完成时间没有多久,于是将这次经历记录在案,以备以后自己查看学习,也方便了看到这个博文并且可能需要用到该框架的同学们。最后,我需要感谢一位同学,他在这个项目中帮助了我很多,。

在进行该教程之前,我们需要了解一下workerman这个框架和thinkphp这个框架,thinkphp框架是PHP几大框架之一,想要了解的同学们可以到ThinkPHP5.0完全开发手册上进行文档的阅读,以下简称tp5,tp5框架是国产为数不多的优秀框架之一,国产的哦!

而workerman则是在workerman官网上有详细的介绍,同学们可以到该网站上进行手册的查看。

bb572d3af45c

在ThinkPHP5.0完全开发手册里面有一篇介绍他们两个融合的文章composer包workerman在这篇文章里介绍了tp5融合workerman的教程,但是介绍的过于简洁,我试验了两次并没有走通,而在这次的记录中我将我的融合过程和这个方法进行对比,分析以前没有走通的原因。

不论在哪个方法中,我们都需要将wokerman的包引入,我们需要用到composer,没有安装composer的同学需要自行安装。

第一步

导入workerman包:

composer require topthink/think-worker

如果windows服务器还要利用以下命令:

composer require workerman/workerman-for-win

如运行出现错误PHP Fatal error: Call to undefined function Workerman\Lib\pcntl_signal(),需要删除vendor\workerman\workerman,防止命名覆盖。

当包引入完成后,我们会在项目根目录下的vendor文件夹下看到workerman文件夹,这样框架就引入了该项目,我们下一步需要配置服务的启动和引用。

第二步

在这一步中我们需要将启动服务文件放入到项目根目录中,在根目录中我们新建启动服务文件server.php

代码如下:

#!/usr/bin/env php

define('APP_PATH', __DIR__ . '/application/');

//这是原来的代码

//define('BIND_MODULE','push/Worker');

//这是我修改后的代码

define('BIND_MODULE','push/Workertest/index');

// 加载框架引导文件

require __DIR__ . '/thinkphp/start.php';

这个代码的意思是绑定workerman的模块是/application/push/Worker这个控制器,但是由于我们没有用tp5这个框架的引用方式这里我们将代码改为如上所示,直接进入到控制器的index方法,在下一步我们会介绍为什么用这种方法。我们在下一步中就会定义模块,也就是实现功能的地方:/application/push/workertest.php

第三步

在这一步开始前我们来看一下tp5开发手册中的worker.php,需要声明的是我并没用利用该方法。

一共有一个变量和5个方法,变量是定义端口和域名的,方法是分别为,连接上时,服务开始时,接到信息时,错误时,断开时的相应处理方法。

代码如下

namespace app\push\controller;

use think\worker\Server;

class Worker extends Server{

protected $socket = 'websocket://push.app:2346';

/**

* 收到信息

* @param $connection

* @param $data

*/

public function onMessage($connection, $data)

{

$connection->send('我收到你的信息了');

}

/**

* 当连接建立时触发的回调函数

* @param $connection

*/

public function onConnect($connection)

{

}

/**

* 当连接断开时触发的回调函数

* @param $connection

*/

public function onClose($connection)

{

}

/**

* 当客户端的连接上发生错误时触发

* @param $connection

* @param $code

* @param $msg

*/

public function onError($connection, $code, $msg)

{

echo "error $code $msg\n";

}

/**

* 每个进程启动

* @param $worker

*/

public function onWorkerStart($worker)

{

}

}

如果用tp5给出的方法,我们需要在其他的控制器中实例化该控制器类,而在我用到该框架时没有看出该用法,误以为实现功能直接在该控制器中调用即可,当我利用到tcp链接时出现了两个链接的同时调用,在tp5文档中的这个方法,我无法同时实例化两个链接,于是我放弃了该方法,想要研究的同学可以继续研究一下,下面给出我的方法。

workertest.php代码如下:

namespace app\push\controller;

use Workerman\Worker;

use Workerman\Lib\Timer;

use Workerman\Connection\AsyncTcpConnection;

class WorkerTest

{

private $connections;

private $connection_to_ws;

public function index()

{

// $connections = array();

$socket = new Worker('websocket://0.0.0.0:2346');

// 设置transport开启ssl,websocket+ssl即wss

// $socket->transport = 'ssl';

// 启动1个进程对外提供服务

$socket->count = 1;

//给这个进程设置一个array()

// 当有客户端连接时

$socket->onConnect = function($connection)

{

var_dump(count($this->connections));

$connection->send("lianjie");

$this->connections[$connection->id]=$connection;

};

// 当有客户端连接时

$socket->onMessage = function($connection,$data)

{

// var_dump($data);

// var_dump(json_decode($data));

$jdata = json_decode($data);

if(isset($jdata->tem))

{

foreach($this->connections as $con){

if(isset($con->endno)&&isset($jdata->endno)&&$con->endno==$jdata->endno){

$con->send($jdata->tem);

}

}

}

else

{

$connection->send("数据已接受");

$connection->endno=$jdata->endno;

$this->connections[$connection->id]=$connection;

}

};

// 当有客户端连接断开时

$socket->onClose = function($connection)

{

if(isset($connection->id))

{

// 连接断开时删除映射

unset($this->connections[$connection->id]);

}

};

$tcp = new Worker('tcp://0.0.0.0:8282');

$tcp->onMessage = function($connection, $data)

{

if(is_null($this->connection_to_ws))

{

var_dump('connect');

$this->connection_to_ws = new AsyncTcpConnection('ws://119.29.170.92:2346');

$this->connection_to_ws->connect();

}

$this->connection_to_ws->send($data);

// var_dump(count($this->connections));

//  foreach($this->connections as $con){

//      if($con->endno==json_decode($data)->endno){

//          $con->send(json_decode($data)->tem);

//      }

//  }

};

// 运行worker

Worker::runAll();

}

}

每当前端浏览器通过websoket上传给服务器对应终端号,workerman就会将本链接放入到队列,与此同时,我通过tcp获取到硬件的值,并经过AsyncTcpConnection这个workerman对象将由tcp端口获取的值转发给websoket端口,再由websoket进行遍历当前队列链接的前端浏览器,通过终端id查找并推送给对应的前端浏览器。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值