Cocos2d-x v3.0 新的事件调度方法 lambda表达式的使用

2024-08-22 15:48

本文主要是介绍Cocos2d-x v3.0 新的事件调度方法 lambda表达式的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

欢迎加入 Cocos2d-x 交流群: 193411763


转载请注明原文出处:http://blog.csdn.net/u012945598/article/details/24603251



Cocos 2d-x 3.0 版本中引入了C++ 11的特性。其中就包含了回调函中使用Lambda对象。

下面我们来看一段TestCpp中的代码:



在上图的触摸事件的回调函数中,共使用了三次Lambda表达式:

                                     [ ](Touch * touch,Event * event){ };

下面我们就来介绍一下Lambda表达式的使用方法。


正常情况下,如果我们需要在很多地方使用相同的操作,通常应该定义一个函数来实现这个功能。

而有些时候,我们只需要在一两个地方使用到一些简单的操作,而又不想去定义这个函数名,那么此时便可以Lambda表达式来实现我们的功能。

一个完整的lambda表达式的表达形式如下:

                                    [capture list](parameter list)->return type (function body) 

                                    [捕获列表]   (参数列表)        ->返回类型    (函数体) 


那么为什么图中的Lambda表达式的形式与上述的形式不一样呢?

原因是 Lambda表达式的参数列表回类型可以忽略的,但是捕获列表函数体一定要包含。


也就是说,Lambda表达式实际上就是一个匿名函数,它的优点与内联函数(又称内嵌函数、内置函数)类似,但Lambda表达式可能会定义在函数内部。

那么内联函数的优点是什么呢?我们举个例子来说明,比如我们的程序中有这样一段代码:

void A(){

       ....//函数体略

}

void main(){

        ....  //省略

        A( );//调用A函数

       ....  //省略

}

上述代码执行的主要过程如下:

1.主调函数main执行完调用A函数前的语句后,在转去调用A函数前,首先需要记录当前执行的指令地址,也就是做一个“保护现场”的操作,用于执行完A函数后继续执行后续代码。

2.然后流程的控制会被转移到A函数的入口,并且执行A函数中的函数体内的语句.

3.执行完成后,流程才会返回到之前记录的地址处,并且根据之前所记录的信息做"恢复现场"操作,保证程序正常执行。


上述过程的每一个操作都需要花费一定的时间,如果A函数需要被频繁的使用,那么我们花费的时间就会很长,从而造成效率降低。

为了解决这个问题,C++为我们提供的了内联函数,所谓内联函数,就是通过将一个函数声明为inline function,从而达到在编译的过程中,直接将所调用的函数的函数体部分直接拷贝到主调函数,而不需要将流程转到这个函数中去,以此来减少程序的运行时间。

这是因为当一个函数的函数体规模很小的时候,函数调用过程中的时间开销会超过执行函数所需要的时间。


这就是使用内联函数的好处,而对于Lambda表达式,我们可以将它理解为一个未命名的内联函数


下面我们对lambda表达式的形式进行逐一分析:

1.“  [捕获列表]  ”

首先我们观察一下上图中的第一个lambda表达式与第三个lambda表达式的捕获列表部分的区别。

可以看到,上图的第一个表达式中捕获列表为空 [ ],而第三个表达式中的捕获列表中包含了一个等号 [=]。


下面我们再观察一下上图中第一个与第三个lambda表达式的函数体内都使用到了哪些变量。

可以看到,第一个表达式中所有的变量,均是在Lambda表达式中定义的(log除外,因为log函数包含在头文件中),

而在第三个表达式中所使用到的sprite1,sprite2等变量,并不是在lambda表达式中定义的,而是当前函数中或是当前类中的变量。

那么我们就可以总结出,在Lambda表达式的函数体内,是不能够访问到外部的变量的,如果想要使用函数体外定义的变量,就需要将它们进行"捕获",上图第三个lambda表达式采用的正是“值捕获”,与它对应的另外一种为“引用捕获”。

[ ]:空捕获列表,即lambda表达式不能够使用所在函数中的变量

[=]:值捕获,即lambda表达式可以以拷贝的方式访问到函数中变量的值

[&]:引用捕获,即lambda表达式中所使用的其所在函数中的变量均是引用方式

当我们不希望在捕获的时候将所有的变量都捕获的时候,我们可以使用如下的方式进行捕获,例如:

[=sprite1,&sprite2]

这里我们仅仅捕获了两个变量,第一个变量是以值拷贝的方式捕获,第二个是以引用方式捕获,变量与变量之间用逗号分隔。


