#pragma once #include "acl_cpp/acl_cpp_define.hpp" #include #include "acl_cpp/stdlib/string.hpp" #include "acl_cpp/stream/aio_handle.hpp" #include "acl_cpp/stream/aio_timer_callback.hpp" #include "acl_cpp/stream/aio_stream.hpp" namespace acl { class aio_ostream; /** * 延迟异步写数据类,基类为 aio_timer_callback (see aio_handle.hpp), * 所谓延迟异步写,就是把异步写流(aio_ostream)放在定时器中,将该异 * 步流的异步写操作解除绑定(即从 aio_handle 的监控中解除),当指定 * 时间到达后再启动异步写操作(在 timer_callback 回调中再重新将异步 * 流的异步写操作绑定),同时该定时器自动销毁(调用 destroy 方法), * 所以如果用户继承了 aio_timer_writer 类,且子类不是在堆上分配的, * 则必须重载 destroy方法,同时在子类的 destroy 中执行与资源释放的 * 相关操作,如果子类未重载 destroy,则当定时器结束后内部会自动调用 * 基类 aio_timer_writer 的 destroy--该类调用了 delete this,此时就 * 会导致非法内存释放操作) * */ class ACL_CPP_API aio_timer_writer : public aio_timer_callback { public: aio_timer_writer(); /** * 在 aio_istream 中调用此函数以释放类对象,子类应该实现该函数 */ virtual void destroy() { delete this; } protected: virtual ~aio_timer_writer(); /** * 延迟读数据时的回调函数,从 aio_timer_callback 类中继承而来 */ virtual void timer_callback(unsigned int id); private: friend class aio_ostream; aio_ostream* out_; int write_delayed_; acl::string buf_; }; /** * 异步写数据流类定义,该类只能在堆上被实例化,在析构时需要调用 close * 函数以释放该类对象 */ class ACL_CPP_API aio_ostream : virtual public aio_stream { public: /** * 构造函数 * @param handle {aio_handle*} 异步事件引擎句柄 */ aio_ostream(aio_handle* handle); /** * 添加异可写时的回调类对象指针,如果该回调类对象已经存在,则只是 * 使该对象处于打开可用状态 * @param callback {aio_callback*} 继承 aio_callback 的子类回调类对象, * 当异步流有数据时会先调用此回调类对象中的 write_callback 接口 */ void add_write_callback(aio_callback* callback); /** * 从写回调对象集合中删除 * @param callback {aio_callback*} 被删除的写回调对象, * 若该值为空,则删除所有的回调写对象 * @return {int} 返回被从回调对象集合中删除的回调对象的个数 */ int del_write_callback(aio_callback* callback = NULL); /** * 禁止回调对象类集合中的某个回调类对象,但并不从回调类对象 * 集合中删除,只是不被调用而已 * @param callback {aio_callback*} 被禁用的写回调对象, * 若该值为空,则禁用所有的写回调对象 * @return {int} 返回被从回调对象集合中禁用的回调对象的个数 */ int disable_write_callback(aio_callback* callback = NULL); /** * 启用所有的回调对象被调用 * @param callback {aio_callback*} 启用指定的写回调对象, * 如果该值为空,则启用所有的写回调对象 * @return {int} 返回被启用的写回调对象的个数 */ int enable_write_callback(aio_callback* callback = NULL); /** * 异步写规定字节数的数据,当完全写成功或出错或超时时会 * 调用用户注册的回调函数,在延迟异步写时,当在一个函数 * 内连续调用此过程时,每个延迟异步写操作会被加入延迟写 * 的队列中,以保证每个延迟异步写操作都可在各自的定时器 * 到达时被执行 * @param data {const void*} 数据地址 * @param len {int} 数据长度 * @param delay {int64} 如果该值 > 0 则采用延迟发送的模式(单位为微秒) * @param callback {aio_timer_writer*} 定时器到达时的回调函数类对象, */ #ifdef WIN32 void write(const void* data, int len, __int64 delay = 0, aio_timer_writer* callback = NULL); #else void write(const void* data, int len, long long int delay = 0, aio_timer_writer* callback = NULL); #endif /** * 格式化方式异步写数据,当完全写成功或出错或超时时会 * 调用用户注册的回调函数 * @param fmt {const char*} 格式字符串 */ void format(const char* fmt, ...) ACL_CPP_PRINTF(2, 3); /** * 格式化方式异步写数据,当完全写成功或出错或超时时会 * 调用用户注册的回调函数 * @param fmt {const char*} 格式字符串 * @param ap {va_list} 数据值列表 */ void vformat(const char* fmt, va_list ap); /** * 异步等待连接流可写,该函数设置异步流的写监听状态,当有可写时, * 回调函数被触发,由用户自己负责数据的读取 * @param timeout {int} 写超时时间(秒),当该值为 0 时,则没有写超时 */ void write_wait(int timeout = 0); /** * 禁止异步流的异步写状态,则将该异步流从异步引擎的监控 * 事件中移除,直到用户调用任何一个写操作时会自动打开异 * 步写状态(此时该流会重新被异步引擎监控) */ void disable_write(); protected: virtual ~aio_ostream(); /** * 释放动态类对象的虚函数 */ virtual void destroy(); /** * hook 写过程 */ void hook_write(); private: friend class aio_timer_writer; std::list timer_writers_; std::list write_callbacks_; bool write_hooked_; static int write_callback(ACL_ASTREAM*, void*); static int write_wakup(ACL_ASTREAM*, void*); }; } // namespace acl