因为SWT的实现机制,在不一样平台下,必须引用不一样swt*.jar. 因为这个瓶颈,咱们要为不一样的平台编译不一样的版本。可是这是能够避免的。这将是本文要讨论的内容。java
我一共google到了3种solution:linux
1,使用swtjar.jar。git
http://mchr3k.github.io/swtjar/github
其主页有详细的介绍。可是彷佛下载连接已经无效了,一个下载的办法是从github上找到引用了它的项目,好比https://github.com/mchr3k/org.intrace/tree/3a1debcbb831f802219b341fb5e37467b365d443/org.intrace/libmacos
swtjar.jar的原理,彷佛是经过替换掉默认的ClassLoader来实现的。app
根据个人测试,使用swtjar.jar的方案,若是引用到JFace,就没办法成功load jface classes。缘由我以后会讲到。eclipse
这个方案我读过代码,可是没有试过,看起来很复杂,可是彷佛功能也很健全,有兴趣的能够读一下。函数
3,http://stackoverflow.com/questions/2706222/create-cross-platform-java-swt-application测试
这个方案最简单明了,本文主要介绍该方案
其实3种方案实质是同样的,把全部平台的swt*.jar都打包进程序,而后根据OS和CPU构架信息,来动态load对应的swt*.jar
咱们所要介绍的第三种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 shooter事实上才是成败的关键:
若是代码中引用了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"));
}
若是你有如下代码
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函数挪挪地方就行了。
若是在Mac OS X下执行出错,由于必须加一个参数:
java -XstartOnFirstThread -jar *.jar
恩,不要问我为何。