国际化是使程序具有足够的灵活性、能在世界上任何地区运行的过程。国际化所要求的必然结果是地方化――使一个程序能够运行在特定地区的过程。本文尝试用一个简单的例子来演示java用户界面本地化。java语言内核基于unicode3.0(java 1.4)提供了对不同国家和不同语言文字的内部支持,由于先天的原因,java对于国际化的支持远远要比c/c++来的优越。
在我看来本地化必须满足以下的三个条件:
1、程序必须能读、写和操作本地化的文本。
2、程序在显示日期和时间、使数字格式化以及排序子串时,必须符合地方习惯。(通过java.text包里面的类可以实现这些要求)
3、所有用户可见的文本都能在运行时获得,而不是直接写入程序中。(通过java.util包里的resourcebundle类和他的子类可以实现这些要求。)
实现这三个方面可以真正实现程序的国际化。
首先让我们来了解一下地区。地区代表一个地理上、政治上或文化上的区域。在java中,地区由java.util.locale类表示。地区常常以一种语言来定义,该语言则由其标准的小写双字母代码表示。(例如:en代表英国,fr代表法国,zh代表中国),但有时候语言是不能代表一个地区的,那就要在语言后面再加上一个国家或该国家的地域(例如:en_us代表美国,zh_tw)。locale类保存着一个静态的默认地区,它可以用locale.setdefault()和locale.getdefault()来设置和查询。一个程序可以生成和使用任意数目的非默认locale对象。
让我们再来看一下unicode字符编码。java使用unicode的字符编码,其本身就是迈向国际化的一大步。unicode编码其每个字符都占两个字节。用\u****的形式表示。unicode的字符可以等价于其他编码的字符(例如:从\u0020到\u007e的字符等价于ascii和iso8859-1字符的0x20到0x7e)。
本文主要是对用户界面地方化,由于我使用的是资源束!所以有必要对资源束作一下解释。
为定义一束地方化的资源,你需要生成一个resourcebundle(资源束)的子类并且提供handlegetobject()和getkeys()方法的定义。为了在程序中使用来自resourcebundle的地方化资源,就需要先调用静态的getbundle()方法,用getbundle()获得一个resourcebundle对象,然后再用getobject()方法去按照名字来查找资源。当然也可以使用getstring()简单的把getobject()的返回值分配给一个string对象。getbundle()方法采用basename_language_country_variait—-没找到的话->basename_language_country—-没找到的话->basename_language—-没找到的话->basename(默认资源文件)的算法寻找合适的资源。如果以上都没找到的话,则会抛出一个missingresourceexception异常。
现在我们来看一个简单的例子,如何使java程序用户界面地方化的。
首先我们的程序需要查找特定locale对象关联的资源包,所以应该定义一个local对象,来获取本地默认的地区!然后可以调用resourcebundle的getbundle方法,并将locale对象作为参数传入。
清单一:
locale locale = locale.getdefault(); //获取地区:默认
//获取资源束。如未发现则会抛出missingresourceexception异常
resourcebundle bundle = resourcebundle.getbundle("properties.dorian",locale);
清单一中的”properties.dorian”代表properties包下以dorian命名的默认资源文件。这样就可以使用资源文件了!让我们来看看资源文件是如何定义的。
清单二:
# dorian.properties是默认的"dorian"资源束文件。
# 作为中国人,我用自己的地区作为默认
title=\u4e2d\u56fd;
red.label=\u7ea2\u8272;
green.label=\u7eff\u8272;
blue.label=\u84dd\u8272;
清单三:
# 文件dorian_en_us.properties,是美国地区的资源束
# 它覆盖了默认资源束
title=america;
red.label=red;
green.label=green;
blue.label=blue ;
清单一和二定义了一个默认资源文件,和美国地区的资源文件。其中等号左边的字符串表示主键,它们是唯一的。为了获得主键对应的值,你可以调用resourcebundle类的getstring方法,并将主键作为参数。此外,文件中以“#”号开头的行表示注释行。需要注意的是清单二中的“\u4e2d\u56fd”,它是字符“中国”的unicode字符码。是使用java自带的native2ascii工具转换的(native2ascii in.properties out.properties),这是为了不在程序界面中产生乱码。
清单四:
cmdred.settext(bundle.getstring("red.label"));
cmdblue.settext (bundle.getstring("blue.label"));
cmdgreen.settext (bundle.getstring("green.label"));
清单二中的cmdred、cmdblue、cmdgreen 为按钮。bundle.getstring("red.label")为得到资源文件中主键是red.label的值。
好了到此为止java程序用户界面的本地化就是这么简单。不过,要提醒你的是在为用户界面事件编写事件监听器代码时,要格外小心。请看下面这段代码。
清单五:
public class myapplet extends japplet implements actionlistener{
public void init(){
jbutton cancelbutton=new jbutton(“cancel”);
cancelbutton.addactionlistener(this);
…
}
public void actionperformed(actionevent e){
string s=e.getactioncommand();
if(arg.equals(“cancel”);
docancel();
else ……
}
}
如果你对清单五的代码不进行本地化,她就可能会运行的很好。但当你的按钮被本地化为中文时,“cancel”变为了“取消”。这时就会出现你不愿意看到的问题。下面有三个方法可以消除这个潜在的问题!
1> 使用内部类而不使用独立的actionperformed程序。
2> 使用引号而不使用标签来标识组件。
3> 使用name属性来标识组件
本例稍后的代码就是采用第一种方法来消除这个问题的。
清单六:完整的代码
//:mynative.java
/**
copyright (c) 2003 dorian. all rights reserved
@(#)mynative.java 2003-12-21
@author dorian
@version 1.0.0
visit http://www.dorian.com/java/
*/
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import java.util.*;
/**
这是一个将java程序界面地方化的例子本例采用读取属性文件来达到目的
@see java.util.locale;
@see java.util.resourcebundle;
@see java.util.missingresourceexception;
*/
public class mynative{
public static void main(string[] args){
jframe frame = new mynativeframe();
frame.setdefaultcloseoperation(jframe.exit_on_close);
frame.setresizable(false);
frame.setvisible(true); // pop the window up.
}
}
class mynativeframe extends jframe{
public mynativeframe(){
locale locale = locale.getdefault();//获取地区:默认
//获取资源束。如未发现则会抛出missingresourceexception异常
//"properties.dorian"为在properties下以dorian为文件名的默认属性文件
resourcebundle bundle = resourcebundle.getbundle("properties.dorian",locale);
settitle(bundle.getstring("title"));//通过getstring()的返回值来设置title
setsize(width,height); // set the window size.
panel=new mynativepanel();
container contentpane=getcontentpane();
contentpane.add(panel);
//通过获取资源束中*.label的值对三个按钮设置其label
panel.setcmdred(bundle.getstring("red.label"));
panel.setcmdblue(bundle.getstring("blue.label"));
panel.setcmdgreen(bundle.getstring("green.label"));
}
private mynativepanel panel;
private static final int width=400;
private static final int height=100;
}
class mynativepanel extends jpanel{
public mynativepanel(){
layout=new borderlayout();
setlayout(layout);
txt=new jtextfield(50);
add(txt,layout.center);
cmdred=new jbutton();
cmdblue=new jbutton();
cmdgreen=new jbutton();
panel.add(cmdred);
panel.add(cmdblue);
panel.add(cmdgreen);
add(panel,layout.south);
cmdred.addactionlistener(new actionlistener(){
public void actionperformed(actionevent e){
string s = e.getactioncommand();
txt.setbackground(color.red);
txt.settext(s);
}
});
cmdblue.addactionlistener(new actionlistener(){
public void actionperformed(actionevent e){
string s = e.getactioncommand();
txt.setbackground(color.blue);
txt.settext(s);
}
});
cmdgreen.addactionlistener(new actionlistener(){
public void actionperformed(actionevent e){
string s = e.getactioncommand();
txt.setbackground(color.green);
txt.settext(s);
}
});
}
public void setcmdred(string s){
cmdred.settext(s);
}
public void setcmdblue(string s){
cmdblue.settext(s);
}
public void setcmdgreen(string s){
cmdgreen.settext(s);
}
jpanel panel=new jpanel();
borderlayout layout;
private jtextfield txt;
private jbutton cmdred,cmdblue,cmdgreen;
}
//~
资源文件:
# dorian.properties是默认的"dorian"资源束文件。
# 作为中国人,我用自己的地区作为默认
title=\u4e2d\u56fd
red.label=\u7ea2\u8272
green.label=\u7eff\u8272
blue.label=\u84dd\u8272
# 文件dorian_en_us.properties,是美国地区的资源束
# 它覆盖了默认资源束
title=america
red.label=red
green.label=green
blue.label=blue
# 文件dorian_zh_cn.properties,是中国大陆地区的资源束
# 这个文件没有任何资源定义,从默认中国资源束继承
