使SWT/JFace支持跨平台

由于SWT的实现机制,在不同平台下,必须引用不同swt*.jar. 由于这个瓶颈,我们要为不同的平台编译不同的版本。但是这是可以避免的。这将是本文要讨论的内容。

我一共google到了3种solution:

1,使用swtjar.jar。

http://mchr3k.github.io/swtjar/

其主页有详细的介绍。但是似乎下载链接已经无效了,一个下载的办法是从github上找到引用了它的项目,比如https://github.com/mchr3k/org.intrace/tree/3a1debcbb831f802219b341fb5e37467b365d443/org.intrace/lib

swtjar.jar的原理,似乎是通过替换掉默认的ClassLoader来实现的。

根据我的测试,使用swtjar.jar的方案,如果引用到JFace,就没办法成功load jface classes。原因我之后会讲到。

 

2, http://sourceblogger.googlecode.com/svn/trunk/multiplatform-swt-loader/src/main/java/com/github/jendap/multiplatformswt/loader/MultiPlatformSwtHelper.java

这个方案我读过代码,但是没有试过,看起来很复杂,但是似乎功能也很健全,有兴趣的可以读一下。

 

3,http://stackoverflow.com/questions/2706222/create-cross-platform-java-swt-application

这个方案最简单明了,本文主要介绍该方案

 

其实3种方案实质是一样的,把所有平台的swt*.jar都打包进程序,然后根据OS和CPU构架信息,来动态load对应的swt*.jar

 

第三种Solution

我们所要介绍的第三种solution,它的办法是,在load class阶段,不load swt*.jar。而是延迟到main函数执行阶段,再根据OS和CPU构架来”手动地”load正确的swt*.jar

1) 首先添加以下方法

         private static void loadSwtJar() {

                   String swtFileName="";

              try {

                  String osName = System.getProperty("os.name").toLowerCase();

                  String osArch = System.getProperty("os.arch").toLowerCase();

                  final ClassLoader parentClassLoader = Thread.currentThread().getContextClassLoader();

                  Method addUrlMethod = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);

                  addUrlMethod.setAccessible(true);

 

                  String swtFileNameOsPart =

                    osName.contains("win") ? "win32.win32" :

                    osName.contains("mac") ? "cocoa.macosx" :

                    osName.contains("linux") || osName.contains("nix") ? "gtk.linux" :

                    ""; // throw new RuntimeException("Unknown OS name: "+osName)

 

                  String swtFileNameArchPart = osArch.contains("64") ? "x86_64" : "x86";

                  swtFileName = "org.eclipse.swt."+swtFileNameOsPart+"."+swtFileNameArchPart+"-4.4.jar";

                  URL swtFileUrl = new URL("rsrc:"+swtFileName); // I am using Jar-in-Jar class loader which understands this URL; adjust accordingly if you don't

                  addUrlMethod.invoke(parentClassLoader, swtFileUrl);

              }

              catch(Exception e) {

                  System.out.println("Unable to add the swt jar to the class path: "+swtFileName);

                  e.printStackTrace();

              }

          }

 

其作用是来根据OS和CPU构架信息,“手动地”加载正确的swt*.jar文件。

(以上代码在调试环境下可能没办法正确运行,需要在开始处添加一句:

URL.setURLStreamHandlerFactory(new RsrcURLStreamHandlerFactory(parentClassLoader));

但是在用ant编译时,则必须把这句话去掉。因为ant编译出的代码,RsrcURLStreamHandlerFactory已经被设置到URL,重复设置将出异常。其具体原因我就不深究了)

 

2) 添加jar-in-jar-loader.jar引用。

在eclipse的plugins目录下找到org.eclipse.jdt.ui_*version_number*.jar,解压它,发现jar-in-jar-loader.zip, 重命名为jar-in-jar-loader.jar。放到项目的lib目录下并引用。

需要添加这个jar的原因是,loadSwtJar方法隐式地使用了位于其中的JarRsrcLoader.class和相关的类。

 

