0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

std::function简介及模板类声明

5jek_harmonyos 来源:编程学习总站 作者:写代码的牛顿 2021-07-28 15:30 次阅读

01

std::function简介

std::function是一个函数包装器,该函数包装器模板能包装任何类型的可调用实体,如普通函数,函数对象,lamda表达式等。包装器可拷贝,移动等,并且包装器类型仅仅依赖于调用特征,而不依赖于可调用元素自身的类型。

std::function是C++11的新特性,包含在头文件《functional》中。一个std::function类型对象实例可以包装下列这几种可调用实体:函数、函数指针、成员函数、静态函数、lamda表达式和函数对象。

std::function对象实例可被拷贝和移动,并且可以使用指定的调用特征来直接调用目标元素。当std::function对象实例未包含任何实际可调用实体时,调用该std::function对象实例将抛出std::bad_function_call异常。02

std::function实战

std::function模板类声明

template《class _Rp, class 。。._ArgTypes》 class _LIBCPP_TEMPLATE_VIS function《_Rp(_ArgTypes.。。)》 : public __function::__maybe_derive_from_unary_function《_Rp(_ArgTypes.。。)》, public __function::__maybe_derive_from_binary_function《_Rp(_ArgTypes.。。)》 { 。。. }std::function模板类成员函数声明

typedef _Rp result_type; // construct/copy/destroy: _LIBCPP_INLINE_VISIBILITY function() _NOEXCEPT { } _LIBCPP_INLINE_VISIBILITY function(nullptr_t) _NOEXCEPT {} function(const function&); function(function&&) _NOEXCEPT; template《class _Fp, class = _EnableIfCallable《_Fp》》 function(_Fp); #if _LIBCPP_STD_VER 《= 14 template《class _Alloc》 _LIBCPP_INLINE_VISIBILITY function(allocator_arg_t, const _Alloc&) _NOEXCEPT {} template《class _Alloc》 _LIBCPP_INLINE_VISIBILITY function(allocator_arg_t, const _Alloc&, nullptr_t) _NOEXCEPT {} template《class _Alloc》 function(allocator_arg_t, const _Alloc&, const function&); template《class _Alloc》 function(allocator_arg_t, const _Alloc&, function&&); template《class _Fp, class _Alloc, class = _EnableIfCallable《_Fp》》 function(allocator_arg_t, const _Alloc& __a, _Fp __f); #endif function& operator=(const function&); function& operator=(function&&) _NOEXCEPT; function& operator=(nullptr_t) _NOEXCEPT; template《class _Fp, class = _EnableIfCallable《_Fp》》 function& operator=(_Fp&&); ~function(); // function modifiers: void swap(function&) _NOEXCEPT; #if _LIBCPP_STD_VER 《= 14 template《class _Fp, class _Alloc》 _LIBCPP_INLINE_VISIBILITY void assign(_Fp&& __f, const _Alloc& __a) {function(allocator_arg, __a, _VSTD::forward《_Fp》(__f)).swap(*this);} #endif // function capacity: _LIBCPP_INLINE_VISIBILITY _LIBCPP_EXPLICIT operator bool() const _NOEXCEPT { return static_cast《bool》(__f_); } // deleted overloads close possible hole in the type system template《class _R2, class.。。 _ArgTypes2》 bool operator==(const function《_R2(_ArgTypes2.。。)》&) const = delete; template《class _R2, class.。。 _ArgTypes2》 bool operator!=(const function《_R2(_ArgTypes2.。。)》&) const = delete; public: // function invocation: _Rp operator()(_ArgTypes.。。) const; #ifndef _LIBCPP_NO_RTTI // function target access: const std::type_info& target_type() const _NOEXCEPT; template 《typename _Tp》 _Tp* target() _NOEXCEPT; template 《typename _Tp》 const _Tp* target() const _NOEXCEPT; #endif // _LIBCPP_NO_RTTI从成员函数里我们知道std::function对象实例不允许进行==和!=比较操作,std::function模板类实例最终调用成员函数_Rp operator()(_ArgTypes.。。) const进而调用包装的调用实体。1、std::function包装函数指针定义一个std::function《int(int)》对象实例

std::function《int(int)》 callback;std::function对象实例包装函数指针

