将java库转换为.net库


【转载请注明出处】

 动机:
    充分利用
java阵营众多的类库

 工具:

IKVM――把java bytecode 转换成IL程序,并提供大部分J2SE 1.4类的.net实现(IKVM.GNU.Classpath.dll

winrar――提取jar,打包jar

Java IDE(可选)――阅读源代码,浏览类之间的关系,我用的是eclipse

反编译工具(可选)――没源代码时用,主要也是浏览类与类之间的关系,java反编译我用的是DJ Java Decompiler.netReflector

 原理:

java也好,.net也好,其执行无非就是运行时 库,把java汇编指令一条一条转换为IL汇编指令,就可以实现java程序到.net程序的转换――你可以通过最纯正的.net使用方式来使用这些类库/程序。ikvm实现的就是这样一个功能。

如果一个java程序调用一大堆java类库,那么必须把这些程序和类库全部转换成.netIL格式,这个程序才能执行。

 

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格式,把后缀改成jarOK

 

例子:转换Geotoolsmain模块

下面以Geotoolsmain模块为例,说明怎样将jar文件转换为IL文件。

geotools是有10年历史的GIS中间件。其main模块有1000多个类,源代码文件大小合计9.74Mjar文件名为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: output file is "vecmath-1.3.dll"

Note: automatically adding reference to "f:\ikvm-0.20.0.0\ikvm\bin\ikvm.gnu.classpath.dll"

 成功转换成vecmath-1.3.dll

 如果碰上了依赖的库没转换,比如运行:

ikvmc -target:library jai_codec.jar

 结果提示有些类找不到:

 Note: output file is "jai_codec.dll"

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

 运行:

     ikvmc -target:library rt.jar

     惨!!!ikvm弹出错误窗口――不是我无能,是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把这个类重写就行了。幸运的是这里没碰见这情况。

 用把little_rt.jar转换为little_rt.dll,然后运行:

ikvmc -target:library -reference:little_rt.dll jai_codec.jar

OK!成功!

这样继续一个jar一个jar的转换,这些jar会依赖rt.jar中更多的类,需要都提取出来,放在little_rt.jar中,再转换为little_rt.dll

就这样把geoapi.jarjai_codec.jar, jai_core.jar, JTS-1.4.jar, vecmath-1.3.jarxalan-2.5.1.jaropengis-legacy-0.1.jar, units-0.01.jar都转换成了dllmailapi-1.3.jarbatik*.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.CoordinateTransformationorg.geotools.renderer.lite.InternalTranscoderorg.geotools.data.DataTestCase这三个类没转换成功,其中org.geotools.data.DataTestCase是测试用的,可以不管,剩下两个类要用到batik中的类,可以按照上述思路把batik转换为dll,也可重写这两个类――剩下的1000个类,已经全部转换成功,可以在.net平台上用了。

检验结果:用reflector打开gt2-main.dll

  




说明:

1rt.jarsun.*;com.*命名空间中的类,IKVM.GNU.Classpath.dll都没有

2,进行.net开发时需要引用IKVM.GNU.Classpath.dll

3ikvmjar->dll转换可能存在bug,最好把test case也全部转换过来,测试测试

4ikvmjar->dll转换可能存在性能问题,需要时可进行重构

5,有少量java 语法产生的指令不能直接转换,会报错。这是极少量的,我转换了好几M的东西了,只报了2处这种错。这时候可能需要改动改动java源代码

posted @ 2005-10-10 15:57  xiaotie  阅读(5734)  评论(14编辑  收藏  举报