c++ 11 基础 :

    std::function

类模版 std::function是一种通用、多态的函数封装。std::function的实例可以对任何可以调用的目标进行存储、复制、和调用操作,这些目标包括函数、lambda表达式、绑定表达式、以及其它函数对象等。

用法示例:

①保存自由函数

  1. void printA(int a){
  2.     cout<<a<<endl;
  3. }
  4.  std::function<void(int a)> func;
  5.  func = printA;
  6.  func(2);

运行输出: 2

②保存lambda表达式

  1. std::function<void()> func_1 = [](){cout<<"hello world"<<endl;};
  2.     func_1();

运行输出:hello world

③保存成员函数

  1. struct Foo {
  2.     Foo(int num) : num_(num) {}
  3.     void print_add(int i) const { cout << num_+<< '\n'; }
  4.     int num_;
  5. };
  6.  // 保存成员函数std::function<void(const Foo&, int)> f_add_display = &Foo::print_add;
  7.     Foo foo(2);
  8.     f_add_display(foo, 1);

运行输出: 3

    bind        bind是一组用于函数绑定的模板。在对某个函数进行绑定时,可以指定部分参数或全部参数,也可以不指定任何参数,还可以调整各个参数间的顺序。对于未指定的参 数,可以使用占位符_1、_2、_3来表示。_1表示绑定后的函数的第1个参数,_2表示绑定后的函数的第2个参数,其他依次类推。

下面通过程序例子了解一下用法:

  1. #include <iostream>using namespace std;
  2. class A
  3. {
  4. public:
  5.     void fun_3(int k,int m){
  6.         cout<<k<<" "<<m<<endl;
  7.     }
  8. };
  9. void fun(int x,int y,int z){
  10.     cout<<x<<"  "<<y<<"  "<<z<<endl;
  11. }
  12. void fun_2(int &a,int &b){
  13.     a++;
  14.     b++;
  15.     cout<<a<<"  "<<b<<endl;
  16. }
  17. int main(int argc, const char * argv[]){
  18.     auto f1 = bind(fun,1,2,3); //表示绑定函数 fun 的第一,二,三个参数值为: 1 2 3
  19.     f1(); //print:1  2  3auto f2 = bind(fun, placeholders::_1,placeholders::_2,3);
  20.     //表示绑定函数 fun 的第三个参数为 3,而fun 的第一,二个参数分别有调用 f2 的第一,二个参数指定
  21.     f2(1,2);//print:1  2  3auto f3 = bind(fun,placeholders::_2,placeholders::_1,3);
  22.     //表示绑定函数 fun 的第三个参数为 3,而fun 的第一,二个参数分别有调用 f3 的第二,一个参数指定//注意: f2  和  f3 的区别。
  23.     f3(1,2);//print:2  1  3int n = 2;
  24.     int m = 3;
  25.     
  26.     auto f4 = bind(fun_2, n,placeholders::_1);
  27.     f4(m); //print:3  4cout<<m<<endl;//print:4  说明:bind对于不事先绑定的参数,通过std::placeholders传递的参数是通过引用传递的cout<<n<<endl;//print:2  说明:bind对于预先绑定的函数参数是通过值传递的
  28.     
  29.     
  30.     A a;
  31.     auto f5 = bind(&A::fun_3, a,placeholders::_1,placeholders::_2);
  32.     f5(10,20);//print:10 20std::function<void(int,int)> fc = std::bind(&A::fun_3, a,std::placeholders::_1,std::placeholders::_2);
  33.     fc(10,20);//print:10 20return 0;
  34. }

    CC_CALLBACK

一、通过 HelloWorldScene 中的 closeItem 开始

在cocos2d-x 2.x 版本中:

  1. CCMenuItemImage *pCloseItem = CCMenuItemImage::create(
  2.                                         "CloseNormal.png",
  3.                                         "CloseSelected.png",
  4.                                         this,
  5.                                         menu_selector(HelloWorld::menuCloseCallback));

在cocos2d-x 3.0 版本中:

  1. auto closeItem = MenuItemImage::create(
  2.                                            "CloseNormal.png",
  3.                                            "CloseSelected.png",
  4.                                            CC_CALLBACK_1(HelloWorld::menuCloseCallback, this));
  1. void HelloWorld::menuCloseCallback(Object* pSender)
  2. {
  3.     Director::getInstance()->end();
  4. #if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
  5.     exit(0);
  6. #endif
  7. }

