加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

Cocos2dx中常用的C++11特性

发布时间:2020-12-14 16:43:14 所属栏目:百科 来源:网络整理
导读:在 Cocos2dx3.x 中提供了对 C++11 标准的支持 , 其中 Lambda 表达式、 function 模板、 bind 模板在游戏编程中比较常用,本文简要介绍一下这三个 C++11 方面的特性。 1.Lambda 表达式 (1)Lambda 表达式语法格式: [ 捕捉列表 ]( 参数列表 )mutable- 返回值类

Cocos2dx3.x中提供了对C++11标准的支持,其中Lambda表达式、function模板、bind模板在游戏编程中比较常用,本文简要介绍一下这三个C++11方面的特性。

1.Lambda表达式

(1)Lambda表达式语法格式:[捕捉列表](参数列表)<mutable>->返回值类型{函数体}
(1.1).
捕捉列表:捕捉列表能够捕捉上下文的变量供lambda函数使用。捕捉列表有如下几种形式:
[var]:
按值传递方式捕捉变量var
[=]:
按值传递方式捕捉父作用域所有变量,包括this
[&]:
按引用传递方式捕获父作用域所有变量,包括this
[&var]:
按引用传递方式捕获变量var
[this]
:按值传递方式捕获当前的this指针,这样就可以使用lambda表达式所在类的所有成员
(1.2).
参数列表:lambda函数参数
(1.3).mutable:
默认lambda函数是const的,如果你不需要const的,加上mutable取消其常量性
(1.4).
返回值类型:如果不需要返回值类型,可以和->一起省略
(1.5).
函数体:可以使用参数列表的参数和捕捉列表的参数(2)Lambda表达式程序示例

#include <iostream>

using namespace std;

class TestLambda
{
public:
	void testFunc();
public:
	int data;
};


void TestLambda::testFunc()
{
	int var1 = 1;
	int var2 = 2;

	/*无捕捉、无参数lambda函数*/
	auto fun1=[](){std::cout << "无捕捉、无参数lambda函数" << endl; };
	fun1();

	/*捕捉this指针*/
	auto fun2=[this](){
		this->data = 10;
	};
	fun2();

	/*用值传递方式捕捉特定变量*/
	auto fun3=[var1]()mutable{
		std::cout << "用值传递方式捕获到了"<<var1 << endl;
		var1 = 11;
	};
	fun3();
	std::cout << "var1改变了吗?" + var1 << endl;


	/*用值传递方式捕获所有变量,包括this*/
	auto fun4=[=](){
		std::cout << var1 << " " << var2 << " " << this->data << endl;
	};
	fun4();

	/*用引用传递捕获特定变量*/
	auto fun5=[&var2](){
		std::cout << var2 << endl;
		var2 = 22;
	};
	fun5();
	std::cout << "var2变了吗?"<<var2 << endl;

	/*用引用传递捕获所有变量,包括this*/
	auto fun6=[&](){
		std::cout << var1 << " " << var2 << " " << this->data << endl;
	};
	fun6();

	/*带参数的lambda*/
	auto fun7=[=](int var3){
		std::cout << "var1+var2+var3=" << var1 + var2 + var3 << endl;
	};
	fun7(3);

	/*带参数和返回值的lambda*/
	auto fun8 = [this](int var4)->int{
		return this->data - var4;
	};
	std::cout << fun8(10) << endl;

}


int main()
{
	TestLambda t1;
	t1.data = 20;
	t1.testFunc();
	return 0;
}

2.function模板:function是一个函数包装模板类,提供了泛型回调机制。function和函数指针比较类似,优点在于它允许用户在目标的实现上具有更大的弹性,即允许目标是普通函数,仿函数对象,类成员函数,甚至lambda表达式。

(1).function对象声明:function<返回值类型(参数类型)> 对象名

例:function<int(int,int)>f;//声明了一个返回值类型为int,两个参数皆为intfunction对象f

特别地,由于类的成员函数含有一个隐含的参数this指针,function对象声明如下:(类,参数列表)> f;

例:classCAdd

{

public:

int add(int a,int b){returna+b;}

}

function<int(CAdd,int,int)> f=&CAdd::add;

CAdd c;

f(c,2,3);

(2)function模板使用举例:

#include <iostream>
#include<functional>

using namespace std;

class TestFunction
{
public:
	int add(int a,int b)
	{
		return a + b;
	}

