国际化问题

chaojinn 2003-07-11 10:00:47
开发一软件,要求有中英语版本。
如何把那些文字资源放到一个资源文件里。
运行是根据所选版本分别显示?
...全文
361 7 打赏 收藏 转发到动态 举报
写回复
用AI写文章
7 条回复
切换为时间正序
请发表友善的回复…
发表回复
无敌魔仙 2003-07-12
  • 打赏
  • 举报
回复
上面先生的想法和我的 如出一辙,呵呵
Skt32 2003-07-12
  • 打赏
  • 举报
回复
用VC资源动态链接库解决国际化问题

● 陈峰
随着计算机应用的普及,应用软件跨国使用越来越频繁,如何实现应用软件的国际化,成为许多程序员关心的问题。这种国际化问题相对使用某一种语言的用户来说,就是本地化问题。本地化的含义比仅仅翻译菜单栏和对话框的文本内容要广泛得多,如特定文化背景的位图和图标对不同的人来说,可能会有不同的含义。在Windows系统中,应用程序开发者可以通过利用VC动态链接库,只用一套源代码就能简洁地支持多种文字。本文介绍用VC的资源动态链接库解决国际化问题的具体方法和步骤。
实现技术
利用VC编程,可以把所有的可见资源封装在一个资源动态链接库中,以简化本地化工作。一般情况下,资源包含在应用程序中,但也可以通过调用AfxSetResourceHandle函数指向一个不同的单元以完成资源的引用。本文例程就是首先调用该函数从动态链接库中采集到所有的应用程序可用资源,然后通过调用GetSystemDefaultLangID函数判断系统默认语种,以载入不同语种的资源动态链接库实现界面与系统的自动适应。
为说明清楚,本文创建了一个默认语种为简体中文的Languages应用程序。该程序不含任何资源,应用程序根据系统的语种设置连接对应的资源链接库,以完成对中文和英文两种语言的支持。
实现步骤
1.创建Languages执行体
首先,用MFC AppWizad(exe)创建新项目的工作区,选择Simple Document类型、中国中文(其他的选项选择缺省),并且为了明确,将工作区目录改为“多语种支持”。
2.添加数据成员
由于要动态地装入资源链接库,所以需要保存链接库的句柄以便在程序结束的时候释放资源。将下述数据成员添加到CLanguagesApp类中:
protected:
//资源链接库句柄
HINSTANCE m_hLangDLL;
3.修改InitInstance函数
应用程序需要判别系统的缺省语种,并装入对应的资源链接库。将下面的代码加到InitInstance函数中:
BOOL CLanguagesApp::InitInstance()
{
 AfxEnableControlContainer();
 // 判定系统缺省语种
 WORD wLangPID=PRIMARYLANGID(::GetSystemDefaultLangID());
 // 载入资源动态链接库
 switch( wLangPID )
 {
  case LANG_CHINESE:
  m_hLangDLL=::LoadLibrary(“chinese.dll”);
    break ;
  default:
   m_hLangDLL=::LoadLibrary(“english.dll”);
break;
}
if( !m_hLangDLL)
{
 AfxMessageBox(_T(“无法装载资源链接库!”)) ;
  return FALSE ;
}
// 连接资源
 AfxSetResourceHandle(m_hLangDLL) ;
……
}
其中,操作系统所使用的默认语种由Win 32函数GetSystemDefaultLangID取得。宏PRIMARYLANGID取出主语言标识符进行判断,以正确选择应该调用的链接库。链接库的加载由Win 32函数LoadLibrary实现。程序中所使用的资源库由AfxSetResourceHandle函数指定。
4.处理ExitInstance函数
程序退出时使用Win 32函数FreeLibrary卸载已经装入的动态链接库。将下列代码添加到ExitInstance函数中:
int CLanguagesApp::ExitInstance()
{
 // 释放资源链接库
 if(m_hLangDLL)
  AfxFreeLibrary(m_hLangDLL);
 return CWinApp::ExitInstance();
}
5.修改OnDraw
为了说明是从资源链接库中动态地获取数据而不是从程序的执行体中获取,该程序从资源链接库中获取了一个图标和一个字符串,并绘制在屏幕上。下面是主要代码:
void CLanguagesView::OnDraw(CDC* pDC)
{
CLanguagesDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
//画出从链接库加载的图标
pDC->DrawIcon(10,10,AfxGetApp()->LoadIcon(IDR_MAINFRAME));
//显示字符串
CString strMsg;
strMsg.LoadString(IDS_HELLO);
pDC->TextOut(60,15,strMsg);
}
6.从项目中移走Languages.rc
由于本文项目自身并不需要含有任何资源,所以从项目列表文件中移走Languages.rc文件(并不删除)。ClassWizad不允许项目不包含资源文件,所以可以创建一个空资源文件并加入到项目中,然后重建信息文件(Languages.CLW),使得ClassWizad可以继续管理程序的消息处理。但在本例由于所有的函数都已经加入,可不必为项目创建空资源文件。
7.建立项目
修改设置,从Project菜单选择Setting…选项。在Link标签下将输出文件名由\Debug\Languages.exe改为Languages.exe。编译项目,由于应用程序中没有任何资源,会出现找不到资源链接库的错误,下面的一系列步骤为本项目分别创建简体中文和美国英语的资源链接库。在这两个链接库建立后,把它们放到Languages项目目录中,程序就可以运行了。
建立中文资源链接库
1.创建新项目
为了统一起见,将新创建的Language_chinese项目和Languages项目放在同一工作区内,并将项目目录作为工作区目录“多语种支持”的子目录。选择Regular DLL-based应用程序并按下Finish键完成创建。动态链接库项目只包含资源,不需要其他文件,所以从项目中删除所有其他文件。
2.拷贝资源文件
把Languages相关的资源文件拷贝到Language_chinese项目的对应目录下。表1是需要拷贝的文件名。
表1
文件名称 注释
Languages.rc 资源描述文件
Resource.h 包含资源定义的头文件
Res\Languages.ico 主程序图标
Res\Languages.rc2 用于定义资源的描述文件
Res\LanguagesDoc.ico 文档图标
Res\Toolbar.bmp 工具条位图
3.将资源文件添加到项目中
把Languages.rc加入到Language_chinese项目中,使得项目中的唯一文件是Languages.rc。
4.修改资源
将ID_MAINFRAME的图标改成代表简体中文的图标,该图标将在Languages应用程序显示窗口中出现。在串资源表中添加串IDS_HELLO=200,标题为“嗨,这是中文版。”。
5.修改连接编译设置
从Project菜单中选择Setting…选项。首先,将输出文件名由\Debug\Language_chinese.dll改为..\chinese.dll,将动态链接库直接建立在Languages项目目录下。然后在链接命令行中添加“/NOENTRY”选项。最后,编译并建立动态链接库。
也可以建立一个初始化链接库原文件,这样就无需在链接命令行中添加“/NOENTRY”选项。程序主要代码如下:
#include <windows.h>
extern “C”
BOOL WINAPI DllMain( HINSTANCE hInstance, DWORD dwReason, LPVOID )
{
return 1;
}
建立英文资源动态链接库
生成英文版资源文件最快捷的方法是创建一个基于英文版的临时项目,并把资源文件拷到英文链接库项目中。其处理方法类似于中文资源链接库。
需要注意的是:在修改资源组件设置时,不但要将Project菜单Setting对话框中Resource标签的缺省文字改为“英语(美国)”,还要删除预处理定义中的“_AFXDLL”。删除_AFXDLL预处理器,可以使用户的链接库包含通用的MFC资源。否则,应用程序会从系统安装的MFC动态链接库中收集通用的MFC资源,应用程序特定资源会被本地化。
小 结
Windows应用程序的资源一般捆绑在执行程序上,当需要时才将其装入。当需要某资源时,需要给Windows提供两个参数:应用程序的实例句柄和资源的标识符。当某个链接库装入时,Windows返回一个实例句柄,可以通过该句柄引用链接库中的资源。
需要注意的是:由于内存不够或其他的一些资源问题,可能会导致加载资源失败。
chaojinn 2003-07-12
  • 打赏
  • 举报
