在很久很久以前,我用写过一篇博客,关于如何查看CCArray与CCDictionary里存放的元素: http://blog.csdn.net/star530/article/details/23877429
现在既然它俩已经都是过去式了,那么之前的查看方法肯定也是不能用了。而Value作为它们的替代者之一,自然也有它查看元素的方法,那就是 getDescription()这个接口,下面我简单举个例子:
Value a(10);std::string a_str = a.getDescription();下面看下程序运行结果,输出如下信息:
CCLOG("a = %s",a_str.c_str());//打印出 a 对应的值
看输出好像没什么问题,但这里有个疑问, 为什么 a = 与 10没有在同一行呢?
在getDescription()这个接口里,到底发生了惨绝人寰的事?带着疑问与好奇,我们到getDescription()的定义里一探虚实。
std::string Value::getDescription(){ std::string ret("\n");//插入换行符 ret += visit(*this, 0); return ret;}看完代码才发现, 原来在返回的字符串里先插入一个 "\n" 换行符。
我们的故事本来到这里应该结束了,但总有些强迫症的孩子心里有点疙瘩,既然都看到这了,干嘛不再看看 visit(*this, 0) 这个函数都做了些什么呢?
好吧,既然你都这样求我了,那...我们继续往下看:
static std::string visit(const Value& v, int depth){ std::stringstream ret;//创建一个字符串流//判断Value内元素的类型 switch (v.getType()) { case Value::Type::NONE: case Value::Type::BYTE: case Value::Type::INTEGER: case Value::Type::FLOAT: case Value::Type::DOUBLE: case Value::Type::BOOLEAN: case Value::Type::STRING: ret << v.asString() << "\n";//如果是上面这几种类型,那么直接将其转换成string型即可。 break; case Value::Type::VECTOR: ret << visitVector(v.asValueVector(), depth); break; case Value::Type::MAP: ret << visitMap(v.asValueMap(), depth); break; case Value::Type::INT_KEY_MAP: ret << visitMap(v.asIntKeyMap(), depth); break; default: CCASSERT(false, "Invalid type!"); break; } return ret.str();}上述代码我没有做过多的注释,我们只要注意两点:
1、普通的数据类型,如int,string等,可以直接转成string型(asString())后放入ret流中即可。而如果元素类型是Vector或者Map,则需要做进一步的处理;
2、visit(const Value& v, int depth) 这个函数的第二个参数 depth是个什么玩意?眼尖的我发现进一步处理Vector类型的元素时,调用到visitMap(v.asValueMap(), depth)这个函数,而depth正好是该函数的第二个参数。
带着疑惑,我们继续:
static std::string visitVector(const ValueVector& v, int depth){ std::stringstream ret; if (depth > 0) ret << "\n"; ret << getTabs(depth) << "[\n"; int i = 0; for (const auto& child : v) { ret << getTabs(depth+1) << i << ": " << visit(child, depth + 1);//在这里读取Vector中的元素 ++i; } ret << getTabs(depth) << "]\n"; return ret.str();}这函数乍看一下有点莫名其妙,其实我们只要知道 getTabs(int depth)这个函数是怎么回事就可以,其他自然迎刃而解。
static std::string getTabs(int depth){ std::string tabWidth; for (int i = 0; i < depth; ++i) { tabWidth += "\t";//插入一个制表符 } return tabWidth;}原来getTabs()的功能就是插入制表符啊(所谓制表符就是tab键),而传入的参数depth是多少就添加多少个制表符。
好了,流程就是这样子,我想很多人现在脑袋瓜子都还蒙蒙的,没事,下面我举个栗子,然后小伙伴们根据程序的输出结果与上面提到的visit()、visitVector()的代码参照下。
Value a(10);Value b("star is so cool");ValueVector star_vec;star_vec.push_back(a);star_vec.push_back(b);//将ValueVector类型转成Value类型才可调用getDescription()std::string star_str = ( (Value)star_vec ).getDescription();CCLOG("-----------------------");CCLOG("%s",star_str.c_str());CCLOG("-----------------------");程序运行后,输出结果如下:
好了,就酱紫吧,能不能理解就看阁下的悟性了。
尊重原创,转载请注明来源:http://blog.csdn.net/star530/article/details/38071517