Cocos2d-x 3.0final 终结者系列教程12-Vector&map&value
北京时间昨天下午,温40度。中午12:16我来到了篮球场点。思维1分钟决定开站
转球:
我和另一个3队友半开始,
我手中的球的那一刻我突然火爆球不稳,突然问,淡淡的味道橡胶和烧烤的味道混合。
个腾空跳投,
球---------爆炸了。。。。。。
。
。
听新闻说昨天在路上都是 “熟人”
一位老大爷不慎被车刮倒了,大爷二话没说立刻爬了起来,围观众人议论纷纷:
“大爷人不错”,“大爷素养真高”。“大爷身体可真好”
大爷说:
“快拉 秦皇岛吧。烫腚”。
预计昨天又有不少沿街乞讨人士和流浪狗失去生命(咋没人管管)
------------------------------------------------切割线---------------------------------------------------
在做Cocos2d-x2.0人的人们预计正在使用CCArray,CCDirectionary,CCDouble,CCFloat这些正搞得非常爽,但不得不说
它们在V3.0中不用了。
看下3.0中的数据结合类Vector,Map,Value(建议大家自行阅读源代码,本文仅仅介绍怎样使用,关于引用计数的概念推荐大家看我的视频)。
下面案例来自TestCpp:
void TemplateVectorTest::onEnter()
{
UnitTestDemo::onEnter();
Vector<Node*> vec;//定义一个向量。事实上是对stl Vector的封装,仅仅是增加了对象的内存计数管理
CCASSERT(vec.empty(),
"");
CCASSERT(vec.capacity() ==
0, "");
CCASSERT(vec.size() ==
0, "");
CCASSERT(vec.max_size() >
0, "");
auto node1 = Node::create();
node1->setTag(1);
vec.pushBack(node1);//在向量加入一个元素
CCASSERT(node1->getReferenceCount() ==
2, "");
auto node2 = Node::create();
node2->setTag(2);
vec.pushBack(node2);//加入元素 编号会从0開始
CCASSERT(vec.getIndex(node1) ==
0, "");
CCASSERT(vec.getIndex(node2) ==
1, "");
auto node3 = Node::create();
node3->setTag(3);
vec.insert(1, node3); //插入元素
CCASSERT(vec.at(0)->getTag() ==
1, "");
CCASSERT(vec.at(1)->getTag() ==
3, "");
CCASSERT(vec.at(2)->getTag() ==
2, "");
// Test copy constructor
Vector<Node*> vec2(vec); //复制一个向量
CCASSERT(vec2.size() == vec.size(),
"");
ssize_t size = vec.size();
for (ssize_t i =
0; i < size; ++i)
{
CCASSERT(vec2.at(i) == vec.at(i),
"");
CCASSERT(vec.at(i)->getReferenceCount() ==
3, "");
CCASSERT(vec2.at(i)->getReferenceCount() ==
3, "");
}
// Test copy assignment operator
Vector<Node*> vec3;
vec3 = vec2;
CCASSERT(vec3.size() == vec2.size(),
"");
size = vec3.size();
for (ssize_t i =
0; i < size; ++i)
{
CCASSERT(vec3.at(i) == vec2.at(i),
"");
CCASSERT(vec3.at(i)->getReferenceCount() ==
4, "");
CCASSERT(vec2.at(i)->getReferenceCount() ==
4, "");
CCASSERT(vec.at(i)->getReferenceCount() ==
4, "");
}
// Test move constructor (Lambda 表达式,定义一个函数用于创建一个Vector,关于Lambda建议看我的视频)
auto createVector = [this](){
Vector<Node*> ret;
for (int i =
0; i < 20; i++)
{
ret.pushBack(Node::create());
}
int j = 1000;
for (auto& child : ret)
{
child->setTag(j++);
}
return ret;
};
Vector<Node*> vec4(createVector());
for (const
auto& child : vec4)
{
CC_UNUSED_PARAM(child);
CCASSERT(child->getReferenceCount() ==
2, "");
}
// Test init Vector<T> with capacity
Vector<Node*> vec5(10);
CCASSERT(vec5.capacity() ==
10, "");
vec5.reserve(20);
CCASSERT(vec5.capacity() ==
20, "");
CCASSERT(vec5.size() ==
0, "");
CCASSERT(vec5.empty(),
"");
auto toRemovedNode = Node::create();
vec5.pushBack(toRemovedNode);
CCASSERT(toRemovedNode->getReferenceCount() ==
2, "");
// Test move assignment operator
vec5 = createVector();
CCASSERT(toRemovedNode->getReferenceCount() ==
1, "");
CCASSERT(vec5.size() ==
20, "size should be 20");
for (const
auto& child : vec5) //这也是C++11新特性,好用吧,不像STL用迭代器那么麻烦
{
CC_UNUSED_PARAM(child);
CCASSERT(child->getReferenceCount() ==
2, "");
}
// Test Vector<T>::find
CCASSERT(vec.find(node3) == (vec.begin() +
1), "");
CCASSERT(std::find(std::begin(vec),
std::end(vec), node2) == (vec.begin() +
2), "");
CCASSERT(vec.front()->getTag() ==
1, "");
CCASSERT(vec.back()->getTag() ==
2, "");
CCASSERT(vec.getRandomObject(),
"");
CCASSERT(!vec.contains(Node::create()),
"");
CCASSERT(vec.contains(node1),
"");
CCASSERT(vec.contains(node2),
"");
CCASSERT(vec.contains(node3),
"");
CCASSERT(vec.equals(vec2),
"");
CCASSERT(vec.equals(vec3),
"");
// Insert
vec5.insert(2, node1);
CCASSERT(vec5.at(2)->getTag() ==
1, "");
CCASSERT(vec5.size() ==
21, "");
vec5.back()->setTag(100);
vec5.popBack();
CCASSERT(vec5.size() ==
20, "");
CCASSERT(vec5.back()->getTag() !=
100, "");
// Erase and clear
Vector<Node*> vec6 = createVector();
Vector<Node*> vec7 = vec6;
// Copy for check
CCASSERT(vec6.size() ==
20, "");
vec6.erase(vec6.begin() +
1); //
CCASSERT(vec6.size() ==
19, "");
CCASSERT((*(vec6.begin() +
1))->getTag() ==
1002, "");
vec6.erase(vec6.begin() +
2, vec6.begin() +
10);
CCASSERT(vec6.size() ==
11, "");
CCASSERT(vec6.at(0)->getTag() ==
1000, "");
CCASSERT(vec6.at(1)->getTag() ==
1002, "");
CCASSERT(vec6.at(2)->getTag() ==
1011, "");
CCASSERT(vec6.at(3)->getTag() ==
1012, "");
vec6.erase(3);
CCASSERT(vec6.at(3)->getTag() ==
1013, "");
vec6.eraseObject(vec6.at(2));
CCASSERT(vec6.at(2)->getTag() ==
1013, "");
vec6.clear();
auto objA =
Node::create();
// retain count is 1
auto objB = Node::create();
auto objC = Node::create();
{
Vector<Node*> array1;
Vector<Node*> array2;
// push back objA 3 times
array1.pushBack(objA);
// retain count is 2
array1.pushBack(objA);
// retain count is 3
array1.pushBack(objA);
// retain count is 4
array2.pushBack(objA);
// retain count is 5
array2.pushBack(objB);
array2.pushBack(objC);
for (auto obj : array1) {
array2.eraseObject(obj);
}
CCASSERT(objA->getReferenceCount() ==
4, "");
}
CCASSERT(objA->getReferenceCount() ==
1, "");
{ //这是一个代码块。出了块引用计数会恢复。是一个AutoReleasePool
Vector<Node*> array1;
// push back objA 3 times
array1.pushBack(objA);
// retain count is 2
array1.pushBack(objA);
// retain count is 3
array1.pushBack(objA);
// retain count is 4
CCASSERT(objA->getReferenceCount() ==
4, "");
array1.eraseObject(objA,
true); // Remove all occurrences in the Vector.
CCASSERT(objA->getReferenceCount() ==
1, "");
array1.pushBack(objA);
// retain count is 2
array1.pushBack(objA);
// retain count is 3
array1.pushBack(objA);
// retain count is 4
array1.eraseObject(objA,
false);
CCASSERT(objA->getReferenceCount() ==
3, "");
// Only remove the first occurrence in the Vector.
}
// Check the retain count in vec7
CCASSERT(vec7.size() ==
20, "");
for (const
auto& child : vec7)
{
CC_UNUSED_PARAM(child);
CCASSERT(child->getReferenceCount() ==
2, "");
}
// Sort 排序。结合Lambda
Vector<Node*> vecForSort = createVector();
std::sort(vecForSort.begin(), vecForSort.end(), [](Node* a,
Node* b){
return a->getTag() >= b->getTag();
});
for (int i =
0; i < 20; ++i)
{
CCASSERT(vecForSort.at(i)->getTag() -
1000 == (19 - i),
"");
}
// Reverse
vecForSort.reverse();
for (int i =
0; i < 20; ++i)
{
CCASSERT(vecForSort.at(i)->getTag() -
1000 == i, "");
}
// Swap
Vector<Node*> vecForSwap = createVector();
vecForSwap.swap(2,
4);
CCASSERT(vecForSwap.at(2)->getTag() ==
1004, "");
CCASSERT(vecForSwap.at(4)->getTag() ==
1002, "");
vecForSwap.swap(vecForSwap.at(2), vecForSwap.at(4));
CCASSERT(vecForSwap.at(2)->getTag() ==
1002, "");
CCASSERT(vecForSwap.at(4)->getTag() ==
1004, "");
// shrinkToFit
Vector<Node*> vecForShrink = createVector();
vecForShrink.reserve(100);
CCASSERT(vecForShrink.capacity() ==
100, "");
vecForShrink.pushBack(Node::create());
vecForShrink.shrinkToFit();
CCASSERT(vecForShrink.capacity() ==
21, "");
// get random object
// Set the seed by time
srand((unsigned)time(nullptr));
Vector<Node*> vecForRandom = createVector();
log("<--- begin ---->");
for (int i =
0; i < vecForRandom.size(); ++i)
{
log("Vector: random object tag = %d", vecForRandom.getRandomObject()->getTag());
}
log("<---- end ---->");
// Self assignment
Vector<Node*> vecSelfAssign = createVector();
vecSelfAssign = vecSelfAssign;
CCASSERT(vecSelfAssign.size() ==
20, "");
for (const
auto& child : vecSelfAssign)
{
CC_UNUSED_PARAM(child);
CCASSERT(child->getReferenceCount() ==
2, "");
}
vecSelfAssign = std::move(vecSelfAssign);
CCASSERT(vecSelfAssign.size() ==
20, "");
for (const
auto& child : vecSelfAssign)
{
CC_UNUSED_PARAM(child);
CCASSERT(child->getReferenceCount() ==
2, "");
}
// const at
Vector<Node*> vecConstAt = createVector();
constFunc(vecConstAt);
}
//---------------------------下面是Map的使用-----------------------------------
void TemplateMapTest::onEnter()
{
UnitTestDemo::onEnter();
auto createMap = [this](){
Map<std::string,
Node*> ret;
for (int i =
0; i < 20; ++i)
{
auto node = Node::create();
node->setTag(1000 + i);
ret.insert(StringUtils::toString(i), node);
}
return ret;
};
// Default constructor
Map<std::string,
Node*> map1;
CCASSERT(map1.empty(),
"");
CCASSERT(map1.size() ==
0, "");
CCASSERT(map1.keys().empty(),
"");
CCASSERT(map1.keys(Node::create()).empty(),
"");
// Move constructor
Map<std::string,
Node*> map2 = createMap();
for (const
auto& e : map2)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
2, "");
}
// Copy constructor
Map<std::string,
Node*> map3(map2);
for (const
auto& e : map3)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
3, "");
}
// Move assignment operator
Map<std::string,
Node*> map4;
auto unusedNode = Node::create();
map4.insert("unused",unusedNode);
map4 = createMap();
CCASSERT(unusedNode->getReferenceCount() ==
1, "");
for (const
auto& e : map4)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
2, "");
}
// Copy assignment operator
Map<std::string,
Node*> map5;
map5 = map4;
for (const
auto& e : map5)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
3, "");
}
// Check size
CCASSERT(map4.size() == map5.size(),
"");
for (const
auto& e : map4)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second == map5.find(e.first)->second,
"");
}
// bucket_count, bucket_size(n), bucket
log("--------------");
log("bucket_count = %d",
static_cast<int>(map4.bucketCount()));
log("size = %d",
static_cast<int>(map4.size()));
for (int i =
0; i < map4.bucketCount(); ++i)
{
log("bucket_size(%d) = %d", i,
static_cast<int>(map4.bucketSize(i)));
}
for (const
auto& e : map4)
{
log("bucket(\"%s\"), bucket index = %d", e.first.c_str(),
static_cast<int>(map4.bucket(e.first)));
}
log("----- all keys---------");
// keys and at
auto keys = map4.keys();
for (const
auto& key : keys)
{
log("key = %s", key.c_str());
}
auto node10Key = map4.at("10");
map4.insert("100", node10Key);
map4.insert("101", node10Key);
map4.insert("102", node10Key);
log("------ keys for object --------");
auto keysForObject = map4.keys(node10Key);
for (const
auto& key : keysForObject)
{
log("key = %s", key.c_str());
}
log("--------------");
// at in const function
constFunc(map4);
// find
auto nodeToFind = map4.find("10");
CC_UNUSED_PARAM(nodeToFind);
CCASSERT(nodeToFind->second->getTag() ==
1010, "");
// insert
Map<std::string,
Node*> map6;
auto node1 = Node::create();
node1->setTag(101);
auto node2 = Node::create();
node2->setTag(102);
auto node3 = Node::create();
node3->setTag(103);
map6.insert("insert01", node1);
map6.insert("insert02", node2);
map6.insert("insert03", node3);
CCASSERT(node1->getReferenceCount() ==
2, "");
CCASSERT(node2->getReferenceCount() ==
2, "");
CCASSERT(node3->getReferenceCount() ==
2, "");
CCASSERT(map6.at("insert01") == node1,
"");
CCASSERT(map6.at("insert02") == node2,
"");
CCASSERT(map6.at("insert03") == node3,
"");
// erase
Map<std::string,
Node*> mapForErase = createMap();
mapForErase.erase(mapForErase.find("9"));
CCASSERT(mapForErase.find("9") == mapForErase.end(),
"");
CCASSERT(mapForErase.size() ==
19, "");
mapForErase.erase("7");
CCASSERT(mapForErase.find("7") == mapForErase.end(),
"");
CCASSERT(mapForErase.size() ==
18, "");
std::vector<std::string> itemsToRemove;
itemsToRemove.push_back("2");
itemsToRemove.push_back("3");
itemsToRemove.push_back("4");
mapForErase.erase(itemsToRemove);
CCASSERT(mapForErase.size() ==
15, "");
// clear
Map<std::string,
Node*> mapForClear = createMap();
auto mapForClearCopy = mapForClear;
mapForClear.clear();
for (const
auto& e : mapForClearCopy)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
2, "");
}
// get random object
// Set the seed by time
srand((unsigned)time(nullptr));
Map<std::string,
Node*> mapForRandom = createMap();
log("<--- begin ---->");
for (int i =
0; i < mapForRandom.size(); ++i)
{
log("Map: random object tag = %d", mapForRandom.getRandomObject()->getTag());
}
log("<---- end ---->");
// Self assignment
Map<std::string,
Node*> mapForSelfAssign = createMap();
mapForSelfAssign = mapForSelfAssign;
CCASSERT(mapForSelfAssign.size() ==
20, "");
for (const
auto& e : mapForSelfAssign)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
2, "");
}
mapForSelfAssign = std::move(mapForSelfAssign);
CCASSERT(mapForSelfAssign.size() ==
20, "");
for (const
auto& e : mapForSelfAssign)
{
CC_UNUSED_PARAM(e);
CCASSERT(e.second->getReferenceCount() ==
2, "");
}
}
//--------------------------下面是Value的使用,它完毕基本数据类型到对象的包装--------
void ValueTest::onEnter()
{
UnitTestDemo::onEnter();
Value v1;
CCASSERT(v1.getType() ==
Value::Type::NONE,
"");
CCASSERT(v1.isNull(),
"");
Value v2(100);
CCASSERT(v2.getType() ==
Value::Type::INTEGER,
"");
CCASSERT(!v2.isNull(),
"");
Value v3(101.4f);
CCASSERT(v3.getType() ==
Value::Type::FLOAT,
"");
CCASSERT(!v3.isNull(),
"");
Value v4(106.1);
CCASSERT(v4.getType() ==
Value::Type::DOUBLE,
"");
CCASSERT(!v4.isNull(),
"");
unsigned char byte =
50;
Value v5(byte);
CCASSERT(v5.getType() ==
Value::Type::BYTE,
"");
CCASSERT(!v5.isNull(),
"");
Value v6(true);
CCASSERT(v6.getType() ==
Value::Type::BOOLEAN,
"");
CCASSERT(!v6.isNull(),
"");
Value v7("string");
CCASSERT(v7.getType() ==
Value::Type::STRING,
"");
CCASSERT(!v7.isNull(),
"");
Value v8(std::string("string2"));
CCASSERT(v8.getType() ==
Value::Type::STRING,
"");
CCASSERT(!v8.isNull(),
"");
auto createValueVector = [&](){
ValueVector ret;
ret.push_back(v1);
ret.push_back(v2);
ret.push_back(v3);
return ret;
};
Value v9(createValueVector());
CCASSERT(v9.getType() ==
Value::Type::VECTOR,
"");
CCASSERT(!v9.isNull(),
"");
auto createValueMap = [&](){
ValueMap ret;
ret["aaa"] = v1;
ret["bbb"] = v2;
ret["ccc"] = v3;
return ret;
};
Value v10(createValueMap());
CCASSERT(v10.getType() ==
Value::Type::MAP,
"");
CCASSERT(!v10.isNull(),
"");
auto createValueMapIntKey = [&](){
ValueMapIntKey ret;
ret[111] = v1;
ret[222] = v2;
ret[333] = v3;
return ret;
};
Value v11(createValueMapIntKey());
CCASSERT(v11.getType() ==
Value::Type::INT_KEY_MAP,
"");
CCASSERT(!v11.isNull(),
"");
}
----------------------------亲,学会了没----------------------------
版权声明:本文博客原创文章。博客,未经同意,不得转载。
Cocos2d-x 3.0final 终结者系列教程12-Vector&map&value的更多相关文章
- Cocos2d-x 3.0final 终结者系列教程16-《微信飞机大战》实现
看到cocos2d-x推出了3.1版本号,真是每月一次新版本号,速度. 另一个好消息就是http://cn.cocos2d-x.org/上线了,祝贺!啥时候把我的视频和教程放上去呢?!! . 视频下载 ...
- Cocos2d-x 3.0final 终结者系列教程13-贪食蛇游戏案例(全)
快过节了.谢谢了屈原,我们爱你. 应该多几个向屈大人一样跳江的,这样我们就能够放假纪念啦. ---------------------------------快过节了.弄个案例,大家最好还是假期做做, ...
- Cocos2d-x 3.0final 终结者系列教程01-无论是从cocos2d-x2.x升级到版本cocos2d-x3.x
诡谲的江湖,易变. 花花世界,车来人往. 最终确定.安家,将Cocos2d-x3.0final相关技术精加工的版本.并推出了博客文章, 不为他人,只为自己. 学习交流QQ群:301954471 --- ...
- Cocos2d-x 3.0final 终结者系列教程04-引擎架构分析
从前,有一个跟我来Android学生,总是问我: 沉老师,为什么Android的形式被称为Activity,为什么要onCreate方法写setContentView(R.layout.main)? ...
- Cocos2d-x 3.0final 终结者系列教程03-源代码文件夹说明
话说今天从霍营到回龙观,走到天鑫家园东路.我肋哥去,堵死我啦.7:30出门,9:10还没到回龙观. 北京这交通真是坑爹.回过头想想.怪自己走小路,有时候确实快.可有时候真堵. 堵了35分钟后果断掉头, ...
- Cocos2d-x 3.0final 终结者系列教程15-win7+vs2012+adt+ndk环境搭建(无Cygwin)
最终不用Cygwin 了.非常高兴 为什么要用Win7? 由于VS2012要求Win7以上系统才干安装! 为什么要用vs2012? 由于VS2012才支持C++11! 为什么要支持C++11? 由于C ...
- Cocos2d-x 3.0final 终结者系列教程02-开发环境的搭建
本文主要以Mac平台和XCode5为基本系统环境和C++编程工具来介绍Cocos2d-x3.0final版的安装. 一.系统准备(预计要花掉半个月工资) MacBook Pro一台(本人的比較老.11 ...
- Cocos2d-x 3.0final 终结者系列教程05-AppDelegate入口类
下面是Cocos2d-x的程序入口: class AppDelegate : private cocos2d::Application { public: AppDelegate(); virtua ...
- Cocos2d-x 3.0final 终结者系列教程14-L新abel-Cocos2d-x公文
目 录 新文本标签类Label 其它文本标签 字体制作工具使用介绍 小结 https://github.com/chukong/cocos-docs/blob/master/manual/framew ...
随机推荐
- mac已安装xctool而简单的执行xctool打包
先安装brew,brew是一个包管理工具,有了它我们就能够非常方便的安装xctool了,brew安装命令例如以下: curl -LsSf http://github.com/mxcl/homebrew ...
- 初步swift语言学习笔记8(保留了很多OC实现)
笔者:fengsh998 原文地址:http://blog.csdn.net/fengsh998/article/details/32715833 转载请注明出处 假设认为文章对你有所帮助,请通过留言 ...
- ssh连接失败,排错经验(转)
一.场景描述 ssh连接服务器,发现连接失败,但是对应服务器的ip能够ping通. 场景: [root@yl-web ~]# ssh root@10.1.101.35 ssh_exchange_ide ...
- [SignalR]在非Hub继承类中使用脚本方法
原文:[SignalR]在非Hub继承类中使用脚本方法 新建一个普通类OutHub,里面包含一个脚本方法OutHubTest. 因为大家知道,若能让脚本调用到的话,必须继承Hub,那怎么实现了?通过G ...
- paip.自适应网页设计 同 响应 与设计的原理的差and实践总结
paip.自适应网页设计 同 响应 与设计的原理的差and实践总结 响应式Web设计(Responsive Web design)的理念是: 1 #-----------自适应布局VS响应式布局 2 ...
- twitter接口开发
前一阵子研究了下twitter接口,发现网上的资料不是很多.遂花了些心血,终于有所收获~ 现在有时间赶紧整理出来便于自己以后查阅,也想帮助有困难的同学们.废话不多说,现在就以最简洁的方式开始了.注意: ...
- 选择一个利于SEO的空间
大家好.今天蜗牛将给大家分享怎么选择一个利于SEO的站点空间. 一.什么是站点空间? 站点空间,是用来存你的站点的HTML,.图片.文件等的一个远程硬盘空间(就像你的电脑里面的空间一回事儿).依据不同 ...
- UIMenuController,UIPasteboard:复制,粘贴详细解释
首先,这四个地图.我想实现的功能.[一张照片讲一个复制到另一UIimageView上] 实现代码例如以下: -(IBAction)panGestureTop:(UILongPressGestureRe ...
- [DB][mybatis]MyBatis mapper文件引用变量#{}与${}差异
MyBatis mapper文件引用变量#{}与${}差异 默认,使用#{}语法,MyBatis会产生PreparedStatement中.而且安全的设置PreparedStatement參数,这个过 ...
- js 通信
js 页面间的通信 看了一下公司原来的代码,原页面ajax post返回一个页面完整的HTML,然后再打开一个新页面并输出ajax返回的所有代码到新页面上,在新页面上以表单提交的形式实现重定向. 任凭 ...