Windows Mobile和Wince下使用TinyXML进行Native C++的开发

简介:

背景

继续讲述Mobile Radio项目的开发,上回讲到如何把自于 www.1radio.com.au 网站的电台数据从JSON转换成XML。这回讲述使用tinyXML在windows mobile下进行XML的开发。

 

Mobile Radio项目可以参考:

Windows Mobile和Wince下的WTL(Windows Template Library)开发

Windows Mobile 和 Wince 下的 WTL(Windows Template Library) 界面开发

Windows Mobile和Wince下使用WTL进行Windows Media Player开发

转换Json到XML的JavaScript实现

 

简介

XML已经成为流行的数据保存和交换的格式,本文讲述如何使用TinyXML在Windows Mobile下进行XML的开发。TinyXML是简单,轻装,跨平台的原生C++ xml解释器,可以十分简便的整合到其他系统中。同时TinyXML提供完整的在线文档,方便开发和使用。目前,由于其简便性和稳定性,使用ZLib license(可以用于开源和商业)等原因,TinyXML已经广泛被用于开源社区和商业系统中。

关于更多TinyXML的介绍请看下面链接:

http://sourceforge.net/projects/tinyxml/

http://www.grinninglizard.com/tinyxml/index.html

http://www.grinninglizard.com/tinyxmldocs/index.html

 

环境搭建

下载

请到http://sourceforge.net/projects/tinyxml/ 下载最新的release。

新建项目

tinyxml1

新建Smart Device项目tinyXML。

 tinyxml2

点击Next

tinyxml3

选择平台,这里选择Windows Mobile 6 Professional SDK。

tinyxml4

选择生成静态库,不需要MFC支持和不需要生成预编译文件,点击完成。

tinyxml7

tinyxml5

把下载的TinyXML源代码文件(包括CPP和H文件)拷贝到项目目录下。

tinyxml9

把源代码文件(包括CPP和H文件)添加到项目中。

设置依赖性

tinyxml6 

在Project –> Project Dependencies设置项目依赖性,Mobile Radio依赖于TinyXML。

如果使用在Windows Mobile环境下,需要更改下面的代码。

// Microsoft compiler security
FILE* TiXmlFOpen( const char* filename, const char* mode )
{
//#if defined(_MSC_VER) && (_MSC_VER >= 1400 )
// FILE* fp = 0;
// errno_t err = fopen_s( &fp, filename, mode );
// if ( !err && fp )
// return fp;
// return 0;
//#else
// return fopen( filename, mode );
//#endif
return fopen( filename, mode );
}

tinyxml.cpp文件

//#define TIXML_SAFE
#define TIXML_SSCANF sscanf

tinyxml.h文件

环境搭建完毕。

使用

使用TinyXML是一个愉悦的过程,所有使用的例子都可以在源代码的xmltest.cpp文件里面找到。所以强烈建议学习和使用TinyXML前先认真阅读xmltest.cpp的代码。

简单讲一下XML文件的结构,XML的结构就是层次性(Hierarchy)文件,包含Element(节点)和Attribute(属性),下面我保留英文,因为TinyXML的接口就也是使用同样的术语。Element就是节点,可以包含Attribute和子Element(Child Elements),Attribute就是Element的属性。

下面是Mobile Radio使用TinyXML的代码。

#include "include/tinyXML/tinyXML.h"

使用TinyXML只需要引用一个头文件就可以了。

const char* CONFIGPATH = "Config\\Stations.xml";
void CMobileRadioView::LoadConfig()
{
std::string p = GetCurrentPath() + std::string(CONFIGPATH);
TiXmlDocument document = TiXmlDocument(p.c_str());
if(!document.LoadFile())
{
MessageBox(L"Can not open the config file.");
return;
}

TiXmlHandle docHandle(&document);
TiXmlElement* cityElement = docHandle.FirstChild("stations").FirstChild("city").Element();
Station* station;
std::string city;
while (cityElement)
{
city = cityElement->Attribute("name");
TiXmlElement* stationElement = cityElement->FirstChildElement("station");
while (stationElement)
{
station = new Station();
station->City = city;
station->Id = atoi(stationElement->Attribute("sid"));
station->Name = stationElement->Attribute("name");
station->Image = stationElement->Attribute("image");
station->Stream = stationElement->Attribute("stream");
station->Website = stationElement->Attribute("website");

stationMap[station->Id] = station;
cityStationMap.insert(CityStationMap::value_type(station->City, station));
stationElement = stationElement->NextSiblingElement();
}
cityElement = cityElement->NextSiblingElement();
}
}

