因为在Jython 调用 Java 碰壁全纪录中已经有了“出色”碰壁经历,此处仿照其大致过程
命令行下无差异,Eclipse中略有差异(classpath的设置方式不同所致)
JavaClass 的定义
public class JavaClass {
private String str = "";
public JavaClass() {
this.str = "JavaClass Init";
}
public String getStr() {
return str;
}
public void setStr(String str) {
this.str = str;
}
}
一、命令行模式
1、进入 python 目录,查看当前目录下只有一个JavaClass.java文件
D:\>cd python
D:\python>dir
2012-03-13 09:59
2012-03-13 09:59
2012-03-13 09:59 227 JavaClass.java
1 个文件 227 字节
2 个目录 37,943,169,024 可用字节
2、启动Python命令行模式(Python版本是 2.7)
D:\python>python
3、导入 JavaClass时,提示找不到JavaClass模块
>>> from jpype import *
>>> jvmPath = getDefaultJVMPath()
>>> startJVM(jvmpath)
>>> startJVM(jvmPath)
>>> JavaClass = JClass("JavaClass")
Traceback (most recent call last):
File "
File "C:\Python27\lib\site-packages\jpype\_jclass.py", line 54, in JClass
raise _RUNTIMEEXCEPTION.PYEXC("Class %s not found" % name)
jpype._jexception.ExceptionPyRaisable: java.lang.Exception: Class JavaClass not found
原因:Python中导入的是.class文件,而JavaClass.java尚未编译生成 .class文件
4、Ctrl-Z退出 Python, 编译 JavaClass.java
D:\python>javac JavaClass.java
D:\python>dir
2012-03-13 10:12
2012-03-13 10:12
2012-03-13 10:12 443 JavaClass.class
2012-03-13 09:59 227 JavaClass.java
2 个文件 670 字节
2 个目录 37,943,169,024 可用字节
5、按第2步,再次进入Python模式后导入 JavaClass时未报异常,说明成功导入,简单调用情况如下
D:\python>python
>>> from jpype import *
>>> jvmPath = getDefaultJVMPath()
>>> startJVM(jvmpath)
>>> startJVM(jvmPath)
>>> JavaClass = JClass("JavaClass")
>>> jc = JavaClass()
>>> jc.str // 调用get方法
'JavaClass Init'
>>> jc.str = "JavaClass Set" // 调用set方法
>>> jc.str
'JavaClass Set'
二、eclipse IDE 环境
workplace 是d:\python,项目名称是mython,其结构如下
mython
......
bin
javaDemo
JavaClass.class
pythonDemo
Py2Ja.py
src
javaDemo
JavaClass.java
pythonDemo
Py2Ja.py
1、JavaClass.java同上(但是第一行多了一个package javaDemo的定义)
2、编辑Py2Ja.py如下:
from jpype import *
jvmPath = getDefaultJVMPath()
startJVM(jvmPath)
JavaClass = JClass("javaDemo.JavaClass")
jc = JavaClass()
print jc.str
jc.str = "JavaClass IDE"
print jc.str
shutdownJVM()
测试运行时报异常
Traceback (most recent call last):
File "D:\python\mython\src\pythonDemo\Py2Ja.py", line 4, in
JavaClass = JClass("javaDemo.JavaClass")
File "C:\Python27\lib\site-packages\jpype\_jclass.py", line 54, in JClass
raise _RUNTIMEEXCEPTION.PYEXC("Class %s not found" % name)
jpype._jexception.ExceptionPyRaisable: java.lang.Exception: Class javaDemo.JavaClass not found
原因:找不到javaDemo是因为“当前路径”下找不到JavaDemo文件或者目录
解决方法:手工添加JVM的启动参数-Djava.class.path ='D:\python\Mython\src
3、再次修改后如下:
from jpype import *
jvmPath = getDefaultJVMPath()
classPath = "f:\workplace\Python\src"
jvmArg = "-Djava.class.path="+classPath
startJVM(jvmPath, jvmArg)
JavaClass = JClass("javaDemo.JavaClass")
jc = JavaClass()
print jc.str
jc.str = "JavaClass IDE"
print jc.str
shutdownJVM()
测试运行时仍然报异常
Traceback (most recent call last):
File "F:\workplace\Python\src\pythonDemo\Py2Ja.py", line 6, in
JavaClass = JClass("javaDemo.JavaClass")
File "C:\Python27\lib\site-packages\jpype\_jclass.py", line 54, in JClass
raise _RUNTIMEEXCEPTION.PYEXC("Class %s not found" % name)
jpype._jexception.ExceptionPyRaisable: java.lang.Exception: Class javaDemo.JavaClass not found
与2中的异常一样:找不到JavaClass。
原因:我们已经设置了-Djava.class.path ='D:\python\Mython\src,仍然找不到JavaClass,只能说明是找不到JavaClass.class文件
在命令行中第3步中提到Python文件导入的应该是.class文件,在看前面mython项目的整个目录发现,class文件在mython\bin\javaDemo中。也就是说可能是因为编译器找到了JavaClass的定义即 JavaClass.java,但是找不到它的.class文件。
那java文件对应的class文件为什么不是在src包下的JavaDemo中,而是在bin包下的JavaDemo中?
右击项目mython->Properties->Java Build Path->source最下面有default output folder中默认的是mython\bin,且这个目录在 Package Explorer视图下是看不到的,在Navigator视图中可以看到。
4、将default output folder 修改为 mython\src,目录结构变化如下
mython
......
src
javaDemo
JavaClass.class
JavaClass.java
pythonDemo
Py2Ja.py
测试运行成功:
JavaClass Init
JavaClass IDE
JVM activity report :
classes loaded : 20
JVM has been shutdown
总结:对于新手不太习惯命令行式的Java程序调试,IDE虽然简单方便,但是都有自己默认的规则。这些规则新手可能注意不到,所以在IDE环境中照搬程序源码的时候,总会报一些奇奇怪怪的错误,使得本来简单的问题看上去很复杂。究其原因:1、对开发环境不熟悉、2、java基础不扎实。
特殊情况备注:
1、***.py文件运行时,默认路径即该文件的所在路径,导入***.class时的路径就变成了 ***.py所在的包 + from中的包,即默认.class文件是在 ***.py所在的包 的 子包中,所以在调用Java文件时,需要特别添加class文件所在路径。即使Java文件和Python文件在同一个包中,Java文件仍然在某个包中,测试运行时仍然要设置java文件所在的包最上级所在的目录,一般为 src。
2、将Java文件和Python文件都放在 src 下时,虽然目录结构中显示一个叫"(default package)",但是在Java文件的内部是没有包的设定的,这种情况下运行Python文件时,可不手工设置class文件所在路径,因为此时 Python文件的所在目录为src,而Java文件也在src中,且没有包名引起的问题。
刚刚接触 Python,因为基础不太扎实,期间诸多碰壁,仅此记录。
成长的道路上不怕跌倒,怕的是的跌倒了再也爬不起来。为自己加油!