注意到在3.0版本中使用到 CC_CALLBACK_1 这样一个宏定义。

  1. // new callbacks based on C++11#define CC_CALLBACK_0(__selector__,__target__, ...) std::bind(&__selector__,__target__, ##__VA_ARGS__)
  2. #define CC_CALLBACK_1(__selector__,__target__, ...) std::bind(&__selector__,__target__, std::placeholders::_1, ##__VA_ARGS__)
  3. #define CC_CALCC_CALLBACK_1(HelloWorld::menuCloseCallback,this)LBACK_2(__selector__,__target__, ...) std::bind(&__selector__,__target__, std::placeholders::_1, std::placeholders::_2, ##__VA_ARGS__)
  4. #define CC_CALLBACK_3(__selector__,__target__, ...) std::bind(&__selector__,__target__, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3 ##__VA_ARGS__)

这里主要注意两点:一是 std::bind,二是##_VA_ARGS_; ##_VA_ARGS_是可变参数宏

原来还有 CC_CALLBACK_0 1 2 3;而其中又有什么区别呢?

1、首先我们看看3.0版本中MenuItemImage的create方法:

  1. MenuItemImage * MenuItemImage::create(const std::string& normalImage, const std::string& selectedImage, const ccMenuCallback& callback)

其中的回调参数是 ccMenuCallback

  1. typedef std::function<void(Object*)> ccMenuCallback

来这里使用到了 C++ 中的 function 语法。

注意到 在 CC_CALLBACK_  的宏定义的中使用到的是 C++ 的 bind 语法,怎么不一致了呢? -- 见下面第四点 function

2、看回 CC_CALLBACK_  的宏定义

原来 CC_CALLBACK_  的宏定义中后面的 0 1 2 3分别表示的是 不事先指定回调函数参数的个数。

例如说 CC_CALLBACK_ 1 表示的是,回调函数中不事先指定参数是一个,而事先指定的回调函数的参数 可以任意多个

而且要注意到其中 不指定回调函数参数  和  指定回调函数参数  的顺序,注意不事先指定的在前,事先指定的在后

下面通过例子说明这一点:

假设回调函数:

  1. // a selector callbackvoid menuCloseCallback(Object* pSender,int a,int b);
  1. void HelloWorld::menuCloseCallback(Object* pSender,int a,int b)
  2. {
  3.     std::cout<<a<<"  "<<b<<std::endl;
  4.     Director::getInstance()->end();
  5. #if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)exit(0);
  6. #endif
  7. }

注意到在回调函数中输出 a b

  1. auto closeItem = MenuItemImage::create(
  2.                                            "CloseNormal.png",
  3.                                            "CloseSelected.png",
  4.                                            CC_CALLBACK_1(HelloWorld::menuCloseCallback,this,1,2));

注意中其中 指定了两个参数 1 2

运行,在 点击closeItem  的时候,就会输出这两个事先指定的参数 1  2。

那么,不事先指定的参数是在什么时候传入的呢?

  1. void MenuItem::activate()
  2. {
  3.     if (_enabled)
  4.     {
  5.         if( _callback )
  6.         {
  7.             _callback(this);
  8.         }
  9.         
  10.         if (kScriptTypeNone != _scriptType)
  11.         {
  12.             BasicScriptData data(this);
  13.             ScriptEvent scriptEvent(kMenuClickedEvent,&data);
  14.             ScriptEngineManager::getInstance()->getScriptEngine()->sendEvent(&scriptEvent);
  15.         }
  16.     }
  17. }

注意到其中的  _callback(this);  对了,这个时候就传入了 这个不事先指定的回调函数参数。

这样,closeItem 的回调函数的 void HelloWorld::menuCloseCallback(Object* pSender,int a,int b) 的三个参数都知道了。

第一个 不事先指定,在menu item调用 activate 的时候,_callback(this) 传入,this 也即是这个 menu item;第二、三个参数是事先指定的 1,2。

已经知道  CC_CALLBACK_  的宏定义是 std::bind 那么我们可以直接使用std::bind。

如下:

  1. auto closeItem = MenuItemImage::create(
  2.                                            "CloseNormal.png",
  3.                                            "CloseSelected.png",
  4.                                           std::bind(&HelloWorld::menuCloseCallback, this,std::placeholders::_1,1,2));