回复
谢谢上面的几位。
有人能教教我传统的做法吗?
“把所有的资源都放到动态库中,一种语言一个动态库,程序运行的时候通过加载不同的动态库来实现多语言功能。”具体怎么做?
我不需要动态切换。
Richuen22 2003-07-11
  • 打赏
  • 举报
回复
gz,笨的方法是在英文WINDOWS下编译为一个版,在中文WINDOWS下编译为另一版。
Skt32 2003-07-11
  • 打赏
  • 举报
回复
http://www.vchelp.net/vchelp/file2003_2/multi_lan.zip
Skt32 2003-07-11
  • 打赏
  • 举报
回复
http://www.vchelp.net/vchelp/file2003_2/multi_lan.asp?type_id=73&class_id=1&cata_id=2&article_id=993
Skt32 2003-07-11
  • 打赏
  • 举报
回复
程序实现多国语言的动态切换解决方案


--------------------------------------------------------------------------------

分类: 系统

作者姓名: 耿海增
邮件地址: genghaizeng@163.com
作者相关信息:

程序下载



开发环境: VC6 Windows XP
测试环境: WindowsXP

更新记录:

使用许可:免费

讲解:
实现思想:传统的做法是把所有的资源都放到动态库中,一种语言一个动态库,程序运行的时候通过加载不同的动态库来实现多语言功能。这样做的缺点是不能动态切换语言,如果更换语言后必须重新启动软件。当然,没有人会需要经常的切换语言玩儿,但是采用动态库的方法,如果程序需要修改资源的话,就要更新所有的动态库,这是一个非常枯燥而且容易出现疏漏的工作。

