供cocos2d-x通常使用的方法,我有一个好脸色。这项研究真的奖励。

向导首先,定义,实施一系列连续动作。

对于我们的行动能回调函数,我们必须申报并加以实施。

    void callBack();
void callBack_1(Node* node);
void callBack_2(Node* node,const char* str); void Nice::callBack()
{
log("Nice::callBack()");
}
void Nice::callBack_1(Node* node)
{
log("This tag is %d",node->getTag());
}
void Nice::callBack_2(Node* node,const char* str)
{
log("This tag is %d, and str is %s",node->getTag(), str);
} //然后就開始创建我们伟大的精灵了。
sp_area = Sprite::create("newAlwaysShow.png");//sp_area is a class member ,Sprite* ap_area;
sp_area->setTag(1314);
auto sp_another = Sprite::create();
sp_another->setTag(520);
addChild(sp_another);
addChild(sp_area);
//翻滚吧。可爱的精灵们!
/*
CallFunc 创建的函数必须是无參数的
CallFuncN 创建的函数能够是一个至多个參数
*/
sp_area->runAction(Sequence::create(MoveTo::create(2, Vec2(200, 200)),
CallFunc::create(CC_CALLBACK_0(Nice::callBack, this)),
CallFuncN::create(CC_CALLBACK_1(Nice::callBack_2, this, "I have tow parameter")),
NULL));
//这样写完之后。发现runAction里面的东西好平淡(平淡有时候才不是真呢! ),于是我就瞎写一通。残忍的把它变成了这个样子。
sp_area->runAction(Sequence::create(MoveTo::create(2, Vec2(200, 200)),
CallFunc::create(CC_CALLBACK_0(Nice::callBack, this)),
CallFunc::create([&]()->void{log("Done1");log("This tag is %d", sp_area->getTag());}),
CallFunc::create(std::bind(&Nice::callBack, this)),
CallFuncN::create(CC_CALLBACK_1(Nice::callBack_1, this)),//1314
CallFuncN::create(CC_CALLBACK_0(Nice::callBack_1, this, sp_another)),//520
CallFuncN::create([](Node* node){log("this tag is %d", node->getTag());}),//1314
CallFuncN::create(std::bind(&Nice::callBack_1, this, sp_area)),//1314
CallFuncN::create(std::bind(&Nice::callBack_1, this, sp_another)),//520
CallFuncN::create(CC_CALLBACK_1(Nice::callBack_2, this, "I have tow parameter")),
NULL));

好吧,他如今已经面目全非了。事实上这么多行也仅仅是用到了两种方法而已,第一种lambda表达式。另外一种std::bind()。

(尼玛,你在逗我吗?不是还有CC_CALLBACK_0吗?)

额。既然这样就让我们来看下源代码吧。

// new callbacks based on C++11
#define CC_CALLBACK_0(__selector__,__target__, ...) std::bind(&__selector__,__target__, ##__VA_ARGS__)
#define CC_CALLBACK_1(__selector__,__target__, ...) std::bind(&__selector__,__target__, std::placeholders::_1, ##__VA_ARGS__)
#define CC_CALLBACK_2(__selector__,__target__, ...) std::bind(&__selector__,__target__, std::placeholders::_1, std::placeholders::_2, ##__VA_ARGS__)
#define CC_CALLBACK_3(__selector__,__target__, ...) std::bind(&__selector__,__target__, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, ##__VA_ARGS__)

果不其然,宏仅仅是表面。内在还是偷偷的和std::bind勾搭在一起了!





我们来看下这个到处勾搭的宏是怎么定义的~~~

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

#define CC_CALLBACK_0( 方法,目标, 可变參数) std::bind(&取方法的地址,目标, 可变參数)

PS:在定义宏的时候是不能用...来当作前面的...作为可变參数的,必需要使用##__VA_ARGS__这个伟大的第三者来替换掉





可变參数在整个cocos-x中也还是用到非常多的。比方创建Sequence::create就是运动了,不然你怎么能在创建一连串的动作的时候传入那么多的參数的说。

那么接下来就来看看源代码吧。

(又是源代码- -)

Sequece::create是分平台实现的(分为win和其它)

