【iOS-cocos2d-X 游戏开发之九】Cocos2dx利用CCSAXParser解析xml数据

简介:

本章Himi给大家分享如何在Cocos2dX中解析xml数据;对于数据存取有很多方式,流文件,plist,xml等,那么为了跨平台更好的支持,Himi想到之前写的CCUserDefault 存储数据一节,Cocos2dx自带的存储类,一旦存入数据都会以xml格式进行保存,适用于iOS、Android等平台,所以这里Himi使用xml进行游戏的一些数据录入  = =.. 另外一方面Himi本章节也是基于Cocos2dx引擎代码进行的一次简单对xml数据解析的封装;

为了更保险的去考虑跨平台,所以对于xml存储这块的解析,也做了些搜索,最后发现王哥(王哲-cocos2dx引擎作者)也有给我们提示过,内容如下:


  
  
  1. cocos2dx里面集成了libxml2,ios上会调用sdk里面内置的, 
  2. android和win32上则带了已经编译好的静态/动态库。 
  3. 你可以参考CCSAXParser里面的代码来使用libxml2 
 

那么既然如此,就对Cocos2dx引擎源码的CCSAXParser类进行了剖析,那么这里Himi,先给出代码,然后再详细讲解下:

Himi封装的HXmlParse类:


  
  
  1. HXmlParse.h 
  2.   
  3. // 
  4. //  HXmlParse.h 
  5. //  HAnimation 
  6. // 
  7. //  Created by Himi on 12-3-22. 
  8. //  Copyright (c) 2012年 Augustimpression. All rights reserved. 
  9. // 
  10.   
  11. #ifndef HAnimation_HXmlParse_h 
  12. #define HAnimation_HXmlParse_h 
  13.   
  14. #include "cocos2d.h" 
  15. #include "CCSAXParser.h" 
  16. #include "CCObject.h" 
  17. #include "CCMutableDictionary.h" 
  18.   
  19. using namespace cocos2d; 
  20.   
  21. class CC_DLL HXmlParse :public CCObject, public CCSAXDelegator 
  22.   
  23. public
  24.   
  25.     static HXmlParse * parserWithFile(const char *tmxFile); 
  26.   
  27.     bool initHXmlParse(const char* xmlName); 
  28.   
  29.     //  使用 CCSAXDelegator 重写3个回调函数 
  30.   
  31.     void startElement(void *ctx, const char *name, const char **atts); 
  32.   
  33.     void endElement(void *ctx, const char *name); 
  34.   
  35.     void textHandler(void *ctx, const char *ch, int len); 
  36.   
  37.     std::string root_name;  
  38.   
  39.     bool isJumpHeadData; 
  40.   
  41.     CCMutableDictionary<std::string,CCString*> *mDic; 
  42.   
  43. private
  44.   
  45.     std::string startXmlElement;//用来记录每个key前字段 
  46.   
  47.     std::string endXmlElement;//用来记录每个key后字段 
  48.   
  49.     std::string currString;//记录每个value的值 
  50.   
  51. }; 
  52.   
  53. #endif  
  54. HXmlParse.cpp 
  55.   
  56. // 
  57. //  HXmlParse.cpp 
  58. //  HAnimation 
  59. // 
  60. //  Created by Himi on 12-3-22. 
  61. //  Copyright (c) 2012年 Augustimpression. All rights reserved. 
  62. // 
  63.   
  64. #include "HXmlParse.h" 
  65. #include "CCSAXParser.h" 
  66.   
  67. HXmlParse * HXmlParse::parserWithFile(const char *tmxFile) 
  68.     HXmlParse *pRet = new HXmlParse(); 
  69.     if(pRet->initHXmlParse(tmxFile)) 
  70.     { 
  71.         pRet->autorelease(); 
  72.         return pRet; 
  73.     } 
  74.     CC_SAFE_DELETE(pRet); 
  75.     return NULL; 
  76.   
  77. bool HXmlParse::initHXmlParse(const char* xmlName) 
  78.   
  79.   
  80.     mDic = new CCMutableDictionary<std::string,CCString*>(); 
  81.   
  82.     CCSAXParser _par;   
  83.   
  84.     if (false == _par.init("UTF-8") ) 
  85.   
  86.     { 
  87.         CCLog("-----请使用utf-8格式!"); 
  88.         return false
  89.     } 
  90.     _par.setDelegator(this); 
  91.     const char* _path =CCFileUtils::fullPathFromRelativePath(xmlName); 
  92.     return _par.parse(_path); 
  93.   
  94.   
  95. //回调函数 
  96.   
  97. void HXmlParse::startElement(void *ctx, const char *name, const char **atts) 
  98.   
  99. {    
  100.   
  101.     CC_UNUSED_PARAM(ctx); 
  102.   
  103.     startXmlElement = (char*)name; 
  104.     if(!isJumpHeadData){//跳过数据头 
  105.         CCLog("------跳过root name"); 
  106.         isJumpHeadData=true
  107.         root_name=startXmlElement; 
  108.         return
  109.     } 
  110.   
  111. //    CCLog("-startElement----%s",startXmlElement.c_str()); 
  112.   
  113.   
  114. void HXmlParse::endElement(void *ctx, const char *name) 
  115.   
  116.     CC_UNUSED_PARAM(ctx); 
  117.   
  118.     endXmlElement = (char*)name; 
  119.     if(endXmlElement==root_name){//数据尾 
  120.         CCLog("读取xml结束"); 
  121.         isJumpHeadData=false
  122.         root_name=""
  123.         return
  124.     } 
  125.   
  126. //    CCLog("-endElement----%s",endXmlElement.c_str()); 
  127. //键值对的结束字段 
  128. void HXmlParse::textHandler(void *ctx, const char *ch, int len) 
  129.   
  130.     CC_UNUSED_PARAM(ctx); 
  131.     currString=string((char*)ch,0,len); 
  132.     CCString *ccStr =new CCString();//备注3 
  133.     ccStr->m_sString=currString; 
  134.     if(root_name!=""){ 
  135.          mDic->setObject(ccStr,startXmlElement); 
  136.         CCLog("-----key:%s, value:%s",startXmlElement.c_str(),mDic->objectForKey(startXmlElement)->m_sString.c_str()); 
  137.     } 
  138.   
  139. //    CCLog("-textHandler----%s",currString.c_str()); 
  140.   
 

OK,代码呢我们先从.h中来说,首先我们使用CCSAXDelegator,为了让CCSAXParser解析数据后将数据回调给如下三个函数:


  
  
  1. //  使用 CCSAXDelegator 重写3个回调函数 
  2.   
  3.    void startElement(void *ctx, const char *name, const char **atts); 
  4.   
  5.    void endElement(void *ctx, const char *name); 
  6.   
  7.    void textHandler(void *ctx, const char *ch, int len); 
 

startElement   函数解析的是xml的每个key前字段

textHandler  函数解析出来的是xml每个key对应的value值

endElement   函数解析出来的是xml的每个key后字段

这里Himi随便写了一个xml来做测试,himi.xml,如下:


  
  
  1. <?xml version="1.0" encoding="utf-8"?><himiTestData><key1>1000</key1><key2>娃哈哈</key2><key3>82.3</key3><key4>4000</key4><key5>himi</key5><key6>true</key6></himiTestData> 

那么CCSAXParser类解析第一次回调 startElement 是读取的是root name(xml数据头标识名称->“<himiTestData>”),然后才读取正式数据key和value,最后读取的也是xml数据尾标识名称“</himiTestData>”

当然在Himi封装的HXmlParse类中对于数据标识的读取都跳过了,使用变量isJumpHeadData来处理的;

其他的都很容易没有什么可说的,主要要说还有一点就是关于CCMutableDictionary的使用,对于此类主要结构是形成map&NSMutableDictionary类似是个键值对容器,key-value;那么使用时候要注意4点:

1.  比如Himi解析数据后都会默认将key和value数据存放在CCMutableDictionary中,那么这里我肯定传入的是两个string,但是细心的童鞋会发现代码中第二个并不是std::string,而是CCString对象,嗯 没错,CCMutableDictionary要求传入的是CCObject对象而不是基本类型!CCString中有m_sString这个属性,所以转换起来也是很方便的;

2. 使用CCMutableDictionary进行添加数据setObject的时候要注意此函数的两个参数:


  
  
  1. bool setObject(_ValueT pObject, const _KeyT& key) 
  2.     { 
  3.         pair<CCObjectMapIter, bool > pr; 
  4.   
  5.         pr = m_Map.insert( Int_Pair(key, pObject) ); 
  6.   
  7.         if(pr.second == true
  8.         { 
  9.             pObject->retain(); 
  10.             return true
  11.         } 
  12.   
  13.         return false
  14.     } 
 

上面这个是cocos2dx引擎中源码中setObject函数实现代码,这里可以很清晰的看到,第一个参数表示《CCObject》,第二个参数才是《Key》!这点对于之前做过java开发的我来说比较郁闷,因为一般都是key在第一个参数。。。。。

3. 请大家仔细看HXmlParse.cpp类中的备注3 ,当你使用CCMutableDictionary的setObject函数的时候,务必要注意,此函数的存入的CCObject参数,引擎中实现代码是对你这个CCObject进行retain()的一个内存地址引用!也就是说这里不要使用一个成员变量来使用,否则你从CCMutableDictionary取出来的数据全部是你最后一个CCObject数据!

4. CCMutableDictionary是cocos2d-x自己封装的类,功能类似NSMutableDictionary。但是Himi通过测试发现!它有一点和NSMutableDictionary是不一样的。NSMutableDictionary的setObjectForKey方法是:如果发现这个key已经存在于字典中的时候,它会自动用新的object覆盖掉原有的object。而CCMutableDictionary由于它是使用map实现的字典功能,而在map里面,如果key已存在,是不会用新的object覆盖掉原有object的。在使用CCMutableDictionary的时候需要特别注意这一点。

HXmlParse解析xml类使用方法很简单:


  
  
  1. #include "HXmlParse.h" 
  2.   
  3. HXmlParse::parserWithFile("himi.xml"); 
 

然后Himi为了证实此解析类在Android也可以正常运行,那么这里Himi将读出的数据展示在画面上,iOS运行截图如下:

 

 









本文转自 xiaominghimi 51CTO博客,原文链接:http://blog.51cto.com/xiaominghimi/841867,如需转载请自行联系原作者
目录
相关文章
|
24天前
|
XML 数据采集 API
用Lxml高效解析XML格式数据:以天气API为例
免费Python教程:实战解析中国天气网XML数据,详解Lxml库高效解析技巧、XPath用法、流式处理大文件及IP封禁应对策略,助你构建稳定数据采集系统。
130 0
|
8月前
|
存储 Java 文件存储
微服务——SpringBoot使用归纳——Spring Boot使用slf4j进行日志记录—— logback.xml 配置文件解析
本文解析了 `logback.xml` 配置文件的详细内容,包括日志输出格式、存储路径、控制台输出及日志级别等关键配置。通过定义 `LOG_PATTERN` 和 `FILE_PATH`,设置日志格式与存储路径;利用 `&lt;appender&gt;` 节点配置控制台和文件输出,支持日志滚动策略(如文件大小限制和保存时长);最后通过 `&lt;logger&gt;` 和 `&lt;root&gt;` 定义日志级别与输出方式。此配置适用于精细化管理日志输出,满足不同场景需求。
1971 1
|
5月前
|
XML 存储 数据格式
抖音卡片链接生成器,xml卡片数据支持快手,通过XPOSED实现制作
本项目介绍抖音/快手卡片链接生成技术,包含技术原理与核心功能实现。通过Xposed框架Hook目标APP关键方法,自定义卡片生成与跳转逻辑。卡片数据以XML格式存储,便于解析和跨平台使用。提供完整代码示例,涵盖Xposed模块配置、XML数据结构、Hook实现及卡片生成器核心类。下载地址:https://www.pan38.com/share.php?code=DuNzA,提取码:8888(仅供学习参考)。
|
8月前
|
数据采集 JSON 数据可视化
JSON数据解析实战:从嵌套结构到结构化表格
在信息爆炸的时代,从杂乱数据中提取精准知识图谱是数据侦探的挑战。本文以Google Scholar为例,解析嵌套JSON数据,提取文献信息并转换为结构化表格,通过Graphviz制作技术关系图谱,揭示文献间的隐秘联系。代码涵盖代理IP、请求头设置、JSON解析及可视化,提供完整实战案例。
494 4
JSON数据解析实战:从嵌套结构到结构化表格
|
8月前
|
XML JavaScript Android开发
【Android】网络技术知识总结之WebView,HttpURLConnection,OKHttp,XML的pull解析方式
本文总结了Android中几种常用的网络技术,包括WebView、HttpURLConnection、OKHttp和XML的Pull解析方式。每种技术都有其独特的特点和适用场景。理解并熟练运用这些技术,可以帮助开发者构建高效、可靠的网络应用程序。通过示例代码和详细解释,本文为开发者提供了实用的参考和指导。
227 15
|
8月前
|
JSON 监控 网络协议
Bilibili直播信息流:连接方法与数据解析
本文详细介绍了自行实现B站直播WebSocket连接的完整流程。解析了基于WebSocket的应用层协议结构,涵盖认证包构建、心跳机制维护及数据包解析步骤,为开发者定制直播数据监控提供了完整技术方案。
|
8月前
|
机器学习/深度学习 JSON 算法
淘宝拍立淘按图搜索API接口系列的应用与数据解析
淘宝拍立淘按图搜索API接口是阿里巴巴旗下淘宝平台提供的一项基于图像识别技术的创新服务。以下是对该接口系列的应用与数据解析的详细分析
|
9月前
|
Java API 数据处理
深潜数据海洋:Java文件读写全面解析与实战指南
通过本文的详细解析与实战示例,您可以系统地掌握Java中各种文件读写操作,从基本的读写到高效的NIO操作,再到文件复制、移动和删除。希望这些内容能够帮助您在实际项目中处理文件数据,提高开发效率和代码质量。
221 4
|
8月前
|
缓存 监控 搜索推荐
【实战解析】smallredbook.item_get_video API:小红书视频数据获取与电商应用指南
本文介绍小红书官方API——`smallredbook.item_get_video`的功能与使用方法。该接口可获取笔记视频详情,包括无水印直链、封面图、时长、文本描述、标签及互动数据等,并支持电商场景分析。调用需提供`key`、`secret`和`num_iid`参数,返回字段涵盖视频链接、标题、标签及用户信息等。同时,文章提供了电商实战技巧,如竞品监控与个性化推荐,并列出合规注意事项及替代方案对比。最后解答了常见问题,如笔记ID获取与视频链接时效性等。
|
8月前
|
存储 缓存 监控
如何高效爬取天猫商品数据?官方API与非官方接口全解析
本文介绍两种天猫商品数据爬取方案:官方API和非官方接口。官方API合法合规,适合企业长期使用,需申请企业资质;非官方接口适合快速验证需求,但需应对反爬机制。详细内容涵盖开发步骤、Python实现示例、反爬策略、数据解析与存储、注意事项及扩展应用场景。推荐工具链包括Playwright、aiohttp、lxml等。如需进一步帮助,请联系作者。

推荐镜像

更多
  • DNS