MenuItem 显示中文乱码问题解决方案
今天在使用Java做系统托盘图标(TrayIcon),需要为其增加一个右键弹出菜单(PopupMenu),在使用菜单项(MenuItem)时,遇到了一个非常痛苦的事情:中文乱码~~~~。这个问题不经常碰到,但是一旦碰到就头疼了,网上一搜,方法一大堆,但很少有方法能解决自己的问题,毕竟情况不一样。
?
今天搞了一天,终于搞出了一套方案。
?
?
先说一下出问题的原因:
?
?
1. 本地系统区域语言字符集的问题,我们的系统可能默认的字符集为GB2312或GBK。
?
2. Java源文件编码字符的问题,如果使用eclipse可以查看一下java source文件的编码方式。
?
3. javac编译时的字符问题,这一个问题很容易被忽略,运行一下javac可以看到有一个encoding的参数可以设置----这一个很重要。
?
4. 要读取文件的编码和读取时的字符集设置问题,如果我们要用的字符串不是硬编码在java源码中,而是从资源文件中读取时,就要注意这个问题了。
?
?
要解决问题需要做到如下:
?
1. 本地系统的语言字符可以不用管,主要看后面三项。
?
2. Java源文件的编码,这一点很重要,最好使用utf8编码。
我们在编辑文件时默认使用的是本地系统的字符集(如GBK),所以对Java源文件要进行字符转换或提前做好设置,对于UE使用 文件--->转换--->...到UTF-8(Unicode 编辑),对于eclipse设置 ?Window-->Preferences-->General-->Content Types--> Java Source File。具体操作此处不详述。
?
3. javac编译时的参数设置,增加encoding参数,如:javac -encoding utf8 Test.java
对于这一点要特别注意,eclipse的编译器是没有使用该参数的,我也没找到该如何设置该参数(注意这里是编译参数javac, 不是运行参数java或jvm, 不是run config中配置的),所以遇到这个问题的情况下不能使用eclipse来编译了,至少在能配置javac之前是这样。
不能使用eclipse了怎么办呢,手工不是太可能,有几种方案:
1. 手工编译使用到MenuItem的类,目前只发现java.awt包会存在这个问题,swing包能很好的解决这种字符集变换的问题。
2. 使用其它编译方式,如ant、Maven,它们都能配置javac。我现在就使用maven,只需要为它的编译器加入encoding配置,如下:
?
??
好了,做到以上条件问题应该解决了。
?
在实际的过程中我们可以通过以下步骤一步一步来达到以上条件,每步都可以测试。
?
1. 查看源文件的编码
这一点很简单,对于eclipse查看一下java source file编码方式就行了,保险一点就是修改一下编码方式,会发现原来的java文件(含中文的)乱码了,再改回来,就好了。 编码方式请设置为utf8
?
2. 编译编码的测试
可以先写一个小例子,使用硬编码的方式加入中文,如MenuItem m = new MenuItem("中文"),你用eclipse运行后可以发现这个menu显示的是乱码(全是小方框)。然后你用命令行的方式编译一下:javac -encoding utf8 Test.java。然后命令行运行,如果发现显示正常了,则这及以前的测试通过了,如果还是乱码,则需要你再次确认第1步。
?
3. 在前2部都确定成功的情况下再考虑读取资源文件的测试,这个参照前面说的就行了。
?
?
--OK,问题应该能解决了,这种方式还可以应用到其它中文乱码的问题上。
?
?
补充说明的是:awt不是不支持中文,是编码转换不够智能。在swing中就解决了这些问题。
关于这个TrayIcon,我看oracle那边已经提交了一个bug,意思是说要编写一个对应的swing的JTrayIcon,从而使用JPopupMenu和JMenuItem。但这个bug不是因字符问题提交的。
对于MenuItem乱码的问题,还可以使用JPopupMenu和JMenuItem来替代,参照如下,但情况不理想(弹出时必须要选择一个菜单,否则不消失):
?
ImageIcon icon = new ImageIcon(UITest.class.getResource("16.gif"));TrayIcon tray = new TrayIcon(icon.getImage());tray.setImageAutoSize(true);SystemTray.getSystemTray().add(tray);tray.addMouseListener(new MouseAdapter() {@Overridepublic void mouseReleased(MouseEvent e) {if (e.isPopupTrigger()) {final JPopupMenu pop = new JPopupMenu();JMenuItem m1 = new JMenuItem("中文");pop.add(m1);pop.add(new JMenuItem("主题"));pop.setLocation(e.getX(), e.getY());pop.setInvoker(pop);pop.setVisible(true);}}});?另外,对于第4点,资源文件的问题,还可以使用Properties.load()方法来加载.properties文件(该文件是被java bin下的navie2asc程序处理过的文件),这样就不用管第4点问题了,但问题很显然,不方便不直观。
?
---EOF---
?
?
?