int (*fun_ptr)(int); int fun1(int a){ return a; } int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; fun_ptr = fun1; //函数指针fun_ptr指向fun1函数 callback = fun_ptr; //std::function对象包装函数指针 std::cout 《《 callback(10) 《《 std::endl; //std::function对象实例调用包装的实体 return 0; }2、std::function包装函数

int fun1(int a){ return a; } int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; callback = fun1; //std::function包装函数 std::cout 《《 callback(42) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }3、std::function包装模板函数

template《typename T》 T fun2(T a){ return a + 2; } int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; callback = fun2《int》; //std::function包装模板函数 std::cout 《《 callback(10) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }4、std::function包装函数对象

struct add{ int operator()(int x){ return x + 9; } }; int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; callback = add(); //std::function包装对象函数 std::cout 《《 callback(2) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }5、std::function包装lamda表达式

int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; auto fun3 = [](int a) {return a * 2;}; //lamda表达式 callback = fun3; //std::function包装lamda表达式 std::cout 《《 callback(9) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }

6、std::function包装模板对象函数

template 《typename T》 struct sub{ T operator()(T a){ return a - 8; } }; int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; callback = sub《int》(); //std::function包装模板对象函数 std::cout 《《 callback(2) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }

7、std::function包装模板对象静态函数

template 《typename T》 struct foo2{ static T foo(T a){ return a * 4; } }; int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; callback = foo2《int》::foo; //std::function包装模板对象静态函数 std::cout 《《 callback(3) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }

8、std::function包装对象静态函数

struct foo1{ static int foo(int a){ return a * 3; } }; int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; callback = foo1::foo; //std::function包装对象静态函数 std::cout 《《 callback(5) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }

9、std::function包装类成员函数

struct foo3{ int foo(int a){ return a * a; } }; int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; foo3 test_foo1; callback = std::bind(&foo3::foo, test_foo1, std::_1); //std::function包装类成员函数 std::cout 《《 callback(9) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }

这里我们用到了std::bind,C++11中std::bind函数的意义就如字面上的意思一样,用来绑定函数调用的某些参数。std::bind的思想其实是一种延迟计算的思想,将可调用对象保存起来,然后在需要的时候再调用。而且这种绑定是非常灵活的,不论是普通函数还是函数对象还是成员函数都可以绑定,而且其参数可以支持占位符。

这里的std::_1是一个占位符,且绑定第一个参数,若可调用实体有2个形参,那么绑定第二个参数的占位符是std::_2。

10、std::function包装模板类成员函数

template 《typename T》 struct foo4{ T foo(T a){ return a * 6; } }; int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; foo4《int》 test_foo2; callback = std::bind(&foo4《int》::foo, test_foo2, std::_1); //std::function包装模板类成员函数 std::cout 《《 callback(7) 《《 std::endl; //std::function对象实例调用包装的调用实体 return 0; }

11、std::function拷贝、移动

