swoole使用纯c语言编写,提供了php语言的异步多线程服务器,异步 tcp/udp 网络客户端,异步 mysql,异步 redis,数据库连接池,asynctask,消息队列,毫秒定时器,异步文件读写,异步dns查询。 swoole内置了http/websocket服务器端/客户端、http2.0服务器端,完美支持php语言。本文讲解linux下搭建swoole实现php消息推送的方法。
swoole简介
swoole是一个面向生产环境的 php 异步网络通信引擎,使 php 开发人员可以编写高性能的异步并发 tcp、udp、unix socket、http,websocket 服务。swoole 可以广泛应用于互联网、移动通信、企业软件、云计算、网络游戏、物联网(iot)、车联网、智能家居等领域。 使用 php + swoole 作为网络通信框架,可以使企业 it 研发团队的效率大大提升。
swoole不是一个像zend framework、cakephp、yii、symfony、thinkphp等一样的框架,也不是一个像wordpress、drupal、discuz、uchome等开源产品看齐的项目。 swoole的目标是向java框架、ruby on rails、python djangopylons等一流框架发起挑战。
swoole虽然是标准的php扩展,实际上与普通的扩展不同。普通的扩展只是提供一个库函数。而swoole扩展在运行后会接管php的控制权,进入事件循环。当io事件发生后底层会自动回调指定的php函数。
包含以下几个特色功能:
1、 类似orm的数据查询,提供sql封装器,让mysql的sql与php的array,会话,cache无缝结合。
2、app mvc分层结构,有效的程序结构分层,提高程序的可维护性和扩展性,实现低耦合,基于接口开发。
3、集成大量,实用的功能,比如方便的数据库操作,模板操作,缓存操作,系统配置,表单处理,分页,数据调用,字典操作,上传处理,内容编辑,调试等。
4、模板-数据反射系统,可以直接在模板中调用数据,提供很多标签,可以无需修改程序,只修改模板,即可实现网站各类更新维护工作。
另外的几个功能
1、swoole包含了大量类,提供众多的功能扩展,基本上web开发能够用到的功能类,大部分都可以在swoole框架中找到。
2、swoole拥有插件系统,fckeditor、adodb、pscws中文分词、中文全文索引系统、最新的key-value数据库思想,tokyotyrant,可以无限扩展框架的功能。
安装swoole服务
1.下载swoole源码,下载地址: https://github.com/swoole/swoole-src/releases
2.将tar包上传到/usr/local/src/swoole下面,这里面存储安装源文件。
3.解压源文件,tar –xvfswoole.tar
4.进入到刚解压的目录下输入命令phpize
5.注意:phpize是php-devel中的东西,它可以给php动态添加扩展,所以,请确保你的机器上安装了php-devel软件包。
6.接着,依次输入如下命令:./configure,执行编译检测make,编译swoole
7.缺少pcre和pcre-devel相关软件包,安装上即可。
8.sudo make install
9.php.ini一般在etc下面 编译安装成功后,我们还需要修改php.ini,在其中加入swoole.so扩展:
10.extension=swoole.so
11.输入php -m中能查看到swoole说明扩展安装成功。
安装apache启动swoole服务
(1)测试是否安装正确启动swoole php服务
(2)在apache中添加server.php文件,和client.php文件。一般目录在var/www/html下面
(3)打开两个终端 在这两个文件所在的目录下面打开这两个文件,可以实现socket长连接服务。
安装lsof:yum -y install lsof
server端php程序
class websocketservice {
private static $instance = null;
public $tagindex = 0;
public $server = null;
public $office_cache = null;
public $timer_arr = []; //该变量保存所有的定时器任务id,每一个客户端可以通过$timer_arr[客户端id]
得到该客户端建立的所有定时器
public $conf = [
'host' => '0.0.0.0',
'port' => 9999, //服务使用端口
'worker_num' => 2, //启动worker进程数
'task_worker_num' => 8, //启动task进程数
'is_daemonize' => 0, //是否后台运行:0-否,1-是
'log_file' =>'/tmp/swoole_websocket_server.log', //日志文件路径
'abc' =>0,
];
public function gettagindex() {
return $this->conf['abc'];
}
public function setsettagindex($index) {
$this->conf['abc'] = 100;
echo $this->gettagindex();
}
public function __construct($config = []) {
$this->office_cache = array('1' => '1', '3'=>'2');
$this->conf = array_merge($this->conf, (array)$config);
}
//静态方法,返回此类唯一实例
public static function getinstance(){
if(empty(self::$instance)){
echo "instance \n";
self::$instance=new websocketservice();
}
return self::$instance;
}
/**
* 服务器端运行websocket的入口
*/
public function run() {
try {
$this->server = new \swoole_websocket_server($this->conf['host'],$this->conf['port']);
$this->server->set(
[
'worker_num' => $this->conf['worker_num'],
'task_worker_num' =>$this->conf['task_worker_num'],
'daemonize' => $this->conf['is_daemonize'],
'log_file' => $this->conf['log_file'],
]
);
//注册方法列表
$this->server->on('open', [$this, 'open']);
$this->server->on('message', [$this, 'message']);
$this->server->on('task', [$this, 'task']);
$this->server->on('finish', [$this, 'finish']);
$this->server->on('close', [$this, 'close']);
$this->server->start();
} catch (\exception $e) {
var_dump($e->getcode() . ':' . $e->getmessage());
}
}
/**
* 建立socket链接时,执行方法
* @param $server
* @param $request
*/
public function open($server, $request) {
$data = [
'client_id' => $request->fd,
'request' => $request
];
echo 'open<<'.$data['client_id'];
$this->doopen($data);
}
/**
* 发送消息时,执行方法
* @param $server
* @param $frame
*/
public function message($server, $frame) {
echo "get_mesage\n";
$data = [
'client_id' => $frame->fd,
'data' => $frame->data,
'frame' => $frame,
];
$this->domessage($data);
}
/**
* 执行具体任务
* @param $server
* @param $task_id
* @param $from_id
* @param $data
*/
public function task($server, $task_id, $from_id, $data) {
$data['task_id'] = $task_id;
$data['from_id'] = $from_id;
$this->dotask($data);
}
/**
* 任务结果处理
* @param $server 服务器对象
* @param $taskid 任务进程id
* @param $data
*/
public function finish($server, $taskid, $data) {
$data['task_id'] = $taskid;
$this->dofinish($data);
}
/**
* 关闭链接
* @param $server 服务器对象
* @param $client_id 客户端唯一标识
*/
public function close($server, $client_id) {
$data = [
'client_id' => $client_id
];
$this->doclose($data);
}
/**
* 客户端成功连接到服务器时,会触发该方法
* 子类根据需求重写该方法
* @param $data
* [
* 'client_id', //客户端唯一标识
* 'data', //一些请求数据
* 'frame', //swoole的数据
* ];
*/
public function doopen($data) {
echo "建立连接成功";
}
/**
* 得到swoole websocket server
*
* @return null|\swoole_websocket_server
*/
public function getserver()
{
return $this->server;
}
/**
* 收到客户端发来的消息,会触发该方法
* 子类根据需求重写该方法
* @param $data
* [
* 'client_id', //客户端唯一标识
* 'data', //客户端发送过来的消息(数据)
* 'frame', //swoole的数据
* ];
*/
public function domessage($data) {
$info = json_decode($data['data'],true);
$officeid = $info['officeid'];
$client_id = $data['client_id'];
echo"<<<<".$officeid.">>>>".$client_id."\r\n";
if ($officeid == 1) {
$this->sendmsgtoclient('2',array('msg' =>'我是1号场地发送给你数据'));
} else {
$this->sendmsgtoclient('1',array('msg' =>'我是3号场地发送给你数据'));
}
}
/**
* 具体的工作任务。需要通过 $this->dojob()来触发
* 子类根据需求重写该方法
* @param $data
* [
* 'client_id', //客户端唯一标识
* ];
*/
public function dotask($data) {
}
/**
* 工作的结果处理。
* 子类根据需求重写该方法
* @param $data
* [
* 'client_id', //客户端唯一标识
* ];
*/
public function dofinish($data) {
}
/**
* 客户端断开时会自动触发该方法
* 子类根据需求重写该方法
* @param $data
* [
* 'client_id', //客户端唯一标识
* ];
*/
public function doclose($data) {
}
/**
* 发送任务
* @param $data 必须是数组,且要有$data['client_id']
*/
public function dojob($data) {
$this->server->task($data);
}
public function finishjob($data) {
$this->server->finish($data);
}
/**
* 发送消息到客户端
* @param $client_id
* @param $msg
*/
public function sendmsgtoclient($client_id, $msg) {
echo "发送信息给客户端:{$client_id} | {$msg['action']['name']} | " . date('y-m-dh:i:s') . "\r\n";
$this->server->push($client_id, json_encode($msg));
}
/**
* 关闭客户端链接
* @param $client_id
*/
public function closeclient($client_id) {
$this->server->close($client_id);
}
/**
* 添加定时器
* @param $client_id
* @param $tick_time
*/
public function addtimer($client_id, $tick_time) {
//暂未实现,先使用swoole原生写法
}
/**
* 清除定时器
* @param $client_id
* @param $arr
*/
public function cleartimer($client_id, &$arr) {
if (is_array($arr)) {
foreach ($arr[$client_id] as $val) {
if (is_array($val)) {
foreach ($val as $v) {
swoole_timer_clear($v);
}
} else {
swoole_timer_clear($val);
}
}
}
unset($arr[$client_id]);
}
}html前端javascript程序
$(document).ready(function(){
varwsurl = "ws://182.92.101.206:9999/";
console.log(wsurl);
varwebsocket = new websocket(wsurl);
websocket.onopen= function(evt){
console.log('server: 打开websocket连接');
};
websocket.onclose= function(evt){
console.log('server: 关闭websocket连接');
};
websocket.onmessage= function(evt){
varres = json.parse(evt.data);
console.log('server: 收到消息(来自:'+res+'请求)');
console.log(res);
};
websocket.οnerrοr= function(evt){
console.log('server: 出现错误');
console.log(evt.data);
}
functiondosend(msg){
console.log('client:发送消息 ' + msg);
websocket.send(msg);
}
});启动start.php文件
启动start.php文件,cd到指定目录下,然后php-cli运行php start.php。这种方式是没有启用线程保护的关掉后程序结束线程还在占用。
require'src/websocketswoole/websocketservice.php'; // require './websocketservice.php'; session_start(); // $demoservice = new\websocketswoole\websocketservice(); $demoservice =\websocketswoole\websocketservice::getinstance(); $demoservice->run();
总结
到此这篇关于linux下搭建swoole实现php消息推送的方法的文章就介绍到这了,更多相关swoole实现php消息推送内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
发表评论