C++ 利用 libxl 将 Excel 文件转化为 Xml 文件
在游戏开发工作中,策划和运营一般会用Excel来编写配置文件,但是程序读取配置,最方便的还是xml文件.所以最好约定一个格式,然后在二者之间做一个转化.
本文利用libxl来读取Excel文件,利用 timyxml2 来写入xml文件
libxl3.65破解版 : http://pan.baidu.com/s/1boYaeRl 提取码:3xbe
tinyxml2 源码: https://github.com/leethomason/tinyxml2
#include <stdio.h>
#include <stdint.h>
#include <string.h>
#include <string>
#include <vector>
#include <iostream>
#include <windows.h>
#include <ctime>
#include <io.h>
#include <direct.h>
#include "libxl.h"
#include "tinyxml2.h" const char *formatErrorMsg_NoSeparateRows = "sheet format error:no separate rows!";
const char *formatErrorMsg_GroupNameNull = "sheet format error:group name null!";
const char *formatErrorMsg_CellValueNull = "cell format error:cell value null!"; enum
{
enmFileExtensionLength = ,
}; enum ErrorDef
{
enmErrorDef_OK = ,
enmErrorDef_CreateBookFailed = ,
enmErrorDef_NotExcelFile = ,
enmErrorDef_LoadExcelFileFailed = ,
enmErrorDef_SaveXmlFileFailed = ,
enmErrorDef_CalculateXmlFilePathError = ,
enmErrorDef_MakeXmlFilePathError = ,
}; int32_t ExcelToXml(const char *excelFilePath, const char *xmlFilePath);
int32_t ExcelToXmls(const char *excelFileDirectory, const char *xmlFileDirectory);
int32_t BookToXmlDoc(libxl::Book* book, tinyxml2::XMLDocument *xmlDoc);
int32_t SheetToXmlEle(libxl::Sheet *sheet, tinyxml2::XMLDocument *xmlDoc, tinyxml2::XMLElement *ele);
void GetPathExtensionName(const char *filePath, char ext[], const uint32_t extLen);
bool IsConfigSheet(const char *sheetName);
bool IsCellEmpty(int32_t cellType);
const char* ReadCellContent(libxl::Sheet *sheet, const int32_t row, const int32_t col);
void GBKToUTF8(const char* asciiBuf, WCHAR wcharTmp[], char utf8Buf[], const int32_t utf8BufLen);
void GetFilesFromDirectory(std::vector<std::string> &files, const std::string &directoryPath);
int32_t CalculateXmlFilePath(const char *excelFileDirectory, const char *xmlFileDirectory, const char *excelFilePath, char xmlFilePath[], const int32_t xmlFilePathMaxLen);
int32_t CreateDirectory(const char *directoryPath); int32_t ExcelToXml(const char *excelFilePath, const char *xmlFilePath)
{
libxl::Book* book = NULL;
char ext[enmFileExtensionLength] = { };
GetPathExtensionName(excelFilePath, ext, enmFileExtensionLength);
if (strcmp(ext, ".xls") == )
{
book = xlCreateBook();
}
else if (strcmp(ext, ".xlsx") == )
{
book = xlCreateXMLBook();
}
else
{
return enmErrorDef_NotExcelFile;
}
if (!book)
{
return enmErrorDef_CreateBookFailed;
}
if (!book->load(excelFilePath))
{
return enmErrorDef_LoadExcelFileFailed;
}
tinyxml2::XMLDocument xmlDoc;
int32_t ret = BookToXmlDoc(book, &xmlDoc);
if (ret != enmErrorDef_OK)
{
return ret;
}
if (xmlDoc.SaveFile(xmlFilePath) != tinyxml2::XML_SUCCESS)
{
return enmErrorDef_SaveXmlFileFailed;
}
book->release();
return enmErrorDef_OK;
} int32_t BookToXmlDoc(libxl::Book* book, tinyxml2::XMLDocument *xmlDoc)
{
tinyxml2::XMLDeclaration *declaration = xmlDoc->NewDeclaration("version=\"1.0\" encoding=\"utf-8\"");
xmlDoc->LinkEndChild(declaration);
for (int32_t i = ; i < book->sheetCount(); ++i)
{
if (!IsConfigSheet(book->getSheet(i)->name())){ continue; }
tinyxml2::XMLElement *ele = xmlDoc->NewElement(book->getSheet(i)->name() + );
xmlDoc->LinkEndChild(ele);
// printf("%s\n", book->getSheet(i)->name());
int32_t ret = SheetToXmlEle(book->getSheet(i), xmlDoc, ele);
if (ret != enmErrorDef_OK)
{
return ret;
}
}
return enmErrorDef_OK;
} int32_t SheetToXmlEle(libxl::Sheet *sheet, tinyxml2::XMLDocument *xmlDoc, tinyxml2::XMLElement *ele)
{
tinyxml2::XMLElement *group = NULL;
static std::vector<std::string> titles;
titles.clear();
for (int32_t row = ; row < sheet->lastRow(); ++row)
{
// 获取该行前两个单元格
int32_t firstCellType = sheet->cellType(row, );
int32_t secondCellType = sheet->cellType(row, );
// 如果都是空白,表明是空白行
if (IsCellEmpty(firstCellType) && IsCellEmpty(secondCellType))
{
titles.clear();
continue;
}
// 如果都不是空白,表明是标题行,获取标题
else if (!IsCellEmpty(firstCellType) && !IsCellEmpty(secondCellType))
{
do
{
// 标题列不为空,表明没有分隔,有问题
if (!titles.empty())
{
group = xmlDoc->NewElement(formatErrorMsg_NoSeparateRows);
break;
}
const char *firstCell = ReadCellContent(sheet, row, );
if (NULL == firstCell)
{
group = xmlDoc->NewElement(formatErrorMsg_GroupNameNull);
break;
}
group = xmlDoc->NewElement(firstCell);
for (int32_t col = ; col <= sheet->lastCol(); ++col)
{
const char *text = ReadCellContent(sheet, row, col);
if (text == NULL || _stricmp(text, "EOF") == )
{
break;
}
titles.push_back(text);
}
} while ();
ele->LinkEndChild(group);
}
// 第一个为空,第二个不为空,表示是数据列
else if (IsCellEmpty(firstCellType) && !IsCellEmpty(secondCellType))
{
tinyxml2::XMLElement *cfg = xmlDoc->NewElement("cfg");
group->LinkEndChild(cfg);
for (uint32_t col = ; col < titles.size(); ++col)
{
const char *text = ReadCellContent(sheet, row, col + );
if (NULL == text)
{
text = formatErrorMsg_CellValueNull;
}
cfg->SetAttribute(titles[col].c_str(), text);
}
}
}
return enmErrorDef_OK;
} void GetPathExtensionName(const char *filePath, char ext[], const uint32_t extLen)
{
int32_t len = strlen(filePath), lastSep = , m = ;
for (lastSep = len - ; lastSep >= ; --lastSep)
{
if (filePath[lastSep] == '.' || filePath[lastSep] == '\\' || filePath[lastSep] == '/')
{
break;
}
}
if (lastSep < )
{
lastSep = ;
}
for (; lastSep < len; ++lastSep)
{
ext[m++] = filePath[lastSep];
}
ext[m] = '\0';
} bool IsConfigSheet(const char *sheetName)
{
if (NULL == sheetName)
{
return false;
}
return sheetName[] == '_';
} bool IsCellEmpty(int32_t cellType)
{
return (cellType == libxl::CELLTYPE_BLANK) || (cellType == libxl::CELLTYPE_EMPTY);
} const char* ReadCellContent(libxl::Sheet *sheet, const int32_t row, const int32_t col)
{
int32_t cellType = sheet->cellType(row, col);
switch (cellType)
{
case libxl::CELLTYPE_STRING:
do
{
const char *text = sheet->readStr(row, col);
// 先判断是不是全是acsii字符
bool isAscii = true;
int32_t textLen = strlen(text);
for (int32_t i = ; i < textLen; ++i)
{
if (!isascii(text[i]))
{
isAscii = false;
break;
}
}
// 如果是,则不用转换格式
if (isAscii)
{
return text;
}
// 否则,需要转成UTF8格式
else
{
const int32_t textUtf8StrLen = * ;
static char textUtf8Str[textUtf8StrLen];
static WCHAR wcharTemp[textUtf8StrLen];
GBKToUTF8(text, wcharTemp, textUtf8Str, textUtf8StrLen);
return textUtf8Str;
}
} while ();
case libxl::CELLTYPE_NUMBER:
do
{
const int32_t doubleStrLen = ;
static char doubleStr[doubleStrLen];
sprintf_s(doubleStr, doubleStrLen, "%f", sheet->readNum(row, col));
for (int32_t i = strlen(doubleStr) - ; i >= ; --i)
{
if (doubleStr[i] > '' && doubleStr[i] <= '')
{
break;
}
else if ((doubleStr[i] == '' || doubleStr[i] == '.') && (i != ))
{
doubleStr[i] = ;
}
}
return doubleStr;
} while ();
default:
break;
}
return NULL;
} void GBKToUTF8(const char* asciiBuf, WCHAR wcharTmp[], char utf8Buf[], const int32_t utf8BufLen)
{
int32_t len = MultiByteToWideChar(CP_ACP, , asciiBuf, -, NULL, );
MultiByteToWideChar(CP_ACP, , asciiBuf, -, wcharTmp, len);
len = WideCharToMultiByte(CP_UTF8, , wcharTmp, -, NULL, , NULL, NULL);
WideCharToMultiByte(CP_UTF8, , wcharTmp, -, utf8Buf, len, NULL, NULL);
} void GetFilesFromDirectory(std::vector<std::string> &files, const std::string &directoryPath)
{
struct _finddata_t fileinfo;
long hFile = ;
char tmpPath[MAX_PATH] = { };
sprintf_s(tmpPath, "%s\\*", directoryPath.c_str());
if ((hFile = _findfirst(tmpPath, &fileinfo)) == -){ return; }
do
{
if ((fileinfo.attrib & _A_SUBDIR))
{
if (strcmp(fileinfo.name, ".") != && strcmp(fileinfo.name, "..") != )
{
sprintf_s(tmpPath, "%s\\%s", directoryPath.c_str(), fileinfo.name);
GetFilesFromDirectory(files, tmpPath);
}
}
else
{
sprintf_s(tmpPath, "%s\\%s", directoryPath.c_str(), fileinfo.name);
files.push_back(tmpPath);
}
} while (_findnext(hFile, &fileinfo) == );
_findclose(hFile);
} int32_t ExcelToXmls(const char *excelFileDirectory, const char *xmlFileDirectory)
{
char xmlFilePath[MAX_PATH];
std::vector<std::string> files;
GetFilesFromDirectory(files, excelFileDirectory);
for (uint32_t i = ; i < files.size(); ++i)
{
printf("%s\n", files[i].c_str());
int32_t ret = CalculateXmlFilePath(excelFileDirectory, xmlFileDirectory, files[i].c_str(), xmlFilePath, MAX_PATH);
if (ret != enmErrorDef_OK){ return ret; }
ret = CreateDirectory(xmlFilePath);
if (ret != enmErrorDef_OK){ return ret; }
ExcelToXml(files[i].c_str(), xmlFilePath);
printf("\t%s\n", xmlFilePath);
}
return enmErrorDef_OK;
} int32_t CalculateXmlFilePath(const char *excelFileDirectory, const char *xmlFileDirectory, const char *excelFilePath, char xmlFilePath[], const int32_t xmlFilePathMaxLen)
{
int32_t excelFileDirectoryLen = strlen(excelFileDirectory);
int32_t excelFilePathLen = strlen(excelFilePath);
if (excelFileDirectoryLen > excelFilePathLen){ return enmErrorDef_CalculateXmlFilePathError; }
sprintf_s(xmlFilePath, xmlFilePathMaxLen, "%s%s", xmlFileDirectory, excelFilePath + excelFileDirectoryLen);
int32_t xmlFilePathLen = strlen(xmlFilePath);
for (int32_t i = xmlFilePathLen - ; i >= ; --i)
{
if (xmlFilePath[i] == '.')
{
xmlFilePath[i + ] = 'x';
xmlFilePath[i + ] = 'm';
xmlFilePath[i + ] = 'l';
xmlFilePath[i + ] = ;
break;
}
}
return enmErrorDef_OK;
} int32_t main()
{
for (int32_t i = ; i < ; ++i)
{
ExcelToXmls("excel", "xml");
}
system("pause");
return ;
} int32_t CreateDirectory(const char *directoryPath)
{
int32_t dirPathLen = strlen(directoryPath);
if (dirPathLen > MAX_PATH)
{
return enmErrorDef_MakeXmlFilePathError;
}
char tmpDirPath[MAX_PATH] = { };
for (int32_t i = ; i < dirPathLen; ++i)
{
tmpDirPath[i] = directoryPath[i];
if (tmpDirPath[i] == '\\' || tmpDirPath[i] == '/')
{
if (_access(tmpDirPath, ) != )
{
int32_t ret = _mkdir(tmpDirPath);
if (ret != )
{
return enmErrorDef_MakeXmlFilePathError;
}
}
}
}
return enmErrorDef_OK;
}
完整工程源码 : http://pan.baidu.com/s/1i47z4up 提取码:06sn
C++ 利用 libxl 将 Excel 文件转化为 Xml 文件的更多相关文章
- WPF: 读取XPS文件或将word、txt文件转化为XPS文件
读取XPS格式文件或将doc,txt文件转化为XPS文件,效果图如下: 1.XAML页面代码: <Window x:Class="WpfWord.MainWindow" xm ...
- WFP: 读取XPS文件或将word、txt文件转化为XPS文件
读取XPS格式文件或将doc,txt文件转化为XPS文件,效果图如下: 1.XAML页面代码: <Window x:Class="WpfWord.MainWindow" ...
- 怎样将word文件转化为Latex文件:word-to-latex-2.56具体解释
首先推荐大家读一读这篇博文:http://blog.csdn.net/ibingow/article/details/8613556 --------------------------------- ...
- jupyter命令把.ipynb文件转化为.py文件
jupyter nbconvert --to script *.ipynb 就能把当前文件夹下面的所有的.ipynb文件转化为.py文件
- 使用vivado将bit文件转化为mcs文件
使用vivado将bit文件转化为mcs文件 1.在Tcl Console中运行脚本: write_cfgmem -force -format MCS -size 64 -interface spix ...
- 将caj文件转化为pdf文件进行全文下载脚本(ubuntu下亲测有用)
最近ubuntu下caj阅读器,突然崩掉了,而偏偏要准备开题,在网上搜索原因未果,准备放弃时候,突然在网上看到一个脚本,说是很好用,可以在指定页面将caj文件转化为pdf文件,亲测有用,这里直接给出脚 ...
- Netlib文件转化为mps文件
Netlib文件转化为mps文件 简单方法1 下载并执行: git clone https://github.com/mtanneau/Netlib_experiments.git cd Netlib ...
- 使用Pull解析器生成XML文件和读取xml文件
有些时候,我们需要生成一个XML文件,生成XML文件的方法有很多,如:可以只使用一个StringBuilder组拼XML内容,然后把内容写入到文件中:或者使用DOM API生成XML文件,或者也可以使 ...
- Maven入门2-pom.xml文件与settings.xml文件
Maven入门2-pom.xml文件与settings.xml文件 本文内容来源于官网文档部分章节,settings.xml文件:参考http://maven.apache.org/settings. ...
随机推荐
- Python~切片Slice
[:] [:10] [-10:] [':10:2]
- c/c++ printf
%d int %s string %p point值 %c char %lu long unsigned int
- 关于kali2.0 rolling无法连接数据的解决办法
在使用kali2.0时,经常遇到metasploit无法连接数据库的问题,经过在网上寻找资料,很多都是对kali2.0的,有些路径都是不同的了,所以未能解决, 最后在自己的摸索下,找到了如下方法: 其 ...
- PHP表单与验证
一.Header()函数 标头 (header) 是服务器以 HTTP 协议传 HTML 资料到浏览器前所送出的字符串,在标头与 HTML 文件之间尚需空一行分隔. /*一.Header()函数 */ ...
- 解析 Linux 内核可装载模块的版本检查机制
转自:http://www.ibm.com/developerworks/cn/linux/l-cn-kernelmodules/ 为保持 Linux 内核的稳定与可持续发展,内核在发展过程中引进了可 ...
- 3.OGG函数
1.COMPUTE 一般用来计算目标端和源段的字段值,使用这个函数计算出目标端对应列想要的值 Use the @COMPUTE function to return the value of ...
- LCA + 树状数组 + 树上RMQ
题目链接:http://poj.org/problem?id=2763 思路:首先求出树上dfs序列,并且标记树上每个节点开始遍历以及最后回溯遍历到的时间戳,由于需要修改树上的某两个节点之间的权值,如 ...
- MIT 6.828 JOS学习笔记9. Exercise 1.5
Lab 1 Exercise 5 再一次追踪一下boot loader的一开始的几句指令,找到第一条满足如下条件的指令处: 当我修改了boot loader的链接地址,这个指令就会出现错误. 找到这样 ...
- win10 系统怎么获取最高管理员权限删除文件
http://www.xitongcity.com/jiaocheng/win8_content_3473.html 很多win8.1系统用户在对磁盘文件进行清理时,经常会遇到“文件夹访问被拒绝,您需 ...
- Pig语言基础-【持续更新中】
***本文参考了Pig官方文档以及已有的一些博客,并加上了自己的一些知识性的理解.目前正在持续更新中.*** Pig作为一种处理大规模数据的高级查询语言,底层是转换成MapReduce实现的, ...