自动将visual stdio创建的文件从gb2312编码转换成utf8编码

来源:互联网 发布:7.3魔兽世界mac客户端 编辑:程序博客网 时间:2024/05/09 02:48

cocos2d-x是一个不错的手机游戏框架,不过由于在中文的windows系统下使用visual stdio2008默认创建的类文件,包括.h和.cpp,其文件编码都是gb2312的。当将这些文件在cygwin上用ndk编译时,虽然编译没问题,但其中的中文在读取的时候会出现乱码,这是cocos2d-x的一个不足,例如CCLabelTTF显示中文的时候会出现乱码,而且CCSpriteFrame的spriteFrameByName方法也不能传入中文,这确实是一件头疼的事。

解决方法有两种,一种是将源代码文件保存为utf8格式,另一种方法在使用中文字符的时候转码,如果使用第二种方法的,为了跨平台,一般使用iconv这个库进行编码转换。但由于iconv是基于LGPL协议,cocos2d-x已经在最新的 cocos2d-1.0.1-x-0.12.0 release里将iconv的静态库去掉了,使用其实并不建议大量频繁地调用iconv进行编码转换,毕竟效率有所降低。

第二种方法则是将源代码文件编码转换为utf8编码,手动的方法是在visual stdio的文件菜单里有一个高级保存选项,将编码设为Unicode(UTF8带签名)然后保存即可,不过这种方法比较笨,纯粹是体力活。我研究过打算修改一下visual stdio的类向导,想把类向导创建的文件修改为utf8格式,但不得其果。如果是单纯的创建.h和.cpp文件则是可以的,方法是将Microsoft Visual Studio 9.0/VC/vcprojectitems下的newc++file.cpp和hfile.h保存为utf8格式,这样每次创建新的.h和.cpp都将会是utf8格式。但一般来说都会使用类向导创建类文件,所以我写了一个小程序,可以将源代码转换了utf8格式。代码如下:

// ChangeFileEncoding.cpp : 定义控制台应用程序的入口点。//#include "stdafx.h"#include "ChangeFileEncoding.h"#include <string>#ifdef _DEBUG#define new DEBUG_NEW#endif// 唯一的应用程序对象CWinApp theApp;using namespace std;void recursiveFile(CString strFileType);void convertGBToUTF8(CString strWritePath, const char* gb2312);int _tmain(int argc, TCHAR* argv[], TCHAR* envp[]){int nRetCode = 0;// 初始化 MFC 并在失败时显示错误if (!AfxWinInit(::GetModuleHandle(NULL), NULL, ::GetCommandLine(), 0)){// TODO: 更改错误代码以符合您的需要_tprintf(_T("错误: MFC 初始化失败\n"));nRetCode = 1;}else{/*for(int i = 0; i < argc; i++){MessageBox(NULL, argv[i], L"Arglist contents", MB_OK);}*///声明一个CFileFind类变量,以用来搜索//接受一个参数作为源代码文件的根目录TCHAR *lpszDirName = argv[1];CString strFileType;strFileType.Format(_T("%s\\*.*"), lpszDirName);//递归此目录下的.h文件和.cpp文件,如果发现不是utf8编码则转换为utf8编码recursiveFile(strFileType);}return nRetCode;}void recursiveFile( CString strFileType){CFileFind finder; BOOL isFinded = finder.FindFile(strFileType);//查找第一个文件while(isFinded){isFinded = finder.FindNextFile(); //递归搜索其他的文件if(!finder.IsDots()) //如果不是"."目录{CString strFoundFile = finder.GetFilePath(); if(finder.IsDirectory()) //如果是目录,则递归地调用{ CString strNextFileType;strNextFileType.Format(_T("%s\\*.*"), strFoundFile);recursiveFile(strNextFileType);}else{ //如果是头文件或cpp文件if(strFoundFile.Right(4) == _T(".cpp") || strFoundFile.Right(2) == _T(".h")) {CFile fileReader(strFoundFile, CFile::modeRead);byte head[3];fileReader.Read(head, 3); //判断是否带有BOM文件头if(head[0] == 0xef && head[1]==0xbb && head[2] == 0xbf ){fileReader.Close();continue;}fileReader.SeekToBegin();int bufLength = 256;char *buf = new char[bufLength];ZeroMemory(buf, bufLength);int nReadLength;std::string strContent;while((nReadLength = fileReader.Read(buf, bufLength))){strContent.append(buf, nReadLength);ZeroMemory(buf, nReadLength);}delete buf; fileReader.Close();convertGBToUTF8(strFoundFile, strContent.c_str());}}}}finder.Close();}void convertGBToUTF8(CString strWritePath, const char* gb2312){CFile fp;fp.Open(strWritePath, CFile::modeCreate|CFile::modeWrite|CFile::typeBinary,NULL);int len = MultiByteToWideChar(CP_ACP, 0, gb2312, -1, NULL, 0);wchar_t* wstr = new wchar_t[len+1];memset(wstr, 0, len+1);MultiByteToWideChar(CP_ACP, 0, gb2312, -1, wstr, len);len = WideCharToMultiByte(CP_UTF8, 0, wstr, -1, NULL, 0, NULL, NULL);char* str = new char[len+1];memset(str, 0, len+1);len = WideCharToMultiByte(CP_UTF8, 0, wstr, -1, str, len, NULL, NULL);if(wstr) delete[] wstr;str[len] = '\n';const unsigned char aryBOM[]  = {0xEF, 0xBB, 0xBF};fp.Write(aryBOM, sizeof(aryBOM));fp.Write(str,len);delete[] str;fp.Close();} 

编码生成ChangeFileEncoding.exe后,将ChangeFileEncoding.exe放在cocos2d-x的主程序项目的根目录下里,例如项目的目录树如下:

Project

----------\cocos2dx

----------\CocosDenshion

----------\Box2D

----------\MainApplication

这里MainApplication就是主程序项目,将ChangeFileEncoding.exe放在此目录下,然后设置MainApplication的项目属性,在"生成事件" --"预生成事件" --"命令行"里填入“ChangeFileEncoding.exe ./Classes”即可。这样每次添加的文件虽然是gb2312编码,但每次在编译前会自动转换成utf8编码。

需要源代码的请留下邮箱。

原创粉丝点击