IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C++经验(十)-- 保存成员函数的函数地址,避免switch-case的大量使用 -> 正文阅读

[C++知识库]C++经验(十)-- 保存成员函数的函数地址,避免switch-case的大量使用

不知道你们有没有遇到过这种情况的,一个函数里面有40多个if-else或者switch-case的,并且还用了两边,我遇到了,大致数了下,这两个包含了40多个if-elseswitch-case的函数总共占用了400 多行

看到这种代码真的是会让人头大。试着研究了下,发现他的每个if或者每个case里面调用的函数格式是一样的,全都是 void XXXX();或者bool XXX();类型。这不就巧了吗,这跟那啥很像啊。然后就开始试着优化。

其实看到这种代码的时候,脑子里第一浮现的应该就是策略模式了。毕竟这完全就是一个简化版的策略嘛,只不过调用的函数都是在一个类里面而已。

但不久,我就放弃用策略了,这么多的 case,每个策略都新建一个类,那岂不是要40多个,这对于偷懒惯了的人可不是一次友好的体验。接着就去重新琢磨其他的方法。

很快,突然想起来观察者这种东西,好像这几年写Qt写的多了,用惯了信号和槽,都忘记了那些比较基础的东西了。观察者模式不就是被观察者保存了观察者某个函数的地址,在合适的地方进行了调用吗?

那这种方式是不是可行呢?

在类构造的时候将成员函数的地址通过某种手段保存下来,然后在需要的时候,精准地找到需要调用的成员函数的地址进行调用。

开始了尝试。

首先就是需要选定一个容器,这个容器要满足能够存储函数的地址,并且方便查找。该说不说,满足二分查找的 map 就可以了。

选定了容器之后,就需要定义容器的key-value的类型了。当然 key的类型好选择,不管是string还是枚举都可以,只要这个是唯一的就行。但是value的类型呢?都知道是需要保存成员函数的地址,但是这个地址该怎么提取出

一个通用的类型呢?

可是,回调不就是这样子的吗?

这些成员函数有着一样的格式,把他们都看作是一个回调函数不就行了吗?

于是就有了下面:

include<iostream>
#include<map>
#include<string>

class Test
{
public:
    Test()
    {
        auto map_insert = [&](std::string key, pFunc ptr)
        {
            m_mFuncPtr.emplace(std::make_pair(key, ptr));
        };

        map_insert("print1", &Test::print1);
        map_insert("print2", &Test::print2);
        map_insert("print3", &Test::print3);
    }

    ~Test() = default;

    void print(const std::string& func)
    {
        auto itor = m_mFuncPtr.find(func);
	    if (itor == m_mFuncPtr.end() || nullptr == itor->second)
		{
            return;
        }

	    (this->*itor->second)();    //调用函数
    }

private:
    void print1()
    {
        std::cout << "this is print1 func..." << std::endl;
    }

    void print2()
    {
        std::cout << "this is print2 func..." << std::endl;
    }

    void print3()
    {
        std::cout << "this is print3 func..." << std::endl;
    }

private:

    typedef void (Test::* pFunc)();

    std::map<std::string, pFunc> m_mFuncPtr;
};

int main(int argc, char* argv[])
{
    Test test;
    test.print("print1");
    return 0;
}

当然,函数的声明是可以放在类外面的,放在类外面的话就是这样的。

class Test;
typedef void (Test::*pFunc)();
struct CmdPtr
{
	pFunc ptr = nullptr;
	CmdPtr()
	{
		reset();
	}
	void reset()
	{
		ptr = nullptr;	
	}	
};

class Test
{
public:
    Test()
    {
        auto map_insert = [&](std::string key, pFunc ptr)
        {
            CmdPtr cfg;
            cfg.ptr = ptr;
            m_mFuncPtr.emplace(std::make_pair(key, cfg));
        };

        map_insert("print1", &Test::print1);
        map_insert("print2", &Test::print2);
        map_insert("print3", &Test::print3);
    }

    void print(const std::string& func)
    {
        auto itor = m_mFuncPtr.find(func);
	    if (itor == m_mFuncPtr.end() || nullptr == itor->second.ptr)
		{
            return;
        }

	    (this->*itor->second.ptr)();    //调用函数
    }

private:
    std::map<std::string, CmdPtr> m_mFuncPtr;
};

其实这样的修改,跟上面的那种直接写在类里面的是没什么区别的。但好处是满足了我们以对象管理资源的建议。

上述代码中,用了 emplace 而不是 insert 来进行map元素的插入,主要原因是因为这种方式效率更高,可以参考《stl标准库系列之–map》的第八节,里面有对 map 容器的几种插入数据的方式进行比较详细的说明和比较。

最后说一句,我们天天在重复造轮子,为什么不能在造轮子的过程中学会偷懒了,学会偷懒,才会简化你的代码。

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-06-29 18:48:02  更:2022-06-29 18:50:33 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 6:53:56-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码