	static void test()
	{
		cout << "类静态成员函数" << endl;
	}
};

int max(int a,int b)
{
	return a > b?a:b;
}

int main()
{
	/*普通函数*/
	function<int(int,int)> f1 = max;
	cout << f1(2,3) << endl;

	/*类成员函数*/
	function<int(TestFunction,int)> f2 = &TestFunction::add;
	TestFunction t;
	cout << f2(t,3) << endl;

	/*类静态函数*/
	function<void()> f3 = TestFunction::test;
	f3();

	/*lambda表达式*/
	function<int(int,int)> f4 = [](int a,int b)->int{
		return a + b;
	};
	cout<<f4(2,3)<<endl;

	return 0;
}

3.bind模板:bind是一个用于函数绑定的模板。bind可以绑定普通函数,仿函数,类成员函数等等。对于某个函数进行绑定时,可以指定全部或部分参数,还可以调整参数的顺序。对于未指定的参数,可以用占位符std::placeholders::_1,std::placeholders::_2,std::placeholders::_3...来代替,其中_1表示参数表里的第一个参数。(1).bind使用示例:

#include<iostream>
#include<functional>

using namespace std;

class TestBind
{
public:
	int add(int a,int b)
	{
		return a + b;
	}
};


void fun1(int a,float b,char c)
{
	cout << a << 't' << b << 't' << c << endl;
}

class MaxFunctor
{
public:
	int operator()(int a,int b)
	{
		return a > b ? a : b;
	}
};

int main()
{
	/*普通函数*/
	auto testBind1 = bind(fun1,placeholders::_1,placeholders::_2,placeholders::_3);
	testBind1(1,1.0f,'1');
	/*使用占位符*/
	auto testBind2 = bind(fun1,std::placeholders::_1,'2');
	testBind2(2.0f,2);

	/*仿函数对象*/
	MaxFunctor max;
	auto testBind3 = bind(max,placeholders::_2);
	cout << testBind3(2,3) << endl;

	/*类成员函数*/
	TestBind t;
	auto testBind4 = bind(&TestBind::add,t,placeholders::_2);
	cout << testBind4(2,3) << endl;

	return 0;
}

4.综合示例:利用function,bind等特性实现一个处理输入的委托。

#include<iostream>
#include<functional>
#include<vector>

using namespace std;

class Event//仅供测试
{

};

class Touch//仅供测试
{

};

class InputHandler
{
public:
	static InputHandler* getInstance();
	void onTouch(Event*,Touch*);
	friend void operator+=(InputHandler*,function<void(Event*,Touch*)>);

private:
	InputHandler();
	virtual ~InputHandler();

private:
	static InputHandler* instance;
	vector<function<void(Event*,Touch*)>> myDelegate;
};

InputHandler* InputHandler::instance = new InputHandler;

InputHandler::InputHandler()
{

}

InputHandler::~InputHandler()
{
	if (instance != nullptr)
		delete instance;
}

InputHandler* InputHandler::getInstance()
{
	return instance;
}

void operator+=(InputHandler* in,Touch*)> f)
{
	in->myDelegate.push_back(f);
}

void InputHandler::onTouch(Event* e,Touch* t)
{
	for (auto f : myDelegate)
	{
		f(e,t);
	}
}

/*类成员函数*/

class TestInput
{
public:
	void fun1(Event* e,Touch* t)
	{
		cout << "fun1被调用" << endl;
	}
};

/*仿函数对象*/

class TestFunctor
{
public:
	void operator()(Event* e,Touch* t)
	{
		cout << "仿函数对象被调用" << endl;
	}
};

/*普通函数*/
void fun2(Event* e,Touch* t)
{
	cout << "fun2被调用" << endl;
}

int main()
{
	Event* e = new Event;
	Touch* t = new Touch;
	InputHandler* input=InputHandler::getInstance();

	/*类成员函数*/
	TestInput testInput;
	input += bind(&TestInput::fun1,testInput,placeholders::_2);
	
	/*仿函数对象*/
	TestFunctor testFunctor;
	input += bind(testFunctor,placeholders::_2);

	/*普通函数*/
	input += bind(fun2,placeholders::_2);

	/*lambda表达式*/
	input += [](Event* e,Touch* t){
		cout << "lambda表达式被调用" << endl;
	};

	input->onTouch(e,t);
	delete e;
	delete t;
	return 0;
}

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读