正常情况下,如果一个变量是值拷贝,Lambda不能改变它的值,如果我们希望改变一个值拷贝的变量的值,就需要在参数列表前加上关键字mutable

例如:
         auto s1=10;

         auto s2=[=s1](){return ++s1};//错误,因为s1是值拷贝,不能改变s1的值

         auto s2=[=s1]() mutable {return ++s1};//正确 

2.(参数列表)

Lambda表达式传递参数时需要注意的是,Lambda表达式不能有默认参数,也就是说Lambda表达式的实参数与形参数必须相等。

其他情况Lambda表达式的参数部分与普通函数并无区别,一般会结合STL使用。

例如:

void Test(){

      vector<int> myVec;   //创建一个int 类型容器

      myVec.push_back(1); //插入数据 1

      myVec.push_back(2);//插入数据 2

      

      int a=10;                   //创建局部变量 a

      for_each(myVec.begin(),myVec.end(),[&](int v)mutable(cout<<v+a<<endl;a++)); //将容器中元素作为参数传到lambda表达式 输出a+v结果为 11 13

       cout<<a<<endl;   //输出a 结果为12

}

3.->return type

之间我们已经提到,Lambda的返回值是可以省略的。

原因是编译器会根据return的类型来推导返回值,但是如果需要return后再做一个类型转换,我们就可以通过写一个返回类型来完成。

例如:cout<<[](float f){return f}(1.5); //这里我们将1.5作为参数传入并打印,返回结果就是实参的值1.5

cout<<[](float f)->int{return f}(1.5); //我们将返回值强制转换为int 输出结果为1


4.函数体

   函数体部分与普通函数并无区别,我们只需要注意以上几点,在函数体部分就不会出现问题。


   现在再回头看看TestCpp中的触摸事件,我们就可以明白其中的道理了。

这篇关于Cocos2d-x v3.0 新的事件调度方法 lambda表达式的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1096708

相关文章

python管理工具之conda安装部署及使用详解

《python管理工具之conda安装部署及使用详解》这篇文章详细介绍了如何安装和使用conda来管理Python环境,它涵盖了从安装部署、镜像源配置到具体的conda使用方法,包括创建、激活、安装包... 目录pytpshheraerUhon管理工具:conda部署+使用一、安装部署1、 下载2、 安装3

Mysql虚拟列的使用场景

《Mysql虚拟列的使用场景》MySQL虚拟列是一种在查询时动态生成的特殊列,它不占用存储空间,可以提高查询效率和数据处理便利性,本文给大家介绍Mysql虚拟列的相关知识,感兴趣的朋友一起看看吧... 目录1. 介绍mysql虚拟列1.1 定义和作用1.2 虚拟列与普通列的区别2. MySQL虚拟列的类型2

使用MongoDB进行数据存储的操作流程

《使用MongoDB进行数据存储的操作流程》在现代应用开发中,数据存储是一个至关重要的部分,随着数据量的增大和复杂性的增加,传统的关系型数据库有时难以应对高并发和大数据量的处理需求,MongoDB作为... 目录什么是MongoDB?MongoDB的优势使用MongoDB进行数据存储1. 安装MongoDB

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

mysql数据库分区的使用

《mysql数据库分区的使用》MySQL分区技术通过将大表分割成多个较小片段,提高查询性能、管理效率和数据存储效率,本文就来介绍一下mysql数据库分区的使用,感兴趣的可以了解一下... 目录【一】分区的基本概念【1】物理存储与逻辑分割【2】查询性能提升【3】数据管理与维护【4】扩展性与并行处理【二】分区的

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

Linux使用fdisk进行磁盘的相关操作

《Linux使用fdisk进行磁盘的相关操作》fdisk命令是Linux中用于管理磁盘分区的强大文本实用程序,这篇文章主要为大家详细介绍了如何使用fdisk进行磁盘的相关操作,需要的可以了解下... 目录简介基本语法示例用法列出所有分区查看指定磁盘的区分管理指定的磁盘进入交互式模式创建一个新的分区删除一个存

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Window Server2016加入AD域的方法步骤

《WindowServer2016加入AD域的方法步骤》:本文主要介绍WindowServer2016加入AD域的方法步骤,包括配置DNS、检测ping通、更改计算机域、输入账号密码、重启服务... 目录一、 准备条件二、配置ServerB加入ServerA的AD域(test.ly)三、查看加入AD域后的变

Window Server2016 AD域的创建的方法步骤

《WindowServer2016AD域的创建的方法步骤》本文主要介绍了WindowServer2016AD域的创建的方法步骤,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、准备条件二、在ServerA服务器中常见AD域管理器:三、创建AD域,域地址为“test.ly”