4步实现C++插件化编程,轻松实现功能定制与扩展(2)

本文涉及的产品
性能测试 PTS,5000VUM额度
注册配置 MSE Nacos/ZooKeeper,118元/月
服务治理 MSE Sentinel/OpenSergo,Agent数量 不受限
简介: 本文是《4步实现C++插件化编程》的延伸,重点介绍了新增的插件“热拔插”功能。通过`inotify`接口监控指定路径下的文件变动,结合`epoll`实现非阻塞监听,动态加载或卸载插件。核心设计包括`SprDirWatch`工具类封装`inotify`,以及`PluginManager`管理插件生命周期。验证部分展示了插件加载与卸载的日志及模块状态,确保功能稳定可靠。优化过程中解决了动态链接库句柄泄露问题,强调了采纳用户建议的重要性。

4步实现C++插件化编程,轻松实现功能定制与扩展(2)

[TOC]

引言

  此文是对先前文章《4步实现C++插件化编程,轻松实现功能定制与扩展》 的延伸,重点记录在原版本基础上新增的插件热拔插功能。

  起因源于读者的一个评论,如下:
image.png


  看到这个问题时,当时的软件尚不具备“热拔插”功能。 但思考了一下,不支持“热拔插”的插件,应属于一种功能缺陷。于是乎,在原有的基础上增加了这一功能。这里,也很感谢这位读者提出这么好的问题。

  插件化编程的实现方案和代码细节已经在上一篇文章中记录了,本篇主要记录下新增的热拔插功能的实现细节。


注:文末提供本文源码获取方式。文章不定时更新,喜欢本公众号系列文章,可以星标公众号,避免遗漏干货文章。源码开源,如果对您有帮助,帮忙分享、点赞加收藏喔!

优化策略

  第一版软件仅在启动时加载插件。在此基础上,新增以下功能:

  • 在主程序运行过程中,若指定路径下新增插件库,程序将自动识别并加载。
  • 若在主程序运行中从指定路径移除或删除插件库,程序将自动卸载对应的已加载插件。

  要实现上述功能,需要对指定路径下的文件变动进行监控。在Linux环境中,可以利用inotify接口来达成这一目的。关于如何使用 inotify 实现实时文件监控的具体方法,可参考先前文章《使用inotify实现实时文件监控》

详细设计

  优化后的插件加载主要拆分为两个大类SprDirWatchPluginManager:
SprDirWatch 是一个工具类。专门用于封装 inotify 接口,以便于监控文件系统中的特定路径变化。
PluginManager 则是插件管理类。负责通过 SprDirWatch 捕获指定路径下文件的变化,并据此触发插件的自动“加载”或“卸载”操作。

  • SprDirWatch类定义
    ```C++
    class SprDirWatch
    {
    public:
    SprDirWatch();
    ~SprDirWatch();

    int GetInotifyFd() const { return mInotifyFd; }
    int AddDirWatch(const std::string& path, uint32_t mask);
    int RemoveDirWatch(int fd);

private:
int mInotifyFd;
std::set mWatchFds;
};

  `SprDirWatch` 的设计只是对 `inotify` 接口的一个简洁封装,其主要目的是为了更好地管理和控制 `inotify` 的监控资源。具体来说:    
① 封装 `inotify` 的使用复杂性,提供了一个更友好、更易于使用的接口。    
② 在`SprDirWatch`的生命周期结束(即析构)时,自动释放句柄(尽管没必要移除监控句柄,好的编程习惯应该是有始有终)。

* **PluginManager类定义**
```C++
class PluginManager
{
public:
    PluginManager();
    ~PluginManager();

    void Init();

private:
    void InitWatchDir();
    void LoadPlugin(const std::string& path);
    void UnloadPlugin(const std::string& path);
    void LoadAllPlugins();
    void UnloadAllPlugins();
    std::string GetDefaultLibraryPath();

private:
    SprContext mContext;
    SprDirWatch mDirWatch;
    std::string mDefaultLibPath;
    std::shared_ptr<PFile> mFilePtr;
    std::map<std::string, void*> mPluginHandles;
    std::map<int, SprObserver*> mPluginModules;
};

PluginManager 的设计则是用于管理所有插件的“加载”和“卸载”。即通过SprDirWatch监听指定路径“插件”的状态:

插件生成
① 当通过 SprDirWatch 监听到指定路径下有新的插件生成时,调用 LoadPlugin 方法加载新插件。
LoadPlugin 使用 dlopen 加载插件库,并保存库地址句柄。
③ 调用插件库的入口函数,启动插件模块。

插件卸载
① 当通过 SprDirWatch 监听到指定路径下的插件被删除时,调用 UnloadPlugin 方法卸载该插件。
UnloadPlugin 调用插件库的退出函数,停止插件模块。
③ 使用 dlclose 关闭插件库,释放资源。

  • 监听动态库,插件“热插拔”实现

    void PluginManager::InitWatchDir()
    {
      // Add a watch on the specified directory. The events to monitor include:
      // - IN_CLOSE_WRITE: Triggered when a file is closed after being written.
      // - IN_DELETE: Triggered when a file or directory is deleted.
      // - IN_MOVED_TO: Triggered when a file or directory is moved to the specified directory.
      // - IN_MOVED_FROM: Triggered when a file or directory is moved from the specified directory.
      // Note: IN_CREATE is not used because it triggers immediately when a file is created,
      // which may result in attempting to process the file before it is fully written and closed.
      mDirWatch.AddDirWatch(mDefaultLibPath.c_str(), IN_CLOSE_WRITE | IN_MOVED_TO | IN_MOVED_FROM | IN_DELETE);
      mFilePtr = std::make_shared<PFile>(mDirWatch.GetInotifyFd(), [&](int fd, void *arg) {
          const int size = 100;
          char buffer[size];
          ssize_t numRead = read(fd, buffer, size);
          if (numRead == -1) {
              SPR_LOGE("read %d failed! (%s)\n", fd, strerror(errno));
              return;
          }
    
          int offset = 0;
          while (offset < numRead) {
              struct inotify_event* pEvent = reinterpret_cast<struct inotify_event*>(&buffer[offset]);
              if (!pEvent) {
                  SPR_LOGE("pEvent is nullptr!\n");
                  return;
              }
    
              if (pEvent->len > 0) {
                  if (pEvent->mask & IN_CLOSE_WRITE || pEvent->mask & IN_MOVED_TO) {
                      SPR_LOGD("File %s is created\n", pEvent->name);
                      LoadPlugin(pEvent->name);
                  }
                  if (pEvent->mask & IN_DELETE || pEvent->mask & IN_MOVED_FROM) {
                      SPR_LOGD("File %s is deleted\n", pEvent->name);
                      UnloadPlugin(pEvent->name);
                  }
              }
              offset += sizeof(struct inotify_event) + pEvent->len;
          }
      });
    
      EpollEventHandler::GetInstance()->AddPoll(mFilePtr.get());
    }
    

      为了避免阻塞或轮询监听动态库路径,使用了 epoll 监听 inotify 的文件描述符,实现触发式监听。

验证

新增插件验证
① 移入插件库

$ mv libpluginonenet.so ../Lib/

② 日志打印确认

$ tail -f /tmp/sprlog/sprlog.log | egrep -i "PlugMgr|EntryOneNet"
10-30 21:08:13.277  19597 PlugMgr      D:   84 File libpluginonenet.so is created
10-30 21:08:13.300  19597 EntryOneNet  D:   58 Load plug-in OneNet modules
10-30 21:08:13.300  19597 PlugMgr      D:  141 Load plugin libpluginonenet.so success!

③ 模块状态确认

                                   Show All Message Queues
-----------------------------------------------------------------------------------------------
 HANDLE  QLSUM  QMUSED  QCUSED  BLOCK   MLLEN MMUSED MLAST  MTOTAL  NAME
-----------------------------------------------------------------------------------------------
      4     10       6       0  BLOCK    1025     51     1      32  /SprMdrQ_20231126
      5     10       1       0  BLOCK    1025     43     1       1  /TimerM_7lTva1nY
      6     10       1       0  BLOCK    1025     43     1       1  /PowerM_E0pil3lu
      7     10       1       0  BLOCK    1025     43     1       1  /OneDrv_BtJzE38A
      8     10       1       0  BLOCK    1025     43     1       1  /OneMgr_yXTdsXPW
      9     10       1       0  BLOCK    1025     51     1       1  /MQTT-OneJson01_y8M
     10     10       1       0  BLOCK    1025     47     1       1  /MQTT-DEV01_z5TzmqV
     11     10       1       0  BLOCK    1025     47     1       1  /PC_TEST_01_nOBnl0w
     12     10       1       0  BLOCK    1025     47     1       1  /PC_TEST_02_VWQQbIw
