c++ 11 基础 :


std::function


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


用法示例:


①保存自由函数


void printA(int a)

{

cout<<a<<endl;

}

std::function<void(int a)> func;

func = printA;

func(2);

运行输出: 2


②保存lambda表达式


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

func_1();

运行输出:hello world


③保存成员函数



struct Foo {

Foo(int num) : num_(num) {}

void print_add(int i) const { cout << num_+i << '\n'; }

int num_;

};

// 保存成员函数

std::function<void(const Foo&,int)> f_add_display = &Foo::print_add;

Foo foo(2);

f_add_display(foo,1);

运行输出: 3


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


下面通过程序例子了解一下



用法:



#include <iostream>

using namespace std;

class A

{

public:

void fun_3(int k,int m)

{

cout<<k<<" "<<m<<endl;

}

};

void fun(int x,int y,int z)

{

cout<<x<<" "<<y<<" "<<z<<endl;

}

void fun_2(int &a,int &b)

{

a++;

b++;

cout<<a<<" "<<b<<endl;

}

int main(int argc,const char * argv[])

{

auto f1 = bind(fun,1,2,3); //表示绑定函数 fun 的第一,二,三个参数值为: 1 2 3

f1(); //print:1 2 3

auto f2 = bind(fun,placeholders::_1,placeholders::_2,3);

//表示绑定函数 fun 的第三个参数为 3,而fun 的第一,二个参数分别有调用 f2 的第一,二个参数指定

f2(1,2);//print:1 2 3

auto f3 = bind(fun,3);

//表示绑定函数 fun 的第三个参数为 3,而fun 的第一,二个参数分别有调用 f3 的第二,一个参数指定

//注意: f2 和 f3 的区别。

f3(1,2);//print:2 1 3

int n = 2;

int m = 3;

auto f4 = bind(fun_2,n,placeholders::_1);

f4(m); //print:3 4

cout<<m<<endl;//print:4 说明:bind对于不事先绑定的参数,通过std::placeholders传递的参数是通过引用传递的

cout<<n<<endl;//print:2 说明:bind对于预先绑定的函数参数是通过值传递的

A a;

auto f5 = bind(&A::fun_3,a,placeholders::_2);

f5(10,20);//print:10 20

std::function<void(int,int)> fc = std::bind(&A::fun_3,std::placeholders::_1,std::placeholders::_2);

fc(10,20);//print:10 20

return 0;

}

CC_CALLBACK


一、通过 HelloWorldScene 中的 closeItem 开始


在cocos2d-x 2.x 版本中:



CcmenuItemImage *pCloseItem = CcmenuItemImage::create(

"Closenormal.png",

"CloseSelected.png",

this,

menu_selector(HelloWorld::menuCloseCallback));

在cocos2d-x 3.0 版本中:



auto closeItem = MenuItemImage::create(

"Closenormal.png",

"CloseSelected.png",

CC_CALLBACK_1(HelloWorld::menuCloseCallback,this));




void HelloWorld::menuCloseCallback(Object* pSender)

{

Director::getInstance()->end();

#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)

exit(0);

#endif

}

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



// new callbacks based on C++11

#define CC_CALLBACK_0(__selector__,__target__,...) std::bind(&__selector__,##__VA_ARGS__)

#define CC_CALLBACK_1(__selector__,##__VA_ARGS__)

#define CC_CALCC_CALLBACK_1(HelloWorld::menuCloseCallback,this)LBACK_2(__selector__,std::placeholders::_2,##__VA_ARGS__)

#define CC_CALLBACK_3(__selector__,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 表示的是,回调函数中不事先指定参数是一个,而事先指定的回调函数的参数 可以任意多个。


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


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


假设回调函数:



// a selector callback

void menuCloseCallback(Object* pSender,int a,int b);


void HelloWorld::menuCloseCallback(Object* pSender,int b)

{

std::cout<<a<<" "<<b<<std::endl;

Director::getInstance()->end();

#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)

exit(0);

#endif

}

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

auto closeItem = MenuItemImage::create(

"Closenormal.png",this,2));



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


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


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



void MenuItem::activate()

{

if (_enabled)

{

if( _callback )

{

_callback(this);

}

if (kScriptTypeNone != _scriptType)

{

BasicScriptData data(this);

ScriptEvent scriptEvent(kMenuClickedEvent,&data);

ScriptEngineManager::getInstance()->getScriptEngine()->sendEvent(&scriptEvent);

}

}

}

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


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


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




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


如下:



auto closeItem = MenuItemImage::create(

"Closenormal.png",

std::bind(&HelloWorld::menuCloseCallback,

COCOS2DX CCCALLBACK的更多相关文章

  1. 详解使用postMessage解决iframe跨域通信问题

    这篇文章主要介绍了详解使用postMessage解决iframe跨域通信问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. html5录音功能实战示例

    这篇文章主要介绍了html5录音功能实战示例的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  3. ios – CLGeocoder错误. GEOErrorDomain代码= -3

    有没有关于apple的地理编码请求的文档?谢谢你提前.更新这是我的整个代码请求解决方法在搜索到答案后,它在Apples文档中!

  4. ios – 调用异步方法的方法的单元测试

    我想我有这些代码行:我想为该代码编写一个单元测试.对于initializeHomeData和initializeAnythingElse,我可以编写单元测试,如:我的问题是,如何测试reset()?我应该在testReset()中调用它们,如:但我认为这不是适当的实施.解决方法你是对的.要测试重置,您需要调用reset,而不是内部方法.话虽这么说,重置目前的编写方式使其不可测试.您能够如此轻松地测

  5. swift 的宏定义

    swift中没有了#Define这种宏定义了,可以用let来声明常量来取代,判断当前系统版本PS:这种用let替代#define只适用于一般的常量宏,如果是表达式或者其他复杂的宏,let也无能无力对于复杂表达式的宏,可以用全局的func函数代替,比如上面的两个系统判断,可以修改成下面的func这样,就可以在三元表达式中使用了:还有RGBA宏可以写成这样:

  6. swift - The Facade Pattern

    Facade(外观)模式为子系统中的各类提供一个简明一致的界面,隐藏子系统的复杂性,使子系统更加容易使用。

  7. swift - The Proxy Pattern

    我在实际工作中vc也仿照过Foundation的delegate:button:内涵业务逻辑,底层实现;每个button是一个类,业务逻辑需要未知的参数和处理之后未知的结果反馈UI:点击button之后界面的改变,UI实现未知的参数和未知的结果反馈,也就是实现这个代理这样以来UI的定制,很灵活很容易,代码思路依然清晰如初。哪个是主体哪个是代理并不重要关键是看定义所说whichisusedwhenanobjectisrequiredtoactasaninterfacetoanotherobjectorres

  8. swift开发笔记9 - 正向和反向页面传参

    在storyboa里segue是这样的:首先看考勤页面(主页面)如何给备注页面传参:在考勤页面(主页面)的viewcontroller中找到prepareForSegue方法,这个方法由xcode自动生成,用于在使用segue跳转前,做一些处理动作:实际上是通过修改segue的目标页面的某个类属性,从而达到传参的目的。

  9. swift之宏定义

    Swift里边没有oc一样的define了,没有原来一样的宏定义了要想实现一样的效果可以用let常量来代替1.判断系统2.负责一些的宏定义,需要使用func函数定义OK!

  10. 如何在Swift中使用预编译宏定义

    OC中这样使用即可解决方案Swift:mb的感觉soeasy~StackOverFlow

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部