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 = 16,
};

enum ErrorDef
{
    enmErrorDef_OK = 0,
    enmErrorDef_CreateBookFailed = 1,
    enmErrorDef_NotExcelFile = 2,
    enmErrorDef_LoadExcelFileFailed = 3,
    enmErrorDef_SaveXmlFileFailed = 4,
    enmErrorDef_CalculateXmlFilePathError = 5,
    enmErrorDef_MakeXmlFilePathError = 5,
};

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] = { 0 };
    GetPathExtensionName(excelFilePath, ext, enmFileExtensionLength);
    if (strcmp(ext, ".xls") == 0)
    {
        book = xlCreateBook();
    }
    else if (strcmp(ext, ".xlsx") == 0)
    {
        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 = 0; i < book->sheetCount(); ++i)
    {
        if (!IsConfigSheet(book->getSheet(i)->name())){ continue; }
        tinyxml2::XMLElement *ele = xmlDoc->NewElement(book->getSheet(i)->name() + 1);
        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 = 0; row < sheet->lastRow(); ++row)
    {
        // 获取该行前两个单元格
        int32_t firstCellType = sheet->cellType(row, 0);
        int32_t secondCellType = sheet->cellType(row, 1);
        // 如果都是空白,表明是空白行
        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, 0);
                if (NULL == firstCell)
                {
                    group = xmlDoc->NewElement(formatErrorMsg_GroupNameNull);
                    break;
                }
                group = xmlDoc->NewElement(firstCell);
                for (int32_t col = 1; col <= sheet->lastCol(); ++col)
                {
                    const char *text = ReadCellContent(sheet, row, col);
                    if (text == NULL || _stricmp(text, "EOF") == 0)
                    {
                        break;
                    }
                    titles.push_back(text);
                }
            } while (0);
            ele->LinkEndChild(group);
        }
        // 第一个为空,第二个不为空,表示是数据列
        else if (IsCellEmpty(firstCellType) && !IsCellEmpty(secondCellType))
        {
            tinyxml2::XMLElement *cfg = xmlDoc->NewElement("cfg");
            group->LinkEndChild(cfg);
            for (uint32_t col = 0; col < titles.size(); ++col)
            {
                const char *text = ReadCellContent(sheet, row, col + 1);
                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 = 0, m = 0;
    for (lastSep = len - 1; lastSep >= 0; --lastSep)
    {
        if (filePath[lastSep] == '.' || filePath[lastSep] == '\\' || filePath[lastSep] == '/')
        {
            break;
        }
    }
    if (lastSep < 0)
    {
        lastSep = 0;
    }
    for (; lastSep < len; ++lastSep)
    {
        ext[m++] = filePath[lastSep];
    }
    ext[m] = '\0';
}

bool IsConfigSheet(const char *sheetName)
{
    if (NULL == sheetName)
    {
        return false;
    }
    return sheetName[0] == '_';
}

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 = 0; i < textLen; ++i)
            {
                if (!isascii(text[i]))
                {
                    isAscii = false;
                    break;
                }
            }
            // 如果是,则不用转换格式
            if (isAscii)
            {
                return text;
            }
            // 否则,需要转成UTF8格式
            else
            {
                const int32_t textUtf8StrLen = 1024 * 5;
                static char textUtf8Str[textUtf8StrLen];
                static WCHAR wcharTemp[textUtf8StrLen];
                GBKToUTF8(text, wcharTemp, textUtf8Str, textUtf8StrLen);
                return textUtf8Str;
            }
        } while (0);
    case libxl::CELLTYPE_NUMBER:
        do 
        {
            const int32_t doubleStrLen = 32;
            static char doubleStr[doubleStrLen];
            sprintf_s(doubleStr, doubleStrLen, "%f", sheet->readNum(row, col));
            for (int32_t i = strlen(doubleStr) - 1; i >= 0; --i)
            {
                if (doubleStr[i] > '0' && doubleStr[i] <= '9')
                {
                    break;
                }
                else if ((doubleStr[i] == '0' || doubleStr[i] == '.') && (i != 0))
                {
                    doubleStr[i] = 0;
                }
            }
            return doubleStr;
        } while (0);
    default:
        break;
    }
    return NULL;
}

void GBKToUTF8(const char* asciiBuf, WCHAR wcharTmp[], char utf8Buf[], const int32_t utf8BufLen)
{
    int32_t len = MultiByteToWideChar(CP_ACP, 0, asciiBuf, -1, NULL, 0);
    MultiByteToWideChar(CP_ACP, 0, asciiBuf, -1, wcharTmp, len);
    len = WideCharToMultiByte(CP_UTF8, 0, wcharTmp, -1, NULL, 0, NULL, NULL);
    WideCharToMultiByte(CP_UTF8, 0, wcharTmp, -1, utf8Buf, len, NULL, NULL);
}

void GetFilesFromDirectory(std::vector<std::string> &files, const std::string &directoryPath)
{
    struct _finddata_t fileinfo;
    long hFile = 0;
    char tmpPath[MAX_PATH] = { 0 };
    sprintf_s(tmpPath, "%s\\*", directoryPath.c_str());
    if ((hFile = _findfirst(tmpPath, &fileinfo)) == -1){ return; }
    do
    {
        if ((fileinfo.attrib &  _A_SUBDIR))
        {
            if (strcmp(fileinfo.name, ".") != 0 && strcmp(fileinfo.name, "..") != 0)
            {
                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) == 0);
    _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 = 0; 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 - 1; i >= 0; --i)
    {
        if (xmlFilePath[i] == '.')
        {
            xmlFilePath[i + 1] = 'x';
            xmlFilePath[i + 2] = 'm';
            xmlFilePath[i + 3] = 'l';
            xmlFilePath[i + 4] = 0;
            break;
        }
    }
    return enmErrorDef_OK;
}

int32_t main()
{
    for (int32_t i = 0; i < 1; ++i)
    {
        ExcelToXmls("excel", "xml");
    }
    system("pause");
    return 0;
}

int32_t CreateDirectory(const char *directoryPath)
{
    int32_t dirPathLen = strlen(directoryPath);
    if (dirPathLen > MAX_PATH)
    {
        return enmErrorDef_MakeXmlFilePathError;
    }
    char tmpDirPath[MAX_PATH] = { 0 };
    for (int32_t i = 0; i < dirPathLen; ++i)
    {
        tmpDirPath[i] = directoryPath[i];
        if (tmpDirPath[i] == '\\' || tmpDirPath[i] == '/')
        {
            if (_access(tmpDirPath, 0) != 0)
            {
                int32_t ret = _mkdir(tmpDirPath);
                if (ret != 0)
                {
                    return enmErrorDef_MakeXmlFilePathError;
                }
            }
        }
    }
    return enmErrorDef_OK;
}

 

完整工程源码 : http://pan.baidu.com/s/1i47z4up 提取码:06sn

 

posted @ 2016-11-27 21:35  你好阿汤哥  Views(2109)  Comments(2Edit  收藏  举报