-----------------------------------------------------------------------------------------------
Press 'Q' to back

通过日志和模块状态,可确认插件OneNet加载成功,涉及到的模块运行正常。

移除插件验证
① 移除插件库

$ mv ../Lib/libpluginonenet.so .

② 日志打印确认

10-30 21:11:04.418  19597 PlugMgr      D:   88 File libpluginonenet.so is deleted
10-30 21:11:04.418  19597 EntryOneNet  D:   83 Unload plug-in OneNet modules
10-30 21:11:04.419  19597 PlugMgr      D:  170 Unload plugin libpluginonenet.so success!

③ 模块状态确认

                                   Show All Message Queues
-----------------------------------------------------------------------------------------------
 HANDLE  QLSUM  QMUSED  QCUSED  BLOCK   MLLEN MMUSED MLAST  MTOTAL  NAME
-----------------------------------------------------------------------------------------------
      4     10       6       0  BLOCK    1025     51     3      26  /SprMdrQ_20231126
      5     10       1       0  BLOCK    1025     43     1       1  /TimerM_7lTva1nY
      6     10       1       0  BLOCK    1025     43     1       1  /PowerM_E0pil3lu
-----------------------------------------------------------------------------------------------
Press 'Q' to back

通过日志和模块状态,可确认插件OneNet卸载成功,涉及到的模块已正常退出。

总结

  • 本次优化实现了插件的“热插拔”功能,通过监控文件变动并相应调用加载或卸载函数来完成。
  • 在此过程中,还发现动态链接库句柄泄露的问题,应确保dlopen返回的句柄得到妥善管理,在插件或程序退出时通过dlclose进行回收。
  • 优化过程中认识到,功能设计需细致入微,同时也应积极采纳并分析他人建议,以提高方案的可行性和实用性。此次就非常感激那位读者提出的问题。
相关文章
|
2月前
|
存储 缓存 C++
C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
C++ 标准模板库(STL)提供了一组功能强大的容器类,用于存储和操作数据集合。不同的容器具有独特的特性和应用场景,因此选择合适的容器对于程序的性能和代码的可读性至关重要。对于刚接触 C++ 的开发者来说,了解这些容器的基础知识以及它们的特点是迈向高效编程的重要一步。本文将详细介绍 C++ 常用的容器,包括序列容器(`std::vector`、`std::array`、`std::list`、`std::deque`)、关联容器(`std::set`、`std::map`)和无序容器(`std::unordered_set`、`std::unordered_map`),全面解析它们的特点、用法
C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
|
2月前
|
存储 机器学习/深度学习 编译器
【C++终极篇】C++11:编程新纪元的神秘力量揭秘
【C++终极篇】C++11:编程新纪元的神秘力量揭秘
|
2月前
|
存储 算法 C++
深入浅出 C++ STL:解锁高效编程的秘密武器
C++ 标准模板库(STL)是现代 C++ 的核心部分之一,为开发者提供了丰富的预定义数据结构和算法,极大地提升了编程效率和代码的可读性。理解和掌握 STL 对于 C++ 开发者来说至关重要。以下是对 STL 的详细介绍,涵盖其基础知识、发展历史、核心组件、重要性和学习方法。
|
2月前
|
存储 安全 算法
深入理解C++模板编程:从基础到进阶
在C++编程中,模板是实现泛型编程的关键工具。模板使得代码能够适用于不同的数据类型,极大地提升了代码复用性、灵活性和可维护性。本文将深入探讨模板编程的基础知识,包括函数模板和类模板的定义、使用、以及它们的实例化和匹配规则。
|
3月前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
75 5
|
4月前
|
算法 网络协议 数据挖掘
C++是一种功能强大的编程语言,
C++是一种功能强大的编程语言,
97 14
|
5月前
|
消息中间件 存储 安全
|
2月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
10天前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
41 12
|
1月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
49 16
下一篇
oss创建bucket