当前位置:首页 > c++ > 正文内容

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

xuwenyan1年前 (2022-07-11)c++1242

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

    扫描二维码推送至手机访问。

    版权声明:本文由艺文笔记发布,如需转载请注明出处。

    本文链接:https://www.xuwenyan.com/archives/2765

    分享给朋友:

    “std::function是怎样实现的(大致实现原理)?” 的相关文章

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

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

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

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

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

    学习ffmpeg时遇到一个问题,ffmpeg解码出RGB颜色后怎么绘制到屏幕上,于是将GDI、MFC_GDI、GDI+等方式都记录一下 1:注意按windows的要求,R、G、B、A顺序要调整为B、G、R、A 。 2:GDI不支持透明通道A,透明通道A的值读进去以后没有作用。想要支持透...

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

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

    现象:程序在xp系统上面总是莫名其妙的崩溃,检查代码看不出任何问题,感觉代码都很好。即使你远程调试,找到了崩溃的点,当你注释了崩溃点之后,还是会崩溃到别的地方。当你遇到了这种情况的时候,不妨参照一下下面的方法看看,说不定可以解决问题。如何解决?将崩溃程序相关的所有工程代码全部关闭全局变量的线程安全检...

    uafxcwd.lib(afxmem.obj) : error LNK2005:

    uafxcwd.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)"解决办法

    如果在编译MFC程序的时候出现下列及类似的错误: 1˃uafxcwd.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)" (??2@YAPAXI@Z) 已经在 LIBCMTD.lib(new...

    C++ 获取进程所在目录(进程全路径)

    C++ 获取进程所在目录(进程全路径)

    打开windows任务管理器,会看到很多的进程在运行,随机挑选一个,如何通过c++代码获取某一个进程的所在全路径呢?这也是在windows软件开发中经常遇到的需求。通过进程名获取进程全路径由于可能很多进程叫同一个名字,所以获得的结果也有可能是多个#include <windows.h...

    static_assert和assert有什么区别?

    static_assert和assert有什么区别?

    static_assert和assert都是断言,都可用于判断一个条件是否成立,并且在条件不成立时及时给出错误提示。那它们用什么不同和需要注意的地方呢? 1:static_assert在编译期执行,而assert在运行期执行。2:static_assert无论在debug模式还是relea...