当前位置: 首页 > news >正文

【FileZilla】 从事件类型到消息类型的函数形参类型转化

本篇其实是前篇【Filezilla】 dispatch函数重载的例子-CSDN博客的一个补充,其中涉及到【FileZilla】事件调用机制代码解析-CSDN博客中的事件分发机制时钩子函数的参数传递怎么实现的。跟【FileZilla】sftp协议的数据传输上传和下载-CSDN博客同样,用事件是 CSftpEvent 即 send_event<T> () 或者说pending_events.emplace_back(handler, evt, deletable ) 调用时的T和evt,开始。

我们的问题是事件处理函数 CSftpControlSocket::OnSftpEvent(sftp_message const& message) 中的是怎么得到的消息message的?

// In event_loop.hpp
typedef std::deque<std::tuple<event_handler*, event_base*, bool>> Events;// Events::value_type ev{};// In event.h
typedef fz::simple_event<sftp_event_type, sftp_message> CSftpEvent;// In sftpcontrolsocket.cpp
void CSftpControlSocket::operator()(fz::event_base const& ev)
{if (fz::dispatch<fz::process_event, CSftpEvent, CSftpListEvent, SftpRateAvailableEvent>(ev, this,&CSftpControlSocket::OnProcessEvent,&CSftpControlSocket::OnSftpEvent,&CSftpControlSocket::OnSftpListEvent,&CSftpControlSocket::OnQuotaRequest)) {return;}CControlSocket::operator()(ev);
}// In sftpcontrolsocket.cpp
void CSftpControlSocket::OnSftpEvent(sftp_message const& message)
{......switch (message.type){......case sftpEvent::Recv:......break;case sftpEvent::Send:......break;case sftpEvent::Transfer:......break;
......case sftpEvent::io_nextbuf:......break;case sftpEvent::io_open:......break;case sftpEvent::io_size:......break;case sftpEvent::io_finalize:......break;default:......}......
}

类似【FileZilla】事件调用机制代码解析-CSDN博客,在SftpInputParser::OnData()函数中 event_ = std::make_unique<CSftpEvent>() 随后调用事件入队列 owner_.send_event(event_.release()) ,以下式子逐步等价:

  • owner_.send_event(event_.release())
  • event_loop.send_event(this, event_.release(), true),this 指针指向子类CSftpControlSocket的实例
  • 实际调用:pending_events_.emplace_back(handler,evt,deletable),handler就是this指针、evt就是CSftpEvent,event_.release()就是从智能指针中释放裸指针,交出所有权

其中owner_是CSftpControlSocket类型,CSftpControlSocket继承自 CControlSocket ,它继承自event_handler,公有父类实现了 send_event() 方法。于是,当调用entry()现成调process_event()时,语句 (*std::get<0>(ev))(*std::get<1>(ev)) 时,相当于以下式子逐步“等价”:

  • (*std::get<0>(ev))(*std::get<1>(ev))
  •  (*handler)(evt)
  • handler->operator()(evt) 
  • CSftpControlSocket的实例->operator()( CSftpEvent * ev ),operator实际接收了event_base类型
  • CSftpControlSocket::operator()(fz::event_base const& ev)

于是从【FileZilla】sftp协议的数据传输上传和下载-CSDN博客中,我们知道事件分发后会调用CSftpControlSocket::OnSftpEvent(sftp_message const& message)。这时我们发现根据【Filezilla】 dispatch函数重载的例子-CSDN博客中dispatch重载的代码handler函数传入的参数是CSftpEvent * ev,如何变成了一个 sftp_message const& message ?

注意到CSftpEvent是一个分支上fz::simple_event模版类(在第一段代码里),而simple_event的解析如下。关键是因为 CSftpEvent ::v_ 是一个 std::tuple<sftp_message>fz::apply() 会自动解包为sftp_message。

