0.  placeholder

头文件:<functional>

namespace: placeholder

placeholder 就是一堆帮助bind占参数位置的东西,名字分别为 _1, _2, _3, ..., _N。 它的作用是告诉编译器,这个参数我目前用它来占个坑,以后填。

1. std::function

为了讲丑陋的函数指针简化,也为了提供现代语言一般都有的闭包,C++ 11提供了std::function类型来表示函数,同时也可以用来表示lambda 表达式(等同于Objective-C中的代码块,Java中的闭包,诸如此类),lambda表达式可以认为是将一段代码封装成一个对象使用,虽然这个解释更适合Objective-C的代码块。在调试时,lambda表达式实际上就是编译器在自动生成的匿名类中加入的成员函数。举个栗子:

    this->a = ;
this->b = ; int sum = ; auto funcSumMember = [=](){
return (this->a + this->b);
}; sum = funcSumMember(); log("Sum member: %d", sum); auto funcSumAB = [](int a, int b) {
return (a+b);
}; sum = funcSumAB( , ); log("a+b:%d", sum); auto funcModifyMember = [&](){
this->a = ;
this->b = ; return (this->a+this->b);
}; sum = funcModifyMember(); log("a(%d) + b(%d) = %d", this->a, this->b, sum);

int ta = 3;

int tb = 5;

auto funcModifyMemberAdvanced = [ta, &tb](){

//ta = 10; unable to assign value to ta for this is a copy only

tb = 10;

return (ta+tb);

};

sum = funcModifyMemberAdvanced();

log("ta(%d) + tb(%d) = %d", ta, tb, sum);

开始的中括号[]内部放的是capture list,指定以什么形式捕获外界的值。

a. []表示这个表达式不需要知道外界的数值,见 funcSumAB(int,int),由于需要用于加法的参数都已经在参数列表中给出,无需访问外部值。

b. [=]表示这个表达式如果需要外部数值,那么都以拷贝形式放进来,就像函数参数一样,对其修改不会影响到外部的本体。参考 funcSumMember()

c. [&]表示这个表达式如果需要外部数值,那么都以引用形式放进来,对外部数据的修改是真实有效的,参考funcModifyMember()