int main(int argc, char *argv[]){ std::cout 《《 “Hello world” 《《 std::endl; std::function《int(int)》 callback2 = callback; //拷贝赋值运算符 std::cout 《《 callback2(7) 《《 std::endl; std::function《int(int)》&& callback3 = std::move(callback); //移动赋值运算符 std::cout 《《 callback3(7) 《《 std::endl; std::cout 《《 callback(7) 《《 std::endl; std::function《int(int)》 callback4(callback); //拷贝 std::cout 《《 callback4(7) 《《 std::endl; return 0; }

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • STD
    STD
    +关注

    关注

    0

    文章

    36

    浏览量

    14340

原文标题:C++ std::function详解与实战

文章出处:【微信号:harmonyos_developer,微信公众号:harmonyos_developer】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    基于ArkTS语言的OpenHarmony APP应用开发:HelloOpenharmony

    1、程序简介 该程序是基于OpenHarmony标准系统编写的UI应用:HelloOpenHarmony。 本案例是基于API 9接口开发。 本案例已在OpenHarmony凌蒙派-RK3568
    发表于 09-14 12:47

    深入解析MIL-STD-1553B模块卡

    MIL-STD-1553B模块
    发表于 09-06 11:43 0次下载

    如何对MIL-STD-1553B进行选型

    MIL-STD-1553B产品选型是一个复杂而细致的过程,‌需要综合考虑多个因素以确保所选产品能够满足特定应用场景的需求。一、‌引言MIL-STD-1553B作为一种广泛应用于航空航天领域的数据总线
    的头像 发表于 09-05 17:37 318次阅读
    如何对MIL-<b class='flag-5'>STD</b>-1553B进行选型

    基于OpenHarmony标准系统的C++公共基础库案例:SafeQueue

    1、程序简介该程序是基于OpenHarmony的C++公共基础库的线程安全队列:SafeQueue。线程安全队列,是在dequeue的基础上封装std::lock_guard,以此实现线程的相关
    的头像 发表于 08-30 12:41 230次阅读
    基于OpenHarmony标准系统的C++公共基础<b class='flag-5'>类</b>库案例:SafeQueue

    基于OpenHarmony标准系统的C++公共基础库案例:SafeStack

    1、程序简介该程序是基于OpenHarmony的C++公共基础库的线程安全队列:SafeQueue。线程安全队列,是在dequeue的基础上封装std::lock_guard,以此实现线程的相关
    的头像 发表于 08-30 12:41 280次阅读
    基于OpenHarmony标准系统的C++公共基础<b class='flag-5'>类</b>库案例:SafeStack

    CW32F003E4芯片入门学习:4.工程模板创建(使用例程或模板)

    模板路径:CW32F003_StandardPeripheralLib_V1.4ExamplesTemplate
    的头像 发表于 04-24 14:14 373次阅读
    CW32F003E4芯片入门学习:4.工程<b class='flag-5'>模板</b>创建(使用例程或<b class='flag-5'>模板</b>)

    verilog task和function区别

    verilog中的task和function都是用于实现模块中的可重复的功能,并且可以接收参数和返回结果。但是它们在编写和使用上有一些区别。下面将详细介绍task和function的区别。 语法结构
    的头像 发表于 02-22 15:53 958次阅读

    verilog中function和task的区别

    非常相似,但它们在功能和使用方式上有一些重要的区别。 定义和声明方式不同: Function:使用关键字"function"来定义和声明。函数可以有一个或多个输入参数,可以有一个返回值
    的头像 发表于 02-22 15:40 1740次阅读

    Samtec - 通过优化焊膏模板开孔来扩大连接器的选择范围

    mm焊膏模板配合使用,同时在良品率为100%的情况下也能满足IPC-J-STD-001 Class 2标准的要求。
    发表于 01-02 15:33 191次阅读
    Samtec - 通过优化焊膏<b class='flag-5'>模板</b>开孔来扩大连接器的选择范围

    verilog语法-如何使用function提高效率?

    function的作用返回一个数值,此数值由一串组合逻辑代码计算得到。
    的头像 发表于 12-25 15:27 2218次阅读
    verilog语法-如何使用<b class='flag-5'>function</b>提高效率?

    如何使用CMW500测试频谱模板

    在使用CMW500测试频谱模板之前,首先我们需要了解什么是频谱模板以及其在通信系统测试中的作用。频谱模板是指在特定的频率范围内,记录了该频率范围内的信号功率谱密度的一种图形表示。在通信系统测试中
    的头像 发表于 12-25 15:10 1442次阅读

    OpenCV边缘模板匹配算法原理详解

    OpenCV中自带的模板匹配算法,完全是像素基本的模板匹配,特别容易受到光照影响,光照稍微有所不同,该方法就会歇菜了!搞得很多OpenCV初学者刚学习到该方法时候很开心,一用该方法马上很伤心
    的头像 发表于 12-07 10:56 1309次阅读
    OpenCV边缘<b class='flag-5'>模板</b>匹配算法原理详解

    如何在std rms模块中读取相关电平值?

    最近又个项目,需要使用Compressor,并读取输入前、输入后及被压缩的电平值,在SigmaStudio中选择std rms模块。但使用开发板仿真在SigmaStudio中也无法看到电平的跳动
    发表于 11-30 07:47

    java的主必须是public

    一下Java程序的基本结构。一个Java程序可以包含多个,但是只能有一个公共的作为主。这个主必须包含一个特殊的方法,即main方法,它是程序的入口点。当程序被执行时,首先执行m
    的头像 发表于 11-28 16:15 5381次阅读

    java抽象可以有普通方法吗

    时使用abstract关键字,我们可以将该方法声明为抽象方法。抽象方法没有具体的实现,只是定义了方法的签名。子类继承抽象后必须实现所有的抽象方法,否则子类也需要被声明为抽象。抽象方
    的头像 发表于 11-21 10:22 1468次阅读