3)在main函数最开始处添加 loadSwtJar()调用。

 

4)修改build.xml文件。

是的,你需要一个build.xml文件,如果没有,用eclipse的导出jar的功能生成一个。 修改build.xml文件的主要目的有2个:1是把SWT*.jar从默认加载列表中去掉,2是把所有平台的的SWT*.jar都放到打包列表中去。一加一减。给个例子:

<?xml version="1.0" encoding="UTF-8" standalone="no"?>

<project default="create_run_jar" name="Create Runnable Jar for Project swtguiexample with Jar-in-Jar Loader">

          <!--this file was created by Eclipse Runnable JAR Export Wizard-->

          <!--ANT 1.7 is required

                                          -->

          <target name="create_run_jar">

          <jar destfile="/home/binhua/Desktop/ bin /swtguiexample.jar">

          <manifest>

                   <attribute name="Main-Class" value="org.eclipse.jdt.internal.jarinjarloader.JarRsrcLoader"/>

                     <attribute name="Rsrc-Main-Class" value=" swtguiexample.Main"/>

                   <attribute name="Class-Path" value="."/>

                   <attribute name="Rsrc-Class-Path" value="./ jar-in-jar-loader.jar org.eclipse.equinox.common-3.6.100.v20120522-1841.jar org.eclipse.core.commands-3.6.1.v20120521-2329.jar org.eclipse.osgi-3.8.0.v20120529-1548.jar"/>

          </manifest>

             <zipfileset src="lib/jar-in-jar-loader.jar"/>

            <fileset dir="/home/binhua/Desktop/code/swtexample/src/swtguiexample/target/classes"/>

          <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/swt/org.eclipse.swt.gtk.linux.x86_64/4.4" includes="org.eclipse.swt.gtk.linux.x86_64-4.4.jar"/>

             <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/swt/org.eclipse.swt.gtk.linux.x86/4.4" includes="org.eclipse.swt.gtk.linux.x86-4.4.jar"/>

             <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/swt/org.eclipse.swt.cocoa.macosx.x86_64/4.4" includes="org.eclipse.swt.cocoa.macosx.x86_64-4.4.jar"/>

             <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/swt/org.eclipse.swt.win32.win32.x86_64/4.4" includes="org.eclipse.swt.win32.win32.x86_64-4.4.jar"/>

             <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/swt/org.eclipse.swt.win32.win32.x86/4.4" includes="org.eclipse.swt.win32.win32.x86-4.4.jar"/>

          <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/equinox/org.eclipse.equinox.common/3.6.100.v20120522-1841" includes="org.eclipse.equinox.common-3.6.100.v20120522-1841.jar"/>

          <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/core/org.eclipse.core.commands/3.6.1.v20120521-2329" includes="org.eclipse.core.commands-3.6.1.v20120521-2329.jar"/>

          <zipfileset dir="/home/binhua/.m2/repository/org/eclipse/osgi/org.eclipse.osgi/3.8.0.v20120529-1548" includes="org.eclipse.osgi-3.8.0.v20120529-1548.jar"/>

          </jar>

          </target>

</project>

以上: 1,在Rsrc<attribute name="Rsrc-Class-Path" 中,删除掉SWT*.jar,2,添加所有的平台的SWT*.jar到zipfileset节点。

 

5) 好了,用ant编译吧。

 

Trouble Shooting

以下trouble shooter事实上才是成败的关键:

#1

如果代码中引用了JFace*.jar,那么以上Solution会报ClassNotFoundException,说JFace下的某个类找不到,这是因为JFace加载失败了,为什么呢。

因为JFace*.jar还是默认加载的,JFace依赖于SWT*.jar,SWT*.jar已经延迟加载了,自然,JFace*.jar不可能加载成功。

解决办法是让JFace也延迟加载:

1,在Build.xml的Rsrc<attribute name="Rsrc-Class-Path" 中,把JFace*.jar也去掉

2,在loadSwtJar()最后一行添加

private static void loadSwtJar() {

  …

  addUrlMethod.invoke(parentClassLoader, new URL("rsrc:org.eclipse.jface-3.8.0.v20120521-2329.jar"));

}

 

2#

如果你有以下代码

public class MyApplicationWindow extends ApplicationWindow implements IExceptionHandler{

public static void main( String[] args )

          {

                   loadSwtJar();

                    …

          }

}

也会报ClassNotFoundException,为什么呢?

因为ApplicationWindow 和IExceptionHandler都是JFace下的类,main函数放在MyApplicationWindow中,要执行main函数,首先要加载ApplicationWindow 和IExceptionHandler,而这个时候,JFace还没被加载呢,记得吗,它被延迟加载了。

解决办法很简单,把main函数挪挪地方就好了。

 

3#

如果在Mac OS X下执行出错,因为必须加一个参数:

java -XstartOnFirstThread -jar *.jar

恩,不要问我为什么。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
作者:talent_marquis邮件:talent_marquis@163.com################19/12/2007更新###################新添加/重写的功能:1. 已实现简单的剪切/复制/粘贴功能(尚不能和外界程序交互)2. 添加创建新文件夹功能3. 添加选择全部/文件/文件夹功能4. 增加对创建新文件夹,重命名,删除,粘贴操作的异常处理5. FileUtil中增加getFileCentent方法,用于读取文本文件内容6. 重写AboutDialog7. 重写ExitAction8. 重写FileUtil中doPaste()方法,显著提高速度,修正拷贝文件时会导致文件不正常的严重Bug9. 修正其他部分已发现Bug10. 版本号升级为0.5尚未实现的功能:1. 与外界程序的交互2. 拖拽3. 国际化4. 输入文件名时对特殊字符的验证################17/12/2007更新###################新添加/重写的功能:1. 排序功能已加入菜单栏和弹出菜单2. 重写图标载入方法,已经可以实现图标动态载入和识别3. 重写TableViewer中tableColumn生成载入方法,已经可以实现tablecolumn的半动态载入和识别4. 实现动态识别文件类型5. 进行删除操作时,对删除的文件进行识别6. 更新添加了一些FileUtil中的方法,并且把这个类的注释写的差不多了 - -注意:1. icons文件夹内的文件结构有变化。2. jar文件没有更新,如果需要打好包的jar文件请等我把功能都实现后的那个版本 - -尚未实现的功能:1. 剪切/复制/粘贴功能2. 拖拽功能3. 详尽的注释4. 异常处理5. 国际化################14/12/2007更新###################已经实现的基本功能:1. 树形浏览结构2. 双击 运行程序/进入下一级菜单目录3. 返回上一级菜单4. 重命名5. 刷新6. 删除7. 按文件名/大小/类型/最后修改日期排序8. 对当前文件夹进行过滤操作9. 对当前文件夹的简要统计尚未实现的功能:1. 剪切/复制/粘贴2. 拖拽3. 弹出菜单中的排序选项4. 动态识别文件类型5. 详尽的注释已知的Bug:1. TreeViewer进行目录切换时,TableViewer窗体有时会变成空白2. 有特殊字符的过滤操作将导致异常3. 重命名没有考虑特殊字符4. 有大量文件(几百个以上)的目录会打开的很慢5. 文件删除失败时没有提示######新添加/重写的功能:1. 已实现简单的剪切/复制/粘贴功能(尚不能和外界程序交互)2. 添加创建新文件夹功能3. 添加选择全部/文件/文件夹功能4. 重写AboutDialog5. 增加对创建新文件夹,重命名,删除,粘贴操作的异常处理6. 重写ExitAction7. 修正部分已发现Bug尚未实现的功能:1. 与外界程序的交互2. 拖拽3. 国际化4. 输入文件名时对特殊字符的验证

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值