我的方法是把所有用到的字符串都放到文件中,一种语言一个文件,根据选择的语言到对应的文件中去加载字符串。这样不但可以动态切换语言,而且用户可以根据需要自己添加新的语言。

具体实现:

1、程序启动时检查所选择的语言,确定该语言文件,保存该文件路径

2、提供一个全局函数,如 g_LoadString(CString szID),根据提供的字符串ID返回其内容,具体是哪个语言的在函数中判断,这样在程序中只要提供一个字符串ID就可以自动加载不同语言的文字了。

3、如何在程序中使用:

原来代码:

CString str;

str = "语言";

改动后:

CString str;

str = g_LoadString("IDS_LANGUAGE");

4、对话框中如何实现

如果像在程序中使用一样,每一个字符串都要去加载一次的话,如果对话框比较多,工作量可就太大了。所以我提供了一个函数g_SetDialogStrings(CDialog *pDlg,UINT uDlgID),每个对话框在初始化的时候调 用该函数,传递对话框的指针,我在函数中循环枚举所有的子控件,逐个设置文字。这样就可以省去很多工作

注意:由于静态文本(CStatic)默认的ID是IDC_STATIC,值都是65535,无法区分,所以在需要改变其文字 的CStatic的ID要改一改,不能用默认的

5、语言文件样例:

中文版:

[Setting]

Language=Chinese

[String]

IDS_MENU_FILE=文件

IDS_MENU_FILE_NEW=新建(&N)

IDS_MENU_FILE_OPEN=打开(&O)

IDS_MENU_FILE_CLOSE=关闭(&C)

IDS_MENU_FILE_EXIT=退出(&E)

英文版:

[Setting]

Language=English

[String]

IDS_MENU_FILE=File

IDS_MENU_FILE_NEW=&New

IDS_MENU_FILE_OPEN=&Open

IDS_MENU_FILE_CLOSE=&Close

IDS_MENU_FILE_EXIT=&Exit
界面图片:



部分关键代码

/*********************************************************************
* 函数名称:g_LoadString
* 说明: 根据标识 szID到选定的语言文件中加载字符串
* 作者: Geng
*********************************************************************/
CString g_LoadString(CString szID)
{
CString szValue;
DWORD dwSize = 1000;
GetPrivateProfileString("String",szID,"Not found",
szValue.GetBuffer(dwSize),dwSize,g_szLanguagePath);
szValue.ReleaseBuffer();

szValue.Replace("\\n","\n"); //替换回换行符号

return szValue;
}

/*********************************************************************
* 函数名称:g_SetDialogStrings(CDialog *pDlg,UINT uDlgID)
* 说明: 当对话框运行时获取其所有可得到的字符串,并保存到语言文件中
每个控件用对话框ID和控件ID唯一标识

* 入口参数:
* CDialog *pDlg -- 对话框的指针
* UINT uDlgID -- 该对话框的ID
* 作者: Geng
*********************************************************************/
void g_SetDialogStrings(CDialog *pDlg,UINT uDlgID)
{
CString szSection = "String";
CString szKey,szText;
bool bSetText = 1; //1:从文件读,设置对话框
//0:从对话框读,保存到文件

if(bSetText) //1:从文件读,设置对话框
{
CString szDefault = "";
DWORD dwSize = 1000;
char* pData = (char*)malloc(dwSize);

//读对话框标题
szKey.Format("IDD%d_Title",uDlgID);
if(GetPrivateProfileString(szSection,szKey,szDefault,
pData,dwSize,g_szLanguagePath) != 0)
{
pDlg->SetWindowText(pData);
}

//写入各个子控件的标题文字
CWnd* pWnd = pDlg->GetWindow(GW_CHILD);
while(pWnd != NULL)
{
szKey.Format("IDD%d_%d",uDlgID,pWnd->GetDlgCtrlID());
if(GetPrivateProfileString(szSection,szKey,szDefault,
pData,dwSize,g_szLanguagePath) != 0)
{
pWnd->SetWindowText(pData);
}

pWnd = pWnd->GetWindow(GW_HWNDNEXT);
}

//释放内存
free(pData);
}
else //0:从对话框读,保存到文件
{
//写入对话框标题
szKey.Format("IDD%d_Title",uDlgID);
pDlg->GetWindowText(szText);
WritePrivateProfileString(szSection,szKey,szText,g_szLanguagePath);

//写入各个子控件的标题文字
CWnd* pWnd = pDlg->GetWindow(GW_CHILD);
while(pWnd != NULL)
{
szKey.Format("IDD%d_%d",uDlgID,pWnd->GetDlgCtrlID());
pWnd->GetWindowText(szText);
WritePrivateProfileString(szSection,szKey,szText,g_szLanguagePath);

pWnd = pWnd->GetWindow(GW_HWNDNEXT);
}
}
}

16,551

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Creator Browser
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