template<typename UniqueType, typename...Values>
class simple_event final : public event_base
{
public:typedef UniqueType unique_type;typedef std::tuple<Values...> tuple_type;using event_base::event_base;template<typename First_Value, typename...Remaining_Values>explicit simple_event(First_Value&& value, Remaining_Values&& ...values): v_(std::forward<First_Value>(value), std::forward<Remaining_Values>(values)...){}/// \brief Returns a unique id for the type such that can be used directly in derived_type.inline static size_t type() {// Exporting templates from DLLs is problematic to say the least. It breaks// ODR, so we use this trick that goes over the type name.static size_t const v = get_unique_type_id(typeid(UniqueType*));return v;}/// \brief Simply returns \ref type()virtual size_t derived_type() const override {return type();}/** \brief The event value, gets built from the arguments passed in the constructor.** You don't need to access this member directly if you use the \ref dispatch mechanism.*/mutable tuple_type v_;
};

这是一个泛型事件类,它:

  • 继承自 event_base(这是 FileZilla 所有事件的基类);

  • 用模板参数构造出不同的事件类型;

  • std::tuple<Values...> 存储事件内容;

  • UniqueType 生成类型唯一标识。

你可以理解为:这是 C++ 中的 type-safe 事件封装器。

好啦,这就解释清楚了消息message是怎么来的,这对于sftp协议也是很重要的一点。在处理sftp事件时,FileZilla和fzsftp通信时利用了下面的函数:

static int fznotify(sftpEventTypes type)
{fprintf(stdout, "%c", (int)type + '0');fflush(stdout);return 0;
}
static int fznotify1(sftpEventTypes type, int data)
{fprintf(stdout, "%c%d\n", (int)type + '0', data);fflush(stdout);return 0;
}

其中的sftpEventType就是message.type的类型啦

enum class sftpEvent {Unknown = -1,Reply = 0,Done,Error,Verbose,Info,Status,Recv,Send,Listentry,AskHostkey,AskHostkeyChanged,AskHostkeyBetteralg,AskPassword,Transfer,RequestPreamble,RequestInstruction,UsedQuotaRecv,UsedQuotaSend,KexAlgorithm,KexHash,KexCurve,CipherClientToServer,CipherServerToClient,MacClientToServer,MacServerToClient,Hostkey,io_size,io_open,io_nextbuf,io_finalize,count
};

相关文章:

  • 使用Beyond Compare显示有差异点进去又没差异 问题解决
  • 提高成功率!课题中的立项依据深度写作
  • 计算机视觉----基于锚点的车道线检测、从Line-CNN到CLRNet到CLRKDNet 本文所提算法Line-CNN 后续会更新以下全部算法
  • 养生:健康生活的核心策略
  • 蓝桥杯11届国B 约数
  • 道通龙鱼系列-混合翼无人机:垂直起降+长时续航
  • HGDB企业版迁移到HGDB安全版
  • 游戏引擎学习第280天:精简化的流式实体sim
  • pg_dump“: CreateProcess error=2, 系统找不到指定的文件
  • Hugging Face 中 LeRobot 使用的入门指南
  • ultalytics代码中模型接收多层输入的处理
  • 《山东欧曼谛:美业梦想的启航港》
  • 面试 Linux 运维相关问题
  • 微信小程序van-dialog确认验证失败时阻止对话框的关闭
  • 嵌软面试每日一阅----FreeRTOS
  • CertiK荣获以太坊基金会两项资助,领跑zkEVM形式化验证
  • 【专栏启动】开篇:为什么是 Django + Vue3?测试平台的技术选型与架构蓝图
  • 基于微信小程序的在线聊天功能实现:WebSocket通信实战
  • C++从入门到实战(十五)String(上)介绍STL与String的关系,为什么有string类,String有什么用
  • 【深度学习之四】知识蒸馏综述提炼
  • 圆桌丨新能源车超充技术元年,专家呼吁重视电网承载能力可能面临的结构性挑战
  • 国税总局上海市税务局回应刘晓庆被举报涉嫌偷漏税:正依法依规办理
  • 夜读丨读《汉书》一得
  • 三亚通报救护车省外拉警报器开道旅游:违规违法,责令公司停业整顿
  • “远践”项目启动公益生态圈,上海青少年公益力量蓬勃生长
  • 国产水陆两栖大飞机AG600批产首架机完成总装下线