当前位置:首页 > c++ > std::function是怎样实现的(大致实现原理)?

std::function是怎样实现的(大致实现原理)?

xuwenyan4周前 (07-11)c++450

std::function是怎样实现的?

首先必须实现一个默认的模板函数,然后根据参数个数特例化不同的实现,在没有变长参数模块语法支持时,只能根据参数个数特例化多个实现,非常繁琐。有了语法支持后,可以使用变长参数模板轻松实现任意个数参数的function。

大致实现如下:

不需要参数时:

namespace my {
  template <typename TR>
  class function{};

  template <typename TR>
  class function<TR()> {
    using PFunc = TR(*)();
    PFunc pfunc_;
  public:
    function(PFunc pfunc)
      : pfunc_(pfunc) {
    }

    TR operator()() {
      return pfunc_();
    }
  };
}

需要一个参数时:

namespace my {
  template <typename TR>
  class function{};

  template <typename TR, typename TArg>
  class function<TR(TArg)> {
    using PFunc = TR(*)(TArg);
    PFunc pfunc_;
  public:
    function(PFunc pfunc)
      : pfunc_(pfunc) {
    }

    TR operator()(TArg arg) {
      return pfunc_(arg);
    }
  };
}

当需要两个参数时:

namespace my {
  template <typename TR>
  class function{};

  template <typename TR, typename TArg1, typename TArg2>
  class function<TR(TArg1, TArg2)> {
    using PFunc = TR(*)(TArg1, TArg2);
    PFunc pfunc_;
  public:
    function(PFunc pfunc)
      : pfunc_(pfunc) {
    }

    TR operator()(TArg1 arg1, TArg2 arg2) {
      return pfunc_(arg1, arg2);
    }
  };
}

更多个数的参数同理增加参数就行,接下来介绍使用变长参数模板如何实现

使用变长参数模块实现

namespace my {
  template <typename>
  class function {};

  template <typename TR, typename ...TArgs>
  class function<TR(TArgs...)> {
    using PFunc = TR(*)(TArgs...);
    PFunc pfunc_;
  public:
    function(PFunc pfunc)
      : pfunc_(pfunc) {
    }

    TR operator()(TArgs... args) {
      return pfunc_(args...);
    }
  };
}

int subtraction(int a, int b) {
  return a - b;
}

int main() {
  std::function<int(int, int)> sub1 = subtraction;
  my::function<int(int, int)> sub2 = subtraction;
  printf("%d
", sub1(5, 2));
  printf("%d
", sub2(5, 2));

  system("pause");
  return 0;
}



    文章作者:xuwenyan
    版权声明:本文为本站原创文章,转载请注明出处,非常感谢,如版权漏申明或您觉得任何有异议的地方欢迎与本站取得联系。

    相关文章

    C++如何实现挂起进程、恢复进程

    C++如何实现挂起进程、恢复进程

    1:枚举进程的所有线程,使用SuspendThread函数挂起每一个线程,需要恢复时使用ResumeThread函数恢复。因为挂起和恢复的顺序是不可预知的,所以可能会导致一些多线程程序崩溃,单线程程...

    C++如何获取控制台程序的输出内容?

    C++如何获取控制台程序的输出内容?

    很多工具程序(如ffmpeg)的进度显示往往都是以控制台字符显示的方法,我们可能需要调用这种控制台工具去完成工作,但同时又希望以友好的ui界面去显示当前的工作状态(如进度)。此时我们能想到的就是运行...

    使用GDI、MFC_GDI、GDI+绘制数组RGBA序列

    使用GDI、MFC_GDI、GDI+绘制数组RGBA序列

    学习ffmpeg时遇到一个问题,ffmpeg解码出RGB颜色后怎么绘制到屏幕上,于是将GDI、MFC_GDI、GDI+等方式都记录一下 1:注意按windows的要求,R、G、B、A顺序要调整...

    VC的ATL工程向导同时生成一个PS工程是做什么的?可以不要吗?

    VC的ATL工程向导同时生成一个PS工程是做什么的?可以不要吗?

    例如,我用VC2015的工程向导新建一个ATL的工程名字叫myAtl,那么VC会同时给我生成一个叫做myAtlPS的工程。这个myAtlPS工程是做什么的?什么情况下可以不需要它?什么情况下它又是必...

    解决程序在xp系统总是莫名奇妙的崩溃问题(/Zc:threadSafeInit- )

    解决程序在xp系统总是莫名奇妙的崩溃问题(/Zc:threadSafeInit- )

    现象: 程序在xp系统上面总是莫名其妙的崩溃,检查代码看不出任何问题,感觉代码都很好。即使你远程调试,找到了崩溃的点,当你注释了崩溃点之后,还是会崩溃到别的地方。 当你遇到了这种情况的时...

    ATL实现windows右键菜单扩展(ContextMenu)

    ATL实现windows右键菜单扩展(ContextMenu)

    右键菜单,即用户右击shell对象时弹出的上下文菜单(context menu)。本文记录了如何创建右键菜单的基本过程,跟着步骤一步一步来,即可创建出一个右键菜单工程。 第一步,新建一个ATL...

    发表评论

    访客

    ◎欢迎参与讨论,请在这里发表您的看法和观点。