d. []列表中可以指定更加详细的方式,例如对变量ta不做修改,只要知道数值就好,对tb可能做出修改,那么就可以: [ta, &tb] ,虽然先前=表示将值拷贝过来,这里ta真的不能写成=ta :(. 另外我也没找到可以在这个列表中带入this->a 和this->b的正确写法,所以只好使用了局部变量ta和tb,参见 funcModifyMemberAdvanced.

目前我看来lambda表达式的好处有:

1. 减少无聊的代码:对一些只有一两行的简单函数,无需在头文件中声明然后在.cxx中实现,然后才能用作函数指针来访问,譬如对每个MenuItem的响应,不长的函数根本没必要郑重的单独拖出来。又如排序用到的compare函数,如果放在以前,用重载操作符的内部类已经算比较好的办法了,现在只需要一个lambda表达式(当然本质上也是利用了内部类)

2. 可以更灵活的将类的内部暴露给其它类。利用capture list, 可以访问到本类中的所有内容,如果将其传递给另一个类,那么另一个类就可以利用这些内容。相对于friend之类的方式,更加灵活宽容

3. 更灵活的代码重用。 现在对于重复的代码,你可以不必一定声明出一个函数来,说它其实只是用来避免重复的,你也可以将它放到一个lambda表达式里,然后随处可用

2. 参数绑定 std::bind

头文件: <functional>

参数: std::bind( Func & func, Args ... args )

返回值: std::function( Args ... args )

Func 是一个std::function类型的函数指针,bind可以绑定参数,使得返回的新函数指针无需带上完整的参数列表.对于缺了具体数值的货色,需要用0. placeholder一节中提到的placeholder来凑数:

    {
int sum = ;
auto funcSum = [](int a, int b, int c){
return (a+b+c);
}; auto funcSumBC = std::bind( funcSum, , std::placeholders::_1, std::placeholders::_2 );
sum = funcSumBC( , );
log("sumBC=%d", sum); auto funcOnlyC = std::bind( funcSum, , , std::placeholders::_1); sum = funcOnlyC();
log("OnlyC=%d", sum); }

需要注意的是,C++中的成员函数,是有自己的this指针的(当然不包括静态函数),对于这部分函数,编译器做的处理实际上是:

this->funcA(arg0, arg1, ...);

变成了:

__XX_funcA( this, arg0, arg1, ... )

这部分内容参考 Inside C++ Object Model

所以成员函数第一个参数是this指针,使用bind的时候千万不要压住人家隐形的翅膀,不然会挂掉的 XD.

cocos2dx中的应用:

CC_CALLBACK_0

CC_CALLBACK_1

...

CC_CALLBACK_N

N表示bind需要用几个placeholder凑数,也就是bind后调用需要带的参数数量。如果你全绑了,那就用CC_CALLBACK_0,缺了1个参数,就用_1,以此类推,不要忘记成员函数的this参数!

3. auto

auto可以认为是"你懂的"。考虑到某些名字实在是长得要命,譬如 cocos2d::extension::Control::EventType 这种逆天而又实在无聊的前缀,还有STL中更常见的 vector<int>::iterator这种让人不得不typedef的类型,auto拯救我等于水火之中。

用于返回值的赋值是比较理想的场合:

auto iB = something.begin(); //我知道返回的类型是什么,大家都知道,但我实在懒得打出来了!

但是让编译器去猜我们自己初始化的变量到底是什么类型就有点不靠谱:

auto factor = 1; //你知道factor到底是int,float还是double还是其它吗?

scale *= factor;

或许编译器的一些规则可以让我们有所判断,但我感觉这种情况下用auto,对可读性是一种阻碍。

注意:虽然带着auto,但是它跟auto_ptr真不是亲戚

4. 迭代

    {
int localArray[] = {, , , , }; for( int iLocal: localArray)
{
log("%d", iLocal);
} std::string localStr = "Hello C++11"; for( auto iStr: localStr)
{
log("%c", iStr);
}
}

for( ReferenceElement : container )

{

...

}

能够使用迭代的条件有4个:

1、实现begin()
     2 、实现end()
     3 、实现 operator++()
     4 、实现 operator!=(class& other)

跟其它语言的遍历一样,用于无需关心迭代到什么地方的场景,如果关心迭代的index,还是老式的for更合适

5. override/final

如果一个函数带有override关键字,那么它必须符合以下两个条件,否则编译将报错:

1. 它是一个虚成员函数,也就是带virtual的

2. 它改写了父类中的方法,也就是说父类中必须也有一个同样的方法(返回类型,函数名,参数列表)

那么它有什么用呢?作用就是在子类改写了父类的方法时,能够有这么一个关键字标志,提醒你这个funcA调用的到底是哪个类的funcA。个人观点认为,有这个关键字之后,编译器对于带有override关键字的函数所发生的调用,就无需去寻找上一层的虚函数表,应该可以用于提高虚函数调用的效率。嗯,由于对这个领域不感兴趣,这部分纯属脑补 XD.

如果一个函数带有final关键字,那么它必须符合以下两个条件,否则编译将报错:

1. 它是一个虚成员函数,也就是带virtual的

2. 它的一切子类都不重写这个方法

它的作用与Java的final相同,不再做讨论

目前暂时记录这么多,如有错漏以后补

cocos2dx 中使用的一些C++ 11 特性的更多相关文章

  1. C++11中对类(class)新增的特性

    C++11中对类(class)新增的特性 default/delete 控制默认函数 在我们没有显式定义类的复制构造函数和赋值操作符的情况下,编译器会为我们生成默认的这两个函数: 默认的赋值函数以内存 ...

  2. Cocos2dx中线程优先级

    Cocos2dx中线程优先级问题 不论是ios还是android,遇到耗时的任务都要另起线程处理,否则程序不能及时用户的反馈.游戏中如果一圈循环不能在1/frameRate(帧率是30则1/30)秒内 ...

  3. cocos2d-x 中的基本概念

    在 cocos2d-x 开头配置(Windows 平台)中,介绍了新建工程,这篇就介绍下 cocos2d-x 的一些概念.(前提是需要有C++的面向对象的基本知识和C++11的常用知识) 层,场景,导 ...

  4. WP8:在Cocos2d-x中使用OpenXLive

    一.    Cocos2d-x for Windows Phone 到2013年底,几大手游引擎都陆续支持WP8了,特别是Unity3D和Cocos2d-x.有过游戏开发经验的朋友们应该对这两个引擎不 ...

  5. C++开发者都应该使用的10个C++11特性

    转载自http://blog.jobbole.com/44015/ 在C++11新标准中,语言本身和标准库都增加了很多新内容,本文只涉及了一些皮毛.不过我相信这些新特性当中有一些,应该成为所有C++开 ...

  6. [转载]TexturePacker 如何使用自带的加密功能及在cocos2dx中的使用

    在cocos2dx中使用纹理图集是非常节省资源的,在这里推荐 TexturePacker,而且 TexturePacker工具的加密接口也非常的好用,下面就来介绍一下... TexturePacker ...

  7. cocos2dx中的三种基本的数据类型

    cocos2dx中提供了三种基本的数据类型:CCString(字符串),CCArray(数组),CCDictionary(数据字典(哈希的功能)) 2.CCString的用法 class  CCStr ...

  8. 转载:每个C++开发者都应该使用的十个C++11特性

    这篇文章讨论了一系列所有开发者都应该学习和使用的C++11特性,在新的C++标准中,语言和标准库都加入了很多新属性,这篇文章只会介绍一些皮毛,然而,我相信有一些特征用法应该会成为C++开发者的日常用法 ...

  9. C++开发者都应该使用的10个C++11特性 转

    http://blog.jobbole.com/44015/// | 分类: C/C++, 开发 | 条评论 | 标签: C++, C语言 分享到: 本文由 伯乐在线 - 治不好你我就不是兽医 翻译自 ...

随机推荐

  1. Android 三级联动选择城市+后台服务加载数据库

    技术渣,大家将就着看 首先我们需要一个xml数据保存到数据库,这里我从QQ下面找到一个loclist.xml文件 <CountryRegion Name="中国" Code= ...

  2. C++11 智能指针

    C++ 11标准库引入了几种智能指针 unique_ptr shared_ptr weak_ptr C++内存管理机制是当一个变量或对象从作用域过期的时候就会从内存中将他干掉.但是如果变量只是一个指针 ...

  3. JavaScript引擎LHS查找和RHS查找

    要想真正理解Javascript脚本中每一句代码的执行过程,需要弄清楚几个基本概念:1.引擎,从头到尾负责整个 JavaScript 程序的编译及执行过程.2.编译器,引擎的好朋友之一,负责语法分析及 ...

  4. PSP记录个人项目耗时情况

    四则运算编程 PSP记录个人项目耗时情况 PSP Personal Software Process Stages Time(%) Planning 计划 7 Estimate 估计这个任务需要多少时 ...

  5. sql语句判断默认值为getdate()的约束是否存在

    TFlowCreateTask--表名 AddDate--字段名 if not exists( select d.name from syscolumns a join sysobjects b on ...

  6. MySql学习(六) —— 数据库优化理论(二) —— 查询优化技术

    逻辑查询优化包括的技术 1)子查询优化  2)视图重写  3)等价谓词重写  4)条件简化  5)外连接消除  6)嵌套连接消除  7)连接消除  8)语义优化 9)非SPJ优化 一.子查询优化 1. ...

  7. java selenium (三) 环境搭建 基于Maven

    现在Java的大部分项目都是基于Maven,  在Maven项目中使用Selenium2. 非常简单. 首先你需要配置好Maven的环境 可以参考本博客的Maven教程系列,Maven入门教程(一) ...

  8. Asp.net页面引用SAP IQ 16 iAnywhere.Data.SQLAnywhere.V4.0.dll报错,语言文件没找到

    参考http://sqlanywhere-forum.sap.com/questions/20420/saconnection-threw-an-exception-cannot-find-the-l ...

  9. 非maven项目导入idea几点心得总结

    这个问题一共有3种解决办法1. 你下载好的文件应该是src 和Webcontent是在同一个目录下的.只要把Webcontent放入到src下就行.这种就变成maven的目录结构.因为你这个目录结构应 ...

  10. Android按钮的各个样式设置

    安卓开发学习之014 Button应用详解(样式.背景.按钮单击.长按.双击.多击事件) 一.Button简介 按钮也是继承自TextView 二.XML定义方法 <Button android ...