安装使用swoole

swoole首页:https://www.swoole.com/

方法1:使用pecl安装

  pecl install swoolephp

  注意,php版本必须是7.0以及7.0以上的版本。html

 

方法2:编译源码安装

第一步:下载swoole的源码

  下载源码的时候要注意,swoole2.0及之后版本再也不支持PHP5.xmysql

git clone  https://github.com/swoole/swoole-src.gitgit

 

第二步:进入源码目录,执行phpize命令

 

第三步:配置php-config的路径

./configure --with-php-config=/usr/local/php/bin/php-configgithub

 

第四步:将php.ini中被禁止的proc_open、proc_get_status、system、exec、shell_exec这几个函数从其中删除,由于在make时要用到这几个函数。

 

第五步:make

  注意:若是下载的swoole 2.x,而php版本低于7.0,在这一步会失败,请下载正确源码版本web

 

第六步:make test

 

第七步检测是否安装成功

php  -msql

 

使用swoole搭建一个http服务器

<?php

    $http = new swoole_http_server("0.0.0.0",9501);
    $http->on("request", function($request, $response){
        print_r($request);
        print_r($response);
    });

    $http->start();

  向服务器发起请求:shell

curl 123.123.123.123:9501/aaa/bbb/index.html

  运行结果:数据库

Swoole\Http\Request Object
(
    [fd] => 1
    [header] => Array
        (
            [user-agent] => curl/7.29.0
            [host] => 123.123.123.123:9501
            [accept] => */*
        )

    [server] => Array
        (
            [request_method] => GET
            [request_uri] => /aaa/bbb/index.html
            [path_info] => /aaa/bbb/index.html
            [request_time] => 1531286716
            [request_time_float] => 1531286716.3838
            [server_port] => 9501
            [remote_port] => 46576
            [remote_addr] => 123.123.123.123
            [master_time] => 1531286716
            [server_protocol] => HTTP/1.1
            [server_software] => swoole-http-server
        )

    [request] =>
    [cookie] =>
    [get] =>
    [files] =>
    [post] =>
    [tmpfiles] =>
)
Swoole\Http\Response Object
(
    [fd] => 1
    [header] =>
    [cookie] =>
    [trailer] =>
)

  能够观察一下上面Swoole\Http\Request和Swoole\Http\Response的结构(属性)。apache

 

使用http服务器返回数据

<?php
    $http = new swoole_http_server("0.0.0.0",9501);
    $http->on("request", function($request, $response){
        $response->header("Content-Type","text/html;chatset=utf-8");
        $response->end("hello world\n");
    });

    $http->start();

  运行脚本,而后请求服务器:

[root@centos ~]# curl xxx.xxx.xxx.xxx:9501/aaa/bbb/index.html
hello world

  

建立WebSocket服务器

  显示index.html的websocket客户端

<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<title>Document</title>
</head>
<body></body>
<script>
	let ws = new WebSocket("ws://ganlixin.cn:9501");

	ws.onopen = function(){
		console.log("connect success");
	}

	ws.onclose = function(){
		console.log("closed")
	}

	ws.onmessage = function(res){
		console.log(res.data)
	}
</script>
</html>

  

  websocket服务器:

<?php
    $ws = new swoole_websocket_server("0.0.0.0",9501);

    //打开链接
    $ws->on("open", function($serv, $request){
        echo "connected \n";
        //向客户端发送信息
        $serv->push($request->fd, "hello world\n");
    });

    //接收到数据
    $ws->on("message", function($serv, $request){
        echo "received data\n";
        $serv->push($request->fd, "get message from client : {$request->data}");
    });

    //关闭链接
    $ws->on("close", function(){
        echo "closed\n";
    });

    $ws->start();

  

  运行index.html,看控制台,有信息输出:

  

定时器

  面向过程的形式

<?php
    //循环定时器
    //int swoole_timer_tick(int $ms, callable $callback, mixed $user_param);
    $timer_id = swoole_timer_tick(5000, function($timer_id) {
        echo "current time " . date("Y-m-d H:i:s",time()) . "\n";
    });
    echo $timer_id;

    //清除定时器
    //swoole_timer_clear($timer_id);

  面向对象的形式

<?php
    //循环定时器
    $server = new swoole_server("0.0.0.0", 9501);
    $timer_id = $server->tick(1000,function($timer_id){
        echo "timer_id : " . $timer_id . " current time " . date("Y-m-d H:i:s",time()) . "\n";
    });

    //注意,没有$swoole_server->clear();
    swoole_timer_clear($timer_id);

  使用定时器,通常都是使用面向过程的形式,由于那样的话,能够不用指定ip以及port。

 

异步TCP服务器

<?php
    //建立TCP服务器,默认是同步阻塞工做模式
    $server = new swoole_server("0.0.0.0",9501);

    //设置异步进程数量
    $server->set( [ "task_worker_num" => 4 ]);

    //接收到请求时
    $server->on("receive", function($serv, $data, $fd){
        $task_id = $serv->task($data); //生成异步任务id
        echo "task id is " . $task_id . "\n";
    });

    //处理异步任务
    $server->on("task", function($serv, $task_id, $from_id, $data){
        echo "执行异步任务 task_id : $task_id \n";
        $serv->finish("data -> OK");
    });

    $server->on("finish", function($serv, $task_id, $data){
        echo "执行任务{$task_id}完成\n";
    });

    $server->start();
?>

  屡次访问服务器绑定的端口,会看到服务器控制台输出内容以下

[root@centos index]# php index.php
task id is 0
执行异步任务 task_id : 0
执行任务0完成
task id is 1
执行异步任务 task_id : 1
执行任务1完成
task id is 2
执行异步任务 task_id : 2
执行任务2完成
task id is 3
执行异步任务 task_id : 3
执行任务3完成

  

TCP客户端

<?php
    //建立tcp客户端
    //swoole_client->__construct(int $sock_type, int $is_sync = SWOOLE_SOCK_SYNC, string $key);
    $client = new swoole_client(SWOOLE_SOCK_TCP);

    //链接服务器
    //bool $swoole_client->connect(string $host, int $port, float $timeout = 0.5, int $flag = 0)
    $client->connect("123.123.123.123", 9501, 5) or die("链接失败");

    //发送数据request
    //int $swoole_client->send(string $data);
    $client->send("hello world");

    //接受服务器的response
    //string $swoole_client->recv(int $size = 65535, int $flags = 0);
    //第二个参数表示是否等到全部数据都到达以后再返回
    $res = $client->recv(65535);
    echo $res;

  

异步TCP客户端

<?php
    //建立一个异步tcp客户端
    $client = new swoole_client(SWOOLE_SOCK_TCP,SWOOLE_SOCK_ASYNC);

    //只有tcp客户端是异步的时候,能够绑定事件
    $client->on("connect", function($cli){
        echo "connect\n";
    });

    $client->on("receive", function($cli, $data){
        echo "data : $data \n";
    });

    $client->on("err", function($cli){
        echo "error\n";
    });

    $client->on("close", function($cli){
        echo "closed\n";
    });

    $client->connect("ganlixin.cn",80,10);
?>

  

 

建立进程执行指定任务

<?php
    //建立一个函数,包含以后建立的进程要干的任务
    function doJob(swoole_process $worker){
/*
        print_r($worker);
        Swoole\Process Object
            (
                [pipe] => 5
                [callback] => doJob
                [msgQueueId] =>
                [msgQueueKey] =>
                [pid] => 22777
                [id] =>
            )
*/
        //执行外部程序
        //bool swoole_process->exec(string $execfile, array $args)
        $worker->exec('/usr/local/apache/bin/apachectl',array("start"));
    }

    //建立进程,将要作的事情(上面指定的函数)传递给构造方法
    $process_1 = new swoole_process("doJob");
    $pid = $process_1->start();
    //echo $pid,"\n";

    $process_2 = new swoole_process("doJob");
    $pid = $process_2->start();
    //echo $pid,"\n";

    //等待进程结束
    swoole_process::wait();

  

 指定进程事件

<?php
    //进程池
    $process_pool = array();

    //进程数量
    $process_num = 4;

    //建立任务
    function doJob(swoole_process $process){
        //向管道中写入本身的pid
        $process->write("my pid is $process->pid");
        echo "pid : $process->pid 已写入信息\n";
        $process->callback;
    }

    for ( $i = 0; $i < $process_num; $i++ ) {
        $process = new swoole_process("doJob");
        $pid = $process->start();
        $process_pool[] = $process; //存入进程池
    }

    //向每个子进程指定须要执行的操做
    foreach($process_pool as $process){
        swoole_event_add($process->pipe, function($pipe) use ($process){
            $data = $process->read();
            echo "接收到数据 $data\n";
        });
    }

  运行:

pid : 25176 已写入信息
接收到数据 my pid is 25176
pid : 25177 已写入信息
接收到数据 my pid is 25177
pid : 25175 已写入信息
接收到数据 my pid is 25175
pid : 25174 已写入信息
接收到数据 my pid is 25174

  

进程队列

<?php
    //进程池
    $process_pool = array();

    //进程数
    $process_num = 4;

    //子进程要执行的任务
    function doJob(swoole_process $process){
        $recv_data = $process->pop();//默认8192字节
        echo "从主进程收到数据 $recv_data\n";
        sleep(1);
        $process->exit(0);//进程退出
    }

    for ( $i = 0; $i < $process_num; $i++ ) {
        $process = new swoole_process("doJob",false,false);
        //加入进程池
        $process_pool[] = $process;
        $process->useQueue();//开启进程队列
        $process->start();
    }

    //主进程执行
    foreach($process_pool as $process){
        //主进程向子进程发送数据
        $process->push("hello, I'm your father, your pid is $process->pid\n");
    }

    //等待子进程结束
    for ( $i = 0; $i < $process_num; $i++ ) {
        //array swoole_process::wait(bool $blocking = true);
        $process = swoole_process::wait();
        echo "子进程{$process["pid"]}退出\n";
    }

    //销毁进程池
    unset($process_pool);
?>

  

  

信号触发

<?php
    swoole_process::signal(SIGALRM,function(){
        //每次收到SIGALRM信号就执行一次
        echo "one";

        /*
        知足某种条件是,取消定时触发信号
        if ( condition ){
            swoole_process::alarm(-1);
        }
        */
    });

    //单位是微秒 1秒=1000000微秒
    //一秒触发一次alarm信号
    swoole_process::alarm(1000000);

    //上面的代码等同于
    //swoole_timer_tick(1000,function(){
    //    echo "one";
    //});

  

互斥锁

<?php
    //建立互斥锁
    $mutex = new swoole_lock(SWOOLE_MUTEX);

    $mutex->lock();
    echo "父进程加锁\n";

    if ( pcntl_fork() > 0){
        //主进程执行
        sleep(2);
        $mutex->unlock();
    } else {
        //子进程执行
        echo "子进程等待父进程解锁\n";
        $mutex->lock();
        echo "子进程加锁\n";
        $mutex->unlock();
        exit("子进程退出\n");
    }

    echo "主进程释放锁\n";
    unset($mutex);
?>

  运行结果:

[root@centos index]# php mutex.php
父进程加锁
子进程等待父进程解锁
主进程释放锁



子进程加锁
子进程退出

  

DNS查询

  经过传入域名,返回ip。

<?php
    swoole_async_dns_lookup("www.swoole.com", function($host, $ip){
        echo "{$host} : {$ip}\n";
    });
?>

  

异步读取文件

  在PHP中读取文件,若是是大文件,可能须要的时间就长一点,那么就要修改php配置参数。

<?php

    //采起分段读的方式
    //bool swoole_async_read(string $filename, mixed $callback, int $size = 8192, int $offset = 0);
    swoole_async_read("./mutex.php", function($filename,$content){
        echo $content;
    },10);

  

  

异步读取文件

  分段写入

<?php

    $content = file_get_contents("beego.tar");
    
    //分段写入
    //swoole_async_write(string $filename, string $content, int $offset = -1, mixed $callback = NULL);
    //offset置为-1时,表示追加方式
    swoole_async_write("data.tar", $content,1);

  

 异步mysql

<?php
    $mysql = new swoole_mysql();

    $config = array(
        "host"      => "localhost",
        "user"      => "root",
        "password"  => "123456",
        "database"  => "test",
        "charset"   => "utf8"
    );

    $mysql->connect($config, function(swoole_mysql $db, $result){
        if ($result === FALSE) {
            echo "$result->connect_errno , $result->connect_error\n";
            exit();
        }
        echo "数据库链接成功\n";

        $sql = "select * from demo";

        $db->query($sql, function($link, $result){
            if ($result === FALSE) {
                echo "执行SQL失败\n";
                exit();
            }
            print_r($result);
            //$result保存着结果集
        });
    });
?>

  运行:

[root@centos index]# php async_mysql.php
数据库链接成功
Array
(
    [0] => Array
        (
            [id] => 1
            [name] => aaa
            [age] => 10
        )

    [1] => Array
        (
            [id] => 2
            [name] => bbb
            [age] => 20
        )

)
相关文章
相关标签/搜索