这是读取XML配置的代码,XML配置文件的结构可以参考 转换Json到XML的JavaScript实现 。大体的文件结构是分两层,第一层是城市,第二层是具体的电台信息。

TiXmlDocument document = TiXmlDocument(p.c_str());
if(!document.LoadFile())
{
MessageBox(L"Can not open the config file.");
return;
}

把XML配置文件加载到TiXmlDocument里。

TiXmlHandle docHandle(&document);

根据TiXmlDocument生成TiXmlHandle。

TiXmlElement* cityElement = docHandle.FirstChild("stations").FirstChild("city").Element();

TinyXML不直接支持XPath,所以只能一层层读,从根节点逐层查找。如果需要XPath支持,可以参考TinyXPath (http://tinyxpath.sourceforge.net)。

while (cityElement)
{
city = cityElement->Attribute("name");
TiXmlElement* stationElement = cityElement->FirstChildElement("station");
while (stationElement)
{
station = new Station();
station->City = city;
station->Id = atoi(stationElement->Attribute("sid"));
station->Name = stationElement->Attribute("name");
station->Image = stationElement->Attribute("image");
station->Stream = stationElement->Attribute("stream");
station->Website = stationElement->Attribute("website");

stationMap[station->Id] = station;
cityStationMap.insert(CityStationMap::value_type(station->City, station));
stationElement = stationElement->NextSiblingElement();
}
cityElement = cityElement->NextSiblingElement();
}

循环取出城市(City)和电台(Station)信息,FirstChildElement()查找第一个子Element。NextSiblingElement()用于读取同一层的兄弟Element,Attribute可以取出Element的Attribute。关于更多的读取例子,请看xmltest.cpp的代码。

上述例子把XML配置信息读取到C++的map和multimap里面。这两个容器的定义如下:

//Id -> Station
typedef std::map<int, Station*> StationMap;

//City -> Station
typedef std::multimap<std::string, Station*> CityStationMap;

StationMap保存ID和电台信息,一对一。CityStationMap保存城市和电台信息,一对多。

界面处理

两个容器初始化完毕以后,界面可以根据容器的信息生成。

for(CityStationMap::iterator it=cityStationMap.begin();
it!=cityStationMap.end(); ++it)
{
if(city.compare(it->first) != 0)
{
city = it->first;
CString c = city.c_str();
m_wndCity.AddString(c);
}
}

根据配置信息显示城市下拉框,由于multimap不支持直接把所有的key的集合读取出来,所以需要遍历,把不同的城市信息显示到m_wndCity下拉框中。

LRESULT CMobileRadioView::OnComboCityCbnSelChange(WORD wNotifyCode, WORD wID, HWND hWndCtl)
{
CString str;
int sel = m_wndCity.GetCurSel();
m_wndCity.GetLBText(sel, str);
m_wndStation.ResetContent();

std::string city = CT2CA(str);
unsigned int i = 0;
for(CityStationMap::iterator it=cityStationMap.find(city);
it!=cityStationMap.end() && i<cityStationMap.count(city); ++it,++i)
{
CString s = it->second->Name.c_str();
int index = m_wndStation.AddString(s);
m_wndStation.SetItemData(index, it->second->Id);
}
return 0;
}

当城市下拉框的选择发生改变时,根据multimap的信息显示该城市下的电台信息。

LRESULT CMobileRadioView::OnComboStationCbnSelChange(WORD wNotifyCode, WORD wID, HWND hWndCtl)
{
int sel = m_wndStation.GetCurSel();
int id = (int)m_wndStation.GetItemData(sel);
//CString image = (GetCurrentPath() + "Image\\" + stationMap[id]->Image).c_str();
//m_wndPic.SetBitmap(LoadBitmap(NULL, image));
m_spWMPPlayer->put_URL(CComBSTR(stationMap[id]->Stream.c_str()));
return 0;
}

当电台下拉框的选择发生改变时,根据map的信息使用Windows Media Player控件播放该电台。

tinyxml10

下一次讲述程序如何支持accelerometer(重力感应器)。

 

关于Mobile Radio - Internet Radio Software for Windows Mobile项目

 

目前(2009年9月份)这个项目基本功能已经完成,只是界面方面需要改进,提高用户体验。我把项目host到 Mobile Radio - Internet Radio Software for Windows Mobile 了,我会持续改进,主要是提高用户体验方面。

需要了解项目最新动态,可以访问 Mobile Radio - Internet Radio Software for Windows Mobile  和我的Blog  精简开发 无线生活

 

源代码: 查看Mobile Radio最新源代码

环境:VS2008 + WM 6 professional SDK + WTL 8.1 + TinyXML



    本文转自Jake Lin博客园博客,原文链接:http://www.cnblogs.com/procoder/archive/2009/07/21/Windows_Mobile_TinyXML.html,如需转载请自行联系原作者


相关文章
|
4月前
|
监控 关系型数据库 MySQL
PowerShell 脚本编写 :自动化Windows 开发工作流程
PowerShell 脚本编写 :自动化Windows 开发工作流程
161 0
|
4月前
|
Ubuntu Linux Python
如何利用wsl-Ubuntu里conda用来给Windows的PyCharm开发
如何在WSL(Windows Subsystem for Linux)的Ubuntu环境中使用conda虚拟环境来为Windows上的PyCharm开发设置Python解释器。
384 0
|
5月前
|
存储 安全 程序员
Windows任务管理器开发原理与实现
Windows任务管理器开发原理与实现
|
7月前
|
Linux Apache C++
FFmpeg开发笔记(三十五)Windows环境给FFmpeg集成libsrt
该文介绍了如何在Windows环境下为FFmpeg集成SRT协议支持库libsrt。首先,需要安装Perl和Nasm,然后编译OpenSSL。接着,下载libsrt源码并使用CMake配置,生成VS工程并编译生成srt.dll和srt.lib。最后,将编译出的库文件和头文件按照特定目录结构放置,并更新环境变量,重新配置启用libsrt的FFmpeg并进行编译安装。该过程有助于优化直播推流的性能,减少卡顿问题。
171 2
FFmpeg开发笔记(三十五)Windows环境给FFmpeg集成libsrt
|
6月前
|
开发者 C# Windows
WPF与游戏开发:当桌面应用遇见游戏梦想——利用Windows Presentation Foundation打造属于你的2D游戏世界,从环境搭建到代码实践全面解析新兴开发路径
【8月更文挑战第31天】随着游戏开发技术的进步,WPF作为.NET Framework的一部分,凭借其图形渲染能力和灵活的UI设计,成为桌面游戏开发的新选择。本文通过技术综述和示例代码,介绍如何利用WPF进行游戏开发。首先确保安装最新版Visual Studio并创建WPF项目。接着,通过XAML设计游戏界面,并在C#中实现游戏逻辑,如玩家控制和障碍物碰撞检测。示例展示了创建基本2D游戏的过程,包括角色移动和碰撞处理。通过本文,WPF开发者可更好地理解并应用游戏开发技术,创造吸引人的桌面游戏。
309 0
|
6月前
|
开发者 iOS开发 C#
Uno Platform 入门超详细指南:从零开始教你打造兼容 Web、Windows、iOS 和 Android 的跨平台应用,轻松掌握 XAML 与 C# 开发技巧,快速上手示例代码助你迈出第一步
【8月更文挑战第31天】Uno Platform 是一个基于 Microsoft .NET 的开源框架,支持使用 C# 和 XAML 构建跨平台应用,适用于 Web(WebAssembly)、Windows、Linux、macOS、iOS 和 Android。它允许开发者共享几乎全部的业务逻辑和 UI 代码,同时保持原生性能。选择 Uno Platform 可以统一开发体验,减少代码重复,降低开发成本。安装时需先配置好 Visual Studio 或 Visual Studio for Mac,并通过 NuGet 或官网下载工具包。
626 0
|
6月前
|
开发框架 JavaScript .NET
【Azure 应用服务】Azure Mobile App (NodeJS) 的服务端部署在App Service for Windows中出现404 Not Found
【Azure 应用服务】Azure Mobile App (NodeJS) 的服务端部署在App Service for Windows中出现404 Not Found
|
2天前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
1月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
68 19
|
1月前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
50 13