java调python

java调python,第1张

很多朋友都想知道java怎么调python?下面就一起来了解一下吧~

java调python主要有两种方法:1.使用Runtime.getRuntime()执行脚本文件;2. 将python脚本写成进程为java提供服务,下面是具体的方法介绍:

第一种:使用Runtime.getRuntime()执行脚本文件

先建立python脚本文件 demo.py

import numpy as npa = np.arange(12).reshape(3,4)print(a)

java调用python程序并输出该结果

import java.io.BufferedReaderimport java.io.IOExceptionimport java.io.InputStreamReaderpublic class Demo {    public static void main(String[] args) {         // TODO Auto-generated method stub         Process proc         try {             proc = Runtime.getRuntime().exec("python D:\\demo.py")// 执行py文件             //用输入输出流来截取结果             BufferedReader in = new BufferedReader(new InputStreamReader(proc.getInputStream()))             String line = null             while ((line = in.readLine()) != null) {      锋清           System.out.println(line)             }             in.close()             proc.waitFor()         } catch (IOException e) {             e.printStackTrace()         } catch (InterruptedException e) {             e.printStackTrace()         }      }}

如若向python程序中函数传递参数并执行出结果,下面就举一例来说明一下。

同样建立python脚本文件demo2.py

import sysdef func(a,b):     return (a+b)if __name__ == '__main__':     a = []     for i in range(1, len(sys.argv)):         a.append((int(sys.argv[i])))    print(func(a[0],a[1]))

其中sys.argv用于获取参数url1,url2等。而sys.argv[0]代表python程序名,所以列表从1开始读取参数。

以上代码实现一个两个链基桐数做加法的程序,下面看看在java中怎么传递函数参数,代码如下:

int a = 18int b = 23try {     String[] args = new String[] { "python", "D:\\demo2.py", String.valueOf(a), String.valueOf(b) }     Process proc = Runtime.getRuntime().exec(args)// 执行py文件棚坦    BufferedReader in = new BufferedReader(new InputStreamReader(proc.getInputStream()))     String line = null     while ((line = in.readLine()) != null) {         System.out.println(line)     }     in.close()     proc.waitFor()} catch (IOException e) {     e.printStackTrace()} catch (InterruptedException e) {     e.printStackTrace()}

其中args是String[] { “python”,path,url1,url2 },path是python程序所在的路径,url1是参数1,url2是参数2,以此类推。

2. 将python脚本写成进程为java提供服务

python脚本文件如下:

import socketimport sysimport threadingimport numpy as npfrom PIL import Imagedef main():     # 创建服务器套接字     serversocket = socket.socket(socket.AF_INET,socket.SOCK_STREAM)     # 获取本地主机名称     host = socket.gethostname()     # 设置一个端口     port = 12345     # 将套接字与本地主机和端口绑定     serversocket.bind((host,port))     # 设置监听最大连接数     serversocket.listen(5)     # 获取本地服务器的连接信息     myaddr = serversocket.getsockname()     print("服务器地址:%s"%str(myaddr))     # 循环等待接受客户端信息     while True:         # 获取一个客户端连接         clientsocket,addr = serversocket.accept()         print("连接地址:%s" % str(addr))         try:             t = ServerThreading(clientsocket)#为每一个请求开启一个处理线程             t.start()             pass         except Exception as identifier:             print(identifier)             pass         pass     serversocket.close()     passclass ServerThreading(threading.Thread):     # words = text2vec.load_lexicon()     def __init__(self,clientsocket,recvsize=1024*1024,encoding="utf-8"):         threading.Thread.__init__(self)         self._socket = clientsocket         self._recvsize = recvsize         self._encoding = encoding        pass    def run(self):         print("开启线程.....")         try:             #接受数据             msg = ''             while True:                 # 读取recvsize个字节                 rec = self._socket.recv(self._recvsize)                 # 解码                 msg += rec.decode(self._encoding)                 # 文本接受是否完毕,因为python socket不能自己判断接收数据是否完毕,                 # 所以需要自定义协议标志数据接受完毕                 if msg.strip().endswith('over'):                     msg=msg[:-4]                     break                         sendmsg = Image.open(msg)             # 发送数据             self._socket.send(("%s"%sendmsg).encode(self._encoding))             pass         except Exception as identifier:             self._socket.send("500".encode(self._encoding))             print(identifier)             pass         finally:             self._socket.close()          print("任务结束.....")                  pass    def __del__(self):         passif __name__ == "__main__":     main()

在java代码中访问python进程的代码: package helloimport java.lang.Systemimport java.io.BufferedReaderimport java.io.IOExceptionimport java.io.InputStreamReaderimport java.net.InetAddressimport java.net.Socketimport java.io.OutputStreamimport java.io.PrintStreamimport java.io.InputStreampublic class hello {     public static void main(String[] args){         //System.out.println("Hello World!")         // TODO Auto-generated method stub        try {             InetAddress addr = InetAddress.getLocalHost()             String host=addr.getHostName()             //String ip=addr.getHostAddress().toString() //获取本机ip             //log.info("调用远程接口:host=>"+ip+",port=>"+12345)             // 初始化套接字,设置访问服务的主机和进程端口号,HOST是访问python进程的主机名称,可以是IP地址或者域名,PORT是python进程绑定的端口号             Socket socket = new Socket(host,12345)             // 获取输出流对象             OutputStream os = socket.getOutputStream()             PrintStream out = new PrintStream(os)             // 发送内容             out.print( "F:\\xxx\\0000.jpg")             // 告诉服务进程,内容发送完毕,可以开始处理             out.print("over")             // 获取服务进程的输入流             InputStream is = socket.getInputStream()             BufferedReader br = new BufferedReader(new InputStreamReader(is,"utf-8"))             String tmp = null             StringBuilder sb = new StringBuilder()             // 读取内容             while((tmp=br.readLine())!=null)                 sb.append(tmp).append('\n')             System.out.print(sb)             // 解析结果             //JSONArray res = JSON.parseArray(sb.toString())         } catch (IOException e) {             e.printStackTrace()         }finally {             try {if(socket!=null) socket.close()} catch (IOException e) {}             System.out.print("远程接口调用结束.")         }       }}

Jython(原JPython),是一个用Java语言写的Python解释器。在没有第三方模块的情况下,通常选择利用Jython来调用Python代码,它是一个开源的JAR包,你可以到官网下载一个HelloPython程序importorg.python.util.PythonInterpreterpublicclassHelloPython{publicstaticvoidmain(String[]args){PythonInterpreterinterpreter=newPythonInterpreter()interpreter.exec("print('hello')")}}什么是PythonInterpreter?它的中文意思即是“Python解释器”。我们知道Python程序都是通过解释器来执行的,我们在Java中创建一个“解释器”对氏隐象,模拟Python解释器的行为,通过exec("Python语句")直接在JVM中执行Python代码,上面代码的输出结果为:hello在Jvm中执行Python脚本interpreter.execfile("D:/labs/mytest/hello.py")如上,将exec改为execfile就可以了。需要注意的是,这个.py文件不能含有第三方模块,因为这个“Python脚本”最终还是在JVM环境下执行的,如果有第三方模块将会报错:javaImportError:Nomodulenamedxxx仅在Java中调用Python编写的函数先完成一个hello.py代码:defhello():return'Hello'在Java代码中调用这个函数:importorg.python.core.PyFunctionimportorg.python.core.PyObjectimportorg.python.util.PythonInterpreterpublicclassHelloPython{publicstaticvoidmain(String[]args){PythonInterpreterinterpreter=newPythonInterpreter()interpreter.execfile("D:/labs/hello.py")PyFunctionpyFunction=interpreter.get("hello",PyFunction.class)//第一个参数为期望获得的函数(变量)的名字,第二个参数为期望返回的对象类型PyObjectpyObject=pyFunction.__call__()//调用函数System.out.println(pyObject)}}上面的代码执行结果为:Hello即便只是调用一个函数,也必须先加载这个.py文件,之后再通过Jython包中所定义的类获取、调用这个函数。如果函数需要参数,在Java中必须先将参数转化为对应的“Python类型”,例如:__call__(newPyInteger(a),newPyInteger(b))a,b的类型为Java中的int型,还有诸如:PyString(Stringstring)、PyList(Iteratoriter)等。详细可以参考官方的api文档。包含第三方模块的情况:一个手写识别程序这是我和舍友合作写的一个小程序,完整代码在这里:,界面上引用了corejava上的一段代码。Python代码是舍友写的,因为在Python程序中使用了第三方亩缺的NumPy模迅核辩块,导致无法通过Jython执行。下面这个方法纯粹是个人思路,没有深入查资料。核心代码如下:importjava.io.*classPyCaller{privatestaticfinalStringDATA_SWAP="temp.txt"privatestaticfinalStringPY_URL=System.getProperty("user.dir")+"\\test.py"publicstaticvoidwriteImagePath(Stringpath){PrintWriterpw=nulltry{pw=newPrintWriter(newFileWriter(newFile(DATA_SWAP)))}catch(IOExceptione){e.printStackTrace()}pw.print(path)pw.close()}publicstaticStringreadAnswer(){BufferedReaderbrStringanswer=nulltry{br=newBufferedReader(newFileReader(newFile(DATA_SWAP)))answer=br.readLine()}catch(FileNotFoundExceptione){e.printStackTrace()}catch(IOExceptione){e.printStackTrace()}returnanswer}publicstaticvoidexecPy(){Processproc=nulltry{proc=Runtime.getRuntime().exec("python"+PY_URL)proc.waitFor()}catch(IOExceptione){e.printStackTrace()}catch(InterruptedExceptione){e.printStackTrace()}}//测试码publicstaticvoidmain(String[]args)throwsIOException,InterruptedException{writeImagePath("D:\\labs\\mytest\\test.jpg")execPy()System.out.println(readAnswer())}}实际上就是通过Java执行一个命令行指令。

可以念没用Python的扩展来仔桐纳实现。Python本来是C实现的,封装二进制兼容的C++是很容易的。Java的话得通过JNI来实现,就是说在Python扩展里用C调用Java。另外,轮坦也可以写一个TCP服务来包装C++/Java的接口,通过网络来调用,这样更通用。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12333123.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-24
下一篇 2023-05-24

发表评论

登录后才能评论

评论列表(0条)

保存