在Java中调用Python代码
极少数时候,我们会碰到类似这样的问题:与A同学合作写代码, A同学只会写Python,不熟悉Java ,而你只会写Java不擅长Python,并且发现难以用Java来重写对方的代码,这时,就不得不想方设法“调用对方的代码”。
下面,我通过一些简单的小例子来说明:如何在Java中调用Python代码。
主要内容如下:
什么是Jython?
Jython(原JPython),可以理解为一个由Java语言编写的Python解释器。
要使用Jython,只需要将Jython-x.x.x.jar文件置于classpath中即可 --> 官网下载
当然,通过Maven导入也OK,如下:
<dependency>
<groupId>org.python</groupId>
<artifactId>jython-standalone</artifactId>
<version>2.7.0</version>
</dependency>
一个HelloPython程序
import org.python.util.PythonInterpreter;
public class HelloPython {
public static void main(String[] args) {
PythonInterpreter interpreter = new PythonInterpreter();
interpreter.exec("print('hello')");
}
}
什么是PythonInterpreter呢?它的中文意思即“Python解释器”。我们知道Python程序都是由解释器执行的,上面的代码在JVM中创建一个“Python解释器”对象,用于模拟Python解释器的行为,并通过exec("Python语句") 直接在JVM中执行Python代码,代码的输出结果为:hello。该程序运行速度相较正常的Java或者Python程序都要慢那么一点。
在JVM中执行Python脚本
interpreter.execfile("D:/labs/mytest/hello.py");
如上,将exec改为execfile就可以了。需要注意的是,这个 .py文件不能含有第三方模块,因为这个“Python脚本”说到底仍是在JVM环境下执行的(而非依赖于本地计算机环境),如果 .py 程序中包含有第三方模块(例如 NumPy)将会在编译期报错:java ImportError: No module named xxx
在JVM中调用Python编写的函数
先写一个hello.py的Python代码:
def hello():
return 'Hello'
在Java代码中调用这个Python函数:
import org.python.core.PyFunction;
import org.python.core.PyObject;
import org.python.util.PythonInterpreter; public class HelloPython {
public static void main(String[] args) {
PythonInterpreter interpreter = new PythonInterpreter();
interpreter.execfile("D:/labs/hello.py"); PyFunction pyFunction = interpreter.get("hello", PyFunction.class); // 第一个参数为期望获得的函数(变量)的名字,第二个参数为期望返回的对象类型
PyObject pyObject = pyFunction.__call__(); // 调用函数 System.out.println(pyObject);
}
}
上面的代码执行结果为:Hello
可以发现:即便只是调用一个函数,也必须先加载整个 .py文件,之后才能通过Jython包中所定义的类获取、调用这个函数。
如果Python函数需要参数,必须先将Java代码中的参数转化为对应的“Python类型”(姑且可以称作 Jython 类型 (●'◡'●),例如:
__call__(new PyInteger(a), new PyInteger(b))
a,b的类型均为Java中的int型,还有一些Jython类型诸如:PyString、PyList等,详细信息可以参考官方的API文档。
在本地环境中调用Python脚本
由于Jython运行过慢并且不支持第三方的Python模块,也许通过Java执行一段终端命令来调用Python脚本才是实际会用的方式。以下为和舍友合作写的“手写识别”程序中的一个代码片段(Python做识别,Java写界面):
import java.io.*;
class PyCaller {
private static final String DATA_SWAP = "temp.txt";
private static final String PY_URL = System.getProperty("user.dir") + "\\test.py";
public static void writeImagePath(String path) {
PrintWriter pw = null;
try {
pw = new PrintWriter(new FileWriter(new File(DATA_SWAP)));
} catch (IOException e) {
e.printStackTrace();
}
pw.print(path);
pw.close();
}
public static String readAnswer() {
BufferedReader br;
String answer = null;
try {
br = new BufferedReader(new FileReader(new File(DATA_SWAP)));
answer = br.readLine();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return answer;
}
public static void execPy() {
Process proc = null;
try {
proc = Runtime.getRuntime().exec("python " + PY_URL);
proc.waitFor();
} catch (IOException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 测试码
public static void main(String[] args) throws IOException, InterruptedException {
writeImagePath("D:\\labs\\mytest\\test.jpg");
execPy();
System.out.println(readAnswer());
}
}
运行流程:Java Swing 界面接收用户输入 --> Java 将用户输入写到本地文件中 --> Java 调用本地 Python 脚本 --> Python 从本地文件拿到用户输入 --> Python 处理用户输入得到最终结果 --> Python 把最终结果写到本地文件 --> Java 对 Python 脚本的调用结束 --> Java 从本地文件中取出最终结果 --> Java 把最终结果返回给用户
完整代码链接:http://pan.baidu.com/s/1sl4l68H
后续补。这块内容已经写了好多年了。但是发现居然有出乎意料的浏览量。感觉有很多不妥之处。恐怕会对新手产生误导(万一 。。。),所以特意补上一段。上面的内容是基于大学初期的一次课设作业,当时好像是必须强行进行合作写代码,完成一个什么任务。。现在好多年过去了,我至今没有发现什么场景是真的需要Jython。而用Java执行终端命令来调Python现在看来也是非常不妥的。总之,实际开发中1000%有更漂亮的方法。上述方法可能仅适合写一些玩具式程序,甚至写玩具式程序都感觉未免“脏”了一些。。。所以, 仅 供 参 考。
在Java中调用Python代码的更多相关文章
- 如何在Java中调用Python代码
有时候,我们会碰到这样的问题:与A同学合作写代码,A同学只会写Python,而不会Java, 而你只会写Java并不擅长Python,并且发现难以用Java来重写对方的代码,这时,就不得不想方设法“调 ...
- 在Java中调用Python
写在前面 在微服务架构大行其道的今天,对于将程序进行嵌套调用的做法其实并不可取,甚至显得有些愚蠢.当然,之所以要面对这个问题,或许是因为一些历史原因,或者仅仅是为了简单.恰好我在项目中就遇到了这个问题 ...
- 使用Runtime.getRuntime().exec()在java中调用python脚本
举例有一个Python脚本叫test.py,现在想要在Java里调用这个脚本.假定这个test.py里面使用了拓展的包,使得pythoninterpreter之类内嵌的编译器无法使用,那么只能采用ja ...
- android开发中调用python代码(带参数)
android开发主要用到的是java代码,但是当开发涉及到一些算法时,往往用python可以提高软件的运行速度,也更加便捷,这里分享自己项目调用python代码的方式,主要有以下几个步骤(个人方法, ...
- 在java中调用python方法
1.http://sourceforge.net/projects/jython/下载jython包,把其中的jython.jar添加到工程目录 示例: 1.摘自:http://blog.csdn.n ...
- python爬虫简单实现,并在java中调用python脚本,将数据保存在json文件中
# coding:utf-8 import urllib2 from bs4 import BeautifulSoup import json import sys reload(sys) sys.s ...
- 『Python CoolBook』C扩展库_其六_从C语言中调用Python代码
点击进入项目 一.C语言运行pyfun的PyObject对象 思路是在C语言中提供实参,传给python函数: 获取py函数对象(PyObject),函数参数(C类型) 获取GIL(PyGILStat ...
- android111 java中调用c代码
MainActivity: package com.itheima.helloworld1; import android.os.Bundle; import android.app.Activity ...
- java调用python代码
同样的我们需要安装jython,具体的步骤如下: 1. 去 http://sourceforge.net/projects/jython/ 下载最新的jython相关的jar包. 2. 下载下来的ja ...
随机推荐
- Android Studio--NDK编译C代码为.so文件,JNI调用
前言: 从Android Studio开始,就支持jni和.so库调用了. 环境: Windows 7+Android Studio2.1.2+NDK版本:android-ndk-r10e 准备工作: ...
- Android仿QQ复制昵称效果2
本文同步自http://javaexception.com/archives/77 背景: 在上一篇文章中,给出了一种复制QQ效果的方案,今天就来讲讲换一种方式实现.主要依赖的是一个开源项目https ...
- 转载:tar 解压缩命令~
转载自:http://blog.csdn.net/dunyanan1/article/details/38869059tar -c: 建立压缩档案-x:解压-t:查看内容-r:向压缩归档文件末尾追加文 ...
- Oracle raw数据类型
RAW的声明方式为RAW(L),L为长度,以字节为单位,它存数的是16进制的数据.作为数据库列最大2000,作为变量最大32767字节. RAW类型的好处就是:在网络中的计算机之间传输 RAW 数据时 ...
- [Spring MVC]学习笔记--@RequestMapping支持的返回类型
下面针对官方文档列出的支持类型进行举例. (本篇例子存于github上, https://github.com/lemonbar/spring-mvc-requestmapping) 可以直接下载, ...
- DropdownList异步刷新GridView数据
前台代码: <div style=" clear:both; width:800px; text-align:center; margin-left:auto; margin-righ ...
- 第六课作业——主从复制和sentinel高可用
第六课时作业 静哥 by 2016.3.21~2016.4.3 [作业描述] 1.配置主从复制,截图看日志 2.配置一个哨兵,一主一从结构,并实现主宕机从接管的过程,截图显示 3.总结哨兵的原理 ...
- MVC4学习笔记之--身份认证过滤器
过滤器作为MVC模式中面向切面编程应用很广泛,例如身份验证,日志,异常,行为截取等.博客园里面的大神对应过滤器的介绍以及很多,MVC4中不同的过滤器也介绍得很清楚.FlyDragon 辉太 禁止吸烟 ...
- js对多行字符串的处理
f = [] g = str(f) h = ''.join(f) dic_ = () gd = str(dic_) hd = ''.join(dic_) 0 老板1 北京2 上海3 天津4 重庆5 河 ...
- JavaScript通过preventDefault()使input[type=text]禁止输入但保留光标
一.说明 取消事件的默认动作. 该方法将通知 Web 浏览器不要执行与事件关联的默认动作(如果存在这样的动作).例如,如果 type 属性是 "submit",在事件传播的任意阶段 ...