CC_CALLBACK原理及应用的更多相关文章

  1. 奇异值分解(SVD)原理与在降维中的应用

    奇异值分解(Singular Value Decomposition,以下简称SVD)是在机器学习领域广泛应用的算法,它不光可以用于降维算法中的特征分解,还可以用于推荐系统,以及自然语言处理等领域.是 ...

  2. node.js学习(三)简单的node程序&&模块简单使用&&commonJS规范&&深入理解模块原理

    一.一个简单的node程序 1.新建一个txt文件 2.修改后缀 修改之后会弹出这个,点击"是" 3.运行test.js 源文件 使用node.js运行之后的. 如果该路径下没有该 ...

  3. 线性判别分析LDA原理总结

    在主成分分析(PCA)原理总结中,我们对降维算法PCA做了总结.这里我们就对另外一种经典的降维方法线性判别分析(Linear Discriminant Analysis, 以下简称LDA)做一个总结. ...

  4. [原] KVM 虚拟化原理探究(1)— overview

    KVM 虚拟化原理探究- overview 标签(空格分隔): KVM 写在前面的话 本文不介绍kvm和qemu的基本安装操作,希望读者具有一定的KVM实践经验.同时希望借此系列博客,能够对KVM底层 ...

  5. H5单页面手势滑屏切换原理

    H5单页面手势滑屏切换是采用HTML5 触摸事件(Touch) 和 CSS3动画(Transform,Transition)来实现的,效果图如下所示,本文简单说一下其实现原理和主要思路. 1.实现原理 ...

  6. .NET Core中间件的注册和管道的构建(1)---- 注册和构建原理

    .NET Core中间件的注册和管道的构建(1)---- 注册和构建原理 0x00 问题的产生 管道是.NET Core中非常关键的一个概念,很多重要的组件都以中间件的形式存在,包括权限管理.会话管理 ...

  7. python自动化测试(2)-自动化基本技术原理

    python自动化测试(2) 自动化基本技术原理 1   概述 在之前的文章里面提到过:做自动化的首要本领就是要会 透过现象看本质 ,落实到实际的IT工作中就是 透过界面看数据. 掌握上面的这样的本领 ...

  8. CRC、反码求和校验 原理分析

    3月份开始从客户端转后台,算是幸运的进入全栈工程师的修炼阶段.这段时间一边是老项目的客户端加服务器两边的维护和交接,一边是新项目加加加班赶工,期间最长经历了连续工作三天只睡了四五个小时的煎熬,人生也算 ...

  9. 菜鸟学Struts2——Struts工作原理

    在完成Struts2的HelloWorld后,对Struts2的工作原理进行学习.Struts2框架可以按照模块来划分为Servlet Filters,Struts核心模块,拦截器和用户实现部分,其中 ...

随机推荐

  1. MSSQLServer基础05(联合查询,连接查询)

    联合结果集union(集合运算符) 集合运算符是对两个集合操作的,两个集合必须具有相同的列数,列具有相同的数据类型(至少能隐式转换的),最终输出的集合的列名由第一个集合的列名来确定.(可以用来连接多个 ...

  2. 115. Distinct Subsequences

    题目: Given a string S and a string T, count the number of distinct subsequences of T in S. A subseque ...

  3. EXP-00091 Exporting questionable statistics

    今天在我们对Oracle做EXP的过程中,出现EXP-00091 Exporting questionable statistics.的信息,但是也提示导出成功.最好查询了下发现其实它就是exp的er ...

  4. POJ2253——Frogger(Floyd变形)

    Frogger DescriptionFreddy Frog is sitting on a stone in the middle of a lake. Suddenly he notices Fi ...

  5. 2014年最火的 21个JavaScript 框架

    下面,我们将会介绍 2014 年最火的 21 款JavaScript 框架,专为前端开发者准备的哦:)众所周知, JavaScript 框架是 JavaScript编程语言最棒的特性之一. JavaS ...

  6. [POJ 2429] GCD & LCM Inverse

    GCD & LCM Inverse Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 10621   Accepted: ...

  7. 表格的一些原生js操作(隔行变色,高亮显示,添加删除,搜索)

    看着网上的视频教程,虽说还是有点简单,但还是不免想记录下.这些操作包括(隔行变色,高亮显示,添加删除,搜索功能),而这儿就是涉及table的原有属性“tBodies” “rows” “cells”等几 ...

  8. SharePoint 2007 页面定制(一)

    转:http://www.nanmu.net/SharePoint-MOSS-WSS-Silverlight/Lists/Posts/Post.aspx?ID=74 本文主要包括以下几方面内容: 1. ...

  9. NGINX开篇

    前言 最近空闲时间比较多, 开始阅读nginx源码, 阅读的过程总结和笔记整理了下, 汇集成了一个系列的文章, 由于nginx功能实在太多, 没法做到面面俱到, 只对已经阅读过的源码进行记录总结, 以 ...

  10. [OFBiz]简介 二

    1. 执行ant run-install后,生成了55个ofbiz的jar.加上最初的E:\apache-ofbiz-10.04\framework\entity\lib\ofbiz-minerva. ...