充分利用java阵营众多的类库
IKVM――把java bytecode 转换成IL程序,并提供大部分J2SE 1.4类的.net实现(IKVM.GNU.Classpath.dll)
winrar――提取jar,打包jar
Java IDE(可选)――阅读源代码,浏览类之间的关系,我用的是eclipse
反编译工具(可选)――没源代码时用,主要也是浏览类与类之间的关系,java反编译我用的是DJ Java Decompiler,.net用Reflector。
java也好,.net也好,其执行无非就是运行时 + 库,把java汇编指令一条一条转换为IL汇编指令,就可以实现java程序到.net程序的转换――你可以通过最纯正的.net使用方式来使用这些类库/程序。ikvm实现的就是这样一个功能。
如果一个java程序调用一大堆java类库,那么必须把这些程序和类库全部转换成.net的IL格式,这个程序才能执行。
IKVM安装:
从www.ikvm.net上下载ikvm,解压缩后就能开始用。
将jar转换为dll:
运行:ikvmc -target:library ××××.jar
会将jar文件自动转换为 ××××.dll
但是,如果jar中引用的类不存在于××××.jar,也不存在于IKVM.GNU.Classpath.dll之中,则会给出警告信息,该类也不会转换。这时,需要引用缺少的类所在的dll,格式如下:
ikvmc -target:library -reference:lib1.dll -reference:lib2.dll -reference:lib3.dll jar1.jar
jar的提取:
jar可以用winrar打开,把你想要的目录或类解压缩。
把java class文件打包为jar:
把目录压缩成zip格式,把后缀改成jar,OK。
例子:转换Geotools的main模块
下面以Geotools的main模块为例,说明怎样将jar文件转换为IL文件。
geotools是有10年历史的GIS中间件。其main模块有1000多个类,源代码文件大小合计9.74M。jar文件名为gt2-main.jar,2.34M.
步骤1:分析main模块依赖的类库,分析类库间的依赖关系
geotools-main依赖的类库有:
rt.jar――java的主要的lib
geoapi.jar――OpenGIS一帮人定义的标准geoapi
jai_codec.jar, jai_core.jar,――Java 2D API
JTS-1.4.jar,――Java 拓扑套件,主要用于拓扑分析,可能依赖
vecmath-1.3.jar――vector数学库
xalan-2.5.1.jar――xml 转换lib
opengis-legacy-0.1.jar,――忘了干什么的,反正和gis有关
units-0.01.jar――不知道是什么
batik*.jar――SVG lib
mailapi-1.3.jar
步骤2:转换类库
先转换那些没依赖的库,比如vecmath-1.3.jar,......
ikvmc -target:library vecmath-1.3.jar
Note: automatically adding reference to "f:\ikvm-0.20.0.0\ikvm\bin\ikvm.gnu.classpath.dll"
ikvmc -target:library jai_codec.jar
Note: automatically adding reference to "f:\ikvm-0.20.0.0\ikvm\bin\ikvm.gnu.classpath.dll"
Warning: class "com.sun.image.codec.jpeg.JPEGCodec" not found
Warning: class "com.sun.image.codec.jpeg.JPEGEncodeParam" not found
Warning: class "com.sun.image.codec.jpeg.JPEGImageEncoder" not found
Warning: class "com.sun.image.codec.jpeg.JPEGDecodeParam" not found
Warning: class "com.sun.image.codec.jpeg.JPEGImageDecoder" not found
Warning: class "com.sun.image.codec.jpeg.ImageFormatException" not found
Warning: class "com.sun.image.codec.jpeg.JPEGQTable" not found
Warning: class "sun.security.action.GetPropertyAction" not found
因此对于有依赖关系的库,需要从最下面那个,大家都依靠它的那个库转换起。在这里就是rt.jar。
怎么办?需要从rt.jar中找出所需要的类,将这些类转换就行了。
用winrar找出需要的类,比如jai_codec需要com.sun.image.codec.jpeg.JPEGCodec,....等8个类,就从rt.jar中找出这些类出来。注意,这些类可能还依赖于别的类,依赖的类也需要找出来,全部打包成一个新的jar,如little_rt.jar。如果有些类在IKVM.GNU.Classpath.dll中已经有了,可以不用提取。
最惨的情况可能是1个类依赖2个类,2个类又依赖8个类,然后又依赖更多的类,这种情况下与其要把这些类提取出来,不如更改这个类,把对外依赖的东东都去掉,全部return null, return 0什么的。先转换成功,然后再用.net把这个类重写就行了。幸运的是这里没碰见这情况。
ikvmc -target:library -reference:little_rt.dll jai_codec.jar
OK!成功!
这样继续一个jar一个jar的转换,这些jar会依赖rt.jar中更多的类,需要都提取出来,放在little_rt.jar中,再转换为little_rt.dll。
就这样把geoapi.jar,jai_codec.jar, jai_core.jar, JTS-1.4.jar, vecmath-1.3.jar,xalan-2.5.1.jar,opengis-legacy-0.1.jar, units-0.01.jar都转换成了dll。mailapi-1.3.jar和batik*.jar没转换。这两个直接转换都没成功,batik*.jar是一系列jar,之间的关系比较复杂我也搞不清。凭我对geotools的了解, main模块里基本没用到mail,而SVG是以插件形式提供的,在这里用的也不多。那就不管这两个,来转换gt2-main.jar先。
运行:
ikvmc -target:library -reference:jai_codec.dll -reference:geoapi.dll -reference:xalan-2.5.1.dll -reference:JTS-1.4.dll -reference:vecmath-1.3.dll -reference:units-0.01.dll -reference:opengis-legacy-0.1.dll -reference:little_rt.dll -reference:jai_core.dll gt2-main.jar
结果:
Note: output file is "gt2-main.dll"
Note: automatically adding reference to "f:\ikvm-0.20.0.0\ikvm\bin\ikvm.gnu.classpath.dll"
Warning: class "org.geotools.ct.CoordinateTransformation" not found
Warning: unable to compile class "org.geotools.renderer.lite.InternalTranscoder"
(missing class "org.apache.batik.transcoder.image.ImageTranscoder")
Warning: class "javax.swing.text.DefaultFormatterFactory" not found
Warning: unable to compile class "org.geotools.ct.CoordinateTransformation$Inverse"
(missing class "org.geotools.ct.CoordinateTransformation")
Warning: class "org.geotools.renderer.style.InternalTranscoder" not found
Warning: class "org.apache.batik.transcoder.TranscoderInput" not found
Warning: class "org.apache.batik.transcoder.SVGAbstractTranscoder" not found
Warning: class "org.apache.batik.transcoder.TranscodingHints$Key" not found
Warning: class "org.apache.batik.transcoder.TranscoderOutput" not found
Warning: class "org.apache.batik.transcoder.TranscodingHints" not found
Warning: class "org.geotools.renderer.lite.InternalTranscoder" not found
Warning: unable to compile class "org.geotools.data.DataTestCase"
(missing class "junit.framework.TestCase")
Warning: class "javax.mail.Session" not found
Warning: class "javax.mail.internet.InternetAddress" not found
Warning: class "javax.mail.Address" not found
Warning: class "javax.mail.internet.MimeMessage" not found
Warning: class "javax.mail.Message" not found
Warning: class "javax.mail.Message$RecipientType" not found
Warning: class "javax.mail.Transport" not found
Warning: class "javax.mail.MessagingException" not found
Warning: class "javax.mail.internet.AddressException" not found
可以看见只有org.geotools.ct.CoordinateTransformation,org.geotools.renderer.lite.InternalTranscoder,org.geotools.data.DataTestCase这三个类没转换成功,其中org.geotools.data.DataTestCase是测试用的,可以不管,剩下两个类要用到batik中的类,可以按照上述思路把batik转换为dll,也可重写这两个类――剩下的1000个类,已经全部转换成功,可以在.net平台上用了。
检验结果:用reflector打开gt2-main.dll
说明:
1,rt.jar中sun.*;com.*命名空间中的类,IKVM.GNU.Classpath.dll都没有
2,进行.net开发时需要引用IKVM.GNU.Classpath.dll
3,ikvm的jar->dll转换可能存在bug,最好把test case也全部转换过来,测试测试
4,ikvm的jar->dll转换可能存在性能问题,需要时可进行重构
5,有少量java 语法产生的指令不能直接转换,会报错。这是极少量的,我转换了好几M的东西了,只报了2处这种错。这时候可能需要改动改动java源代码