• php libevent 详解与使用


    libevent是一个基于事件驱动的高性能网络库。支持多种 I/O 多路复用技术, epoll、 poll、 dev/poll、 select 和 kqueue 等;支持 I/O,定时器和信号等事件;注册事件优先级。
     
    常量如下:
    常量名含义
    1 EV_TIMEOUT 超过时间后事件成为激活状态
    2 EV_READ FD就绪,可以读取的时候 ,事件成为激活状态
    4 EV_WRITE FD就绪,可以写入的时候 ,事件成为激活状态
    8 EV_SIGNAL 用于实现信号检测
    16 EV_PERSIST 表示事件是持久的
    32 EV_ET 表示底层是否支持边沿触发事件
    1 EVLOOP_ONCE 如果设置了EVLOOP_ONCE,循环将等待某些事件成为激活的,执行激活的事件直到没有更多的事件可以执行,然会返回。
    2 EVLOOP_NONBLOCK 如果设置了EVLOOP_NONBLOCK,循环不会等待事件被触发:循环将仅仅检测是否有事件已经就绪,可以立即触发,如果有,则执行事件的回调。
     
    函数如下:
    event_base_free() 释放资源,这不能销毁绑定事件
    event_base_loop() 处理事件,根据指定的base来处理事件循环
    event_base_loopbreak() 立即取消事件循环,行为各break语句相同
    event_base_loopexit() 在指定的时间后退出循环
    event_base_new() 创建并且初始事件
    event_base_priority_init() 设定事件的优先级
    event_base_set() 关联事件到事件base
    event_buffer_base_set() 关联缓存的事件到event_base
    event_buffer_disable() 禁用一个缓存的事件
    event_buffer_enable() 启用一个指定的缓存的事件
    event_buffer_fd_set() 改变一个缓存的文件系统描述
    event_buffer_free() 释放缓存事件
    event_buffer_new() 建立一个新的缓存事件
    event_buffer_priority_set() 缓存事件的优先级设定
    event_buffer_read() 读取缓存事件中的数据
    event_buffer_set_callback() 给缓存的事件设置或重置回调函数
    event_buffer_timeout_set() 给一个缓存的事件设定超时的读写时间
    event_buffer_watermark_set 设置读写事件的水印标记
    event_buffer_write() 向缓存事件中写入数据
    event_add() 向指定的设置中添加一个执行事件
    event_del() 从设置的事件中移除事件
    event_free() 清空事件句柄
    event_new() 创建一个新的事件
    event_set() 准备想要在event_add中添加事件
    
    例1如下:
    <?php
    //创建和初始化新的事件库
    $base = event_base_new();
    //创建并返回一个新的事件资源
    $event = event_new();
    //准备一个事件
    event_set($event, 0, EV_TIMEOUT, function() {
        echo "run...";
    });
    //关联事件到事件库
    event_base_set($event, $base);
    //添加事件
    event_add($event, 5000000);
    //根据指定的事件库来处理事件循环
    event_base_loop($base);
    
    上述代码在等待了5秒后打印出run...后程序结束。
    我们把上述代码修改一下:
    <?php
    //创建和初始化新的事件库
    $base = event_base_new();
    //创建并返回一个新的事件资源
    $event = event_new();
    //准备一个事件
    event_set($event, 0, EV_TIMEOUT | EV_PERSIST, function() {
        echo "run...
    ";
    });
    //关联事件到事件库
    event_base_set($event, $base);
    //添加事件
    event_add($event, 1000000);
    //根据指定的事件库来处理事件循环
    event_base_loop($base);
    
    上述代码会每隔1秒钟打印出run...,只是因为我们在event_set时使用了EV_PERSIST,表示该事件是一个持久的。
     
    例2如下:
    <?php
    function printContent($fd, $events, $args) {
        $data = trim(fgets($fd));
        echo "input: ", $data, "
    ";
    }
    
    //创建和初始化新的事件库
    $base = event_base_new();
    //创建并返回一个新的事件资源
    $event = event_new();
    //这里把标准输入流传入到函数printContent中
    $fd = STDIN;
    event_set($event, $fd, EV_PERSIST | EV_TIMEOUT, 'printContent', array($event, $base));
    //关联事件到事件库
    event_base_set($event, $base);
    //添加事件
    event_add($event, 1000000);
    //根据指定的事件库来处理事件循环
    event_base_loop($base);
    
    代码每隔1秒钟,会从标准输入中读取用户输入的内容,再打印出来。
     
    例3如下:
    <?php
    //创建套接字
    $socket = stream_socket_server('tcp://0.0.0.0:8888', $errno, $errstr);
    //设置资源流为非阻塞模式
    stream_set_blocking($socket, 0);
    
    //创建和初始化新的事件库
    $base = event_base_new();
    //创建并返回一个新的事件资源
    $event = event_new();
    //准备一个事件
    event_set($event, $socket, EV_PERSIST | EV_READ, 'accept', $base);
    //关联事件到事件库
    event_base_set($event, $base);
    //添加事件
    event_add($event);
    //根据指定的事件库来处理事件循环
    event_base_loop($base);
    
    function accept($socket, $events, $base) {
        //获取客户端连接后的socket
        $client = stream_socket_accept($socket);
        //设置该socket为非阻塞模式
        stream_set_blocking($client, 0);
        //创建一个新的缓存事件
        $buffer = event_buffer_new($client, 'read', NULL, 'error', $client);
        //关联缓存事件到事件库
        event_buffer_base_set($buffer, $base);
        //缓存事件超时时间
        event_buffer_timeout_set($buffer, 30, 30);
        //设置缓存事件读写的水印标记
        event_buffer_watermark_set($buffer, EV_READ, 0, 0xffffff);
        //设置缓存事件的优先级
        event_buffer_priority_set($buffer, 10);
        //启用缓存事件
        event_buffer_enable($buffer, EV_READ | EV_PERSIST);
        //这里好像必需把$buffer赋给一个全局的变量 
        $GLOBALS['_'] = $buffer;
    }
    
    function read($buffer, $client) {
        //从缓存事件中读取数据
        while ($read = event_buffer_read($buffer, 256)) {
        }
        //给连接的socket客户端写入数据
        fwrite($client, date('Y-m-d H:i:s'));
        error($buffer, '', $client);
    }
    
    function error($buffer, $error, $client) {
        //禁用一个缓存事件
        event_buffer_disable($buffer, EV_READ | EV_WRITE);
        //释放缓存事件
        event_buffer_free($buffer);
        //关闭客户端socket
        fclose($client);
    }
    
  • 相关阅读:
    使用vs code开发.net core2.2时OmniSharp.MSBuild.ProjectLoader无法解析"xxx"的解决方法
    An unexpected exception occurred while binding a dynamic operation 错误的一种情况
    node excel export包导致find函数被覆盖
    C#字典Dictionay多线程读是否是安全的
    mongodb数据分组按字符串split
    IIS做反向代理重定向到NodeJS服务器
    .net core 时间戳unix与本地时间互转、获取当前时间戳
    使用Postman调试asp.net core 控制器的action
    监听浏览器tab选项卡选中事件,点击浏览器tab标签页回调事件,浏览器tab切换监听事件
    vue、element-ui开发技巧
  • 原文地址:https://www.cnblogs.com/jkko123/p/6294591.html
Copyright © 2020-2023  润新知