#if (CC_TARGET_PLATFORM == CC_PLATFORM_WP8) || (CC_TARGET_PLATFORM == CC_PLATFORM_WINRT)
// WP8 in VS2012 does not support nullptr in variable args lists and variadic templates are also not supported
typedef FiniteTimeAction* M;
static Sequence* create(M m1, std::nullptr_t listEnd) { return variadicCreate(m1, NULL); }
static Sequence* create(M m1, M m2, std::nullptr_t listEnd) { return variadicCreate(m1, m2, NULL); }
static Sequence* create(M m1, M m2, M m3, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, M m5, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, m5, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, M m5, M m6, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, m5, m6, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, M m5, M m6, M m7, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, m5, m6, m7, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, M m5, M m6, M m7, M m8, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, m5, m6, m7, m8, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, M m5, M m6, M m7, M m8, M m9, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, m5, m6, m7, m8, m9, NULL); }
static Sequence* create(M m1, M m2, M m3, M m4, M m5, M m6, M m7, M m8, M m9, M m10, std::nullptr_t listEnd) { return variadicCreate(m1, m2, m3, m4, m5, m6, m7, m8, m9, m10, NULL); } // On WP8 for variable argument lists longer than 10 items, use the other create functions or variadicCreate with NULL as the last argument
static Sequence* variadicCreate(FiniteTimeAction* item, ...);
#else
static Sequence* create(FiniteTimeAction *action1, ...) CC_REQUIRES_NULL_TERMINATION;
#endif Sequence* Sequence::create(FiniteTimeAction *action1, ...)
{
va_list params;
va_start(params, action1); Sequence *ret = Sequence::createWithVariableList(action1, params);//传给了createWithVariableList让他来处理 va_end(params); return ret;
} Sequence* Sequence::createWithVariableList(FiniteTimeAction *action1, va_list args)
{
FiniteTimeAction *now;
FiniteTimeAction *prev = action1;
bool bOneAction = true; while (action1)
{
now = va_arg(args, FiniteTimeAction*);
//假设第二个动作存在
if (now)
{
prev = createWithTwoActions(prev, now);
bOneAction = false;
}
//假设第二个动作不存在
else
{
// If only one action is added to Sequence, make up a Sequence by adding a simplest finite time action.
if (bOneAction)
{
prev = createWithTwoActions(prev, ExtraAction::create());
}
break;
}
} return ((Sequence*)prev);
}

我们就依照他的思路来实现一个可变參数的函数

void manyArgument(int num, ...);
void Nice::manyArgument(int num, ...)
{
va_list params;
//va_start函数的第二个參数要和manyArgument的第一个參数一样
va_start(params, num); int now;
log("fist is %d", num);
while (true)
{
now = va_arg(params, int);
if (now != -1)
{
log("now is %d", now);
}
else
{
break;
}
} va_end(params);
}
manyArgument(1, 2, 3, 4, -1);

能够发现多个參数的的时候必需要有一个能够终止的条件(比如最后一个为-1)。可是假设在中途就要传入-1怎么办?

这个时候仅仅能把传入的參数改为int*

    void manyArgument(int* num, ...);

然后再把最后终止的条件换为!= null,可是这样在传入參数的时候就麻烦(有得必有失)

    void manyArgument(int* num, ...);
void Nice::manyArgument(int* num, ...)
{
va_list params;
//va_start函数的第二个參数要和manyArgument的第一个參数一样
va_start(params, num); int* now;
log("fist is %d", *num);
while (true)
{
now = va_arg(params, int*);
if (now != nullptr)
{
log("now is %d", *now);
}
else
{
break;
}
} va_end(params);
}
int arr[] {3,6,9,12};
manyArgument(arr, arr+1, arr+2, arr+3, nullptr);

接着试试定义一个可变參数得宏  ##__VA_ARGS__ 就代表着前面传入进来的可变參数

    #define MANYARGUMENT(__num__, ...) manyArgument(__num__, ##__VA_ARGS__)
MANYARGUMENT(arr, arr+1, arr+2, nullptr);

别忘记我们还有std::placeholders::_1 占位没有玩过呢= =

    auto add_num = std::bind(&Nice::add, this, std::placeholders::_1, std::placeholders::_2, 10);
log("sum is %d", add_num(1, 1));//12
log("sum is %d", add_num(1, 1, 2));//12
log("sum is %d", add_num(1, 1, 2, 4));//12
//不管怎么改变參数的第三位也是固定死了是10的,奇妙的是竟然能传入四个參数,明摆着就是仅仅认定前两个參数是有效的,后面的统统无效。

学到这里。

貌似对这个流程还是比較清楚的嘛。尽管可能还有很多其它的仅仅是没有发现,可是我们还有很多其它的时间没实用呀!

转摘请注明出处:http://blog.csdn.net/zhouyunxuan

版权声明:本文博主原创文章,博客,未经同意不得转载。

cocos2d-x 3.1.1 学习笔记[17] 关于这些活动功能的更多相关文章

  1. [XMPP]iOS聊天软件学习笔记[一]

    @import url(http://i.cnblogs.com/Load.ashx?type=style&file=SyntaxHighlighter.css);@import url(/c ...

  2. cocos2d-x 3.1.1学习笔记[23]寻找主循环 mainloop

    文章出自于  http://blog.csdn.net/zhouyunxuan cocos2d到底是怎样把场景展示给我们的,我一直非常好奇. 凭个人猜想,引擎内部的结构类似于这样 while(true ...

  3. cocos2d-x 3.1.1 学习笔记[3]Action 动作

    这些动画貌似都非常多的样子,就所有都创建一次. 代码例如以下: /* 动画*/ auto sp = Sprite::create("card_bg_big_26.jpg"); Si ...

  4. cocos2d-x 3.1.1 学习笔记[2]Sprite 精灵

    Sprite应该是用到最多的一个类吧.无法想像一个游戏没有精灵将怎样进行愉快的玩耍. Sprite继承于Node 和 TextureProtocol. Sprite是一个2d的图像. Sprite能够 ...

  5. cocos2d-x 3.1.1 学习笔记[21]cocos2d-x 创建过程

    文章出自于  http://blog.csdn.net/zhouyunxuan RootViewController.h #import <UIKit/UIKit.h> @interfac ...

  6. cocos2d-x 3.1.1 学习笔记[4]GridActions 网格动画

    文章写的  http://blog.csdn.net/zhouyunxuan 老样子.见代码. //GridActions can only used on NodeGrid auto nodeGri ...

  7. cocos2d-x 3.1.1 学习笔记[11] http请求 + json解析

    //http须要引入的头文件和命名空间 #include <network/HttpClient.h> using namespace network; //json须要引入的头文件 #i ...

  8. [XMPP]iOS聊天软件学习笔记[四]

    昨天完成了聊天界面,基本功能算告一段落 开发时间:五天(工作时间) 开发工具:xcode6 开发平台:iOS8 XMPP框架:XMPPFramework git clone https://githu ...

  9. [XMPP]iOS聊天软件学习笔记[三]

    今天做了好友界面,其实xmpp内部已经写好很多扩展模块,所以使用起来还是很方便的 开发时间:五天(工作时间) 开发工具:xcode6 开发平台:iOS8 XMPP框架:XMPPFramework gi ...

随机推荐

  1. Projecet客户端登陆无法通过验证

    客户反映使用Project客户端登陆project服务器的时候,只有域管理员账户才能够登陆成功,其他的账户登陆都无法验证通过,无论是https的方式还是客户端的方式,但是域账户登陆计算机是可以登陆成功 ...

  2. 福利 城市名的python list

    ["上海","北京","北京市","朝阳","朝阳区","海淀","元 ...

  3. Android数字签名解析(二)

    在Android数字签名解析(一)中,介绍了android进行签名的两种方式,当中用密钥对进行签名用到了signapk.jar这个java库. 以下我们就看看signapk签名实现过程,signapk ...

  4. java去全半角空格,trim(), replaceAll(" +",""),replaceAll("\\s*", ""), replaceAll(" | ", "")

    JAVA中去掉空格 . String.trim() trim()是去掉首尾空格 .str.replace(" ", ""); 去掉所有空格,包括首尾.中间 St ...

  5. windows phone (13) 样式继承

    原文:windows phone (13) 样式继承 在上一遍文章中已经介绍到可以在Resources集合中定义样式,我们也可以在一个样式上引用其他的样式,这就是继承的概念,使用方法是将引用的样式放置 ...

  6. NYOJ 47 河问题

    时间限制:1000 ms  |  内存限制:65535 KB 难度:5 描写叙述 在漆黑的夜里,N位旅行者来到了一座狭窄并且没有护栏的桥边.假设不借助手电筒的话,大家是不管怎样也不敢过桥去的.不幸的是 ...

  7. ECToch随笔

    1.去掉后台Powered by ECTouch.Cn mobile\include\apps\admin\view\index.php第五行<title>{$lang['cp_home' ...

  8. Leetcode 3Sum Closet

    二手和3Sum像几乎相同的想法.二进制搜索.关键修剪.但是,在修剪做出很多错误. 然后还有一个更加速了原来的想法O(n^2). #include<iostream> #include &l ...

  9. 对HGE游戏引擎的一次封装

    HGE游戏引擎是一个开源2D游戏引擎,基于directX. 它的渲染及逻辑是基于帧回调的框架模式, 其提供一些主要的图像操作和输入控制功能. 我在之前写一个2D游戏的时候对它整个框架进行了一次封装,非 ...

  10. [Elasticsearch] 控制相关性 (一) - 后面的相关度分值理论计算

    从第一章翻译Elasticsearch官方指南Controlling Relevance一章. 控制相关度(Controlling Relevance) 对于仅处理结构化数据(比方日期.数值和字符枚举 ...