java調用python返回值
① python 怎麼把一個類傳遞給java,然後Java調用python類的回調函數
走api啊,標准json傳輸,然後分別處理被
② java調python
很多朋友都想知道java怎麼調python?下面就一起來了解一下吧~
java調python主要有兩種方法:1.使用Runtime.getRuntime()執行腳本文件;2. 將python腳本寫成進程為java提供服務,下面是具體的方法介紹:
第一種:使用Runtime.getRuntime()執行腳本文件
先建立python腳本文件 demo.py
import numpy as np a = np.arange(12).reshape(3,4)print(a)
java調用python程序並輸出該結果
import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;public 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 sys def 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 = 18;int b = 23;try { 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 hello;import java.lang.System;import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.net.InetAddress;import java.net.Socket;import java.io.OutputStream;import java.io.PrintStream;import java.io.InputStream;public 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("遠程介面調用結束."); } }}
③ 用java執行python
1.直接執行Python腳本代碼
引用 org.python包
1 PythonInterpreter interpreter = new PythonInterpreter();
2 interpreter.exec("days=('mod','Tue','Wed','Thu','Fri','Sat','Sun'); "); ///執行python腳本
2. 執行python .py文件
1 PythonInterpreter interpreter = new PythonInterpreter();
2 InputStream filepy = new FileInputStream("D:\\demo.py");
3 interpreter.execfile(filepy); ///執行python py文件
4 filepy.close();
3. 使用Runtime.getRuntime()執行腳本文件
這種方式和.net下面調用cmd執行命令的方式類似。如果執行的python腳本有引用第三方包的,建議使用此種方式。使用上面兩種方式會報錯java ImportError: No mole named arcpy。
1 Process proc = Runtime.getRuntime().exec("python D:\\demo.py");
2 proc.waitFor();
④ java執行python腳本獲取返回值問題
java執行這個腳本並獲取返回值是等待腳本執行完畢再獲取返回的。
我不清楚你是通過什麼方式來執行的。
不過你可以啟動兩個線程,一個線程開始執行腳本,一個線程去獲取輸出。
⑤ 如何在Java中調用Python代碼
Jython(原JPython),是一個用Java語言寫的Python解釋器。
在沒有第三方模塊的情況下,通常選擇利用Jython來調用Python代碼,
它是一個開源的JAR包,你可以到官網下載
一個HelloPython程序
importorg.python.util.PythonInterpreter;
publicclassHelloPython{
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:Nomolenamedxxx
僅在Java中調用Python編寫的函數
先完成一個hello.py代碼:
defhello():
return'Hello'
在Java代碼中調用這個函數:
importorg.python.core.PyFunction;
importorg.python.core.PyObject;
importorg.python.util.PythonInterpreter;
publicclassHelloPython{
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(Iterator
⑥ bufferedreaderreadline換行符不完全卡死
bufferedreaderreadline換行符不完全卡死
問題1 Java調用python返回值一直為null
問題2 BufferedReader的readline函數無任何輸出結果,進程長時間卡住
最近在做項目時,需要用到Java調用python文件並讀取運行結果。一般Java調用簡單的Python文件(不包含第三方庫的)是沒有任何問題的,但是我需要用到tensorflow、keras、numpy等第三方庫,所以遇到了很多麻煩,經過各種網路查找,最終解決了問題,在這里做一個總結歸納。
問題1 Java調用python返回值一直為null
這個問題是我沒有創建虛擬環境造成的,因為自己電腦上用的一直是Anaconda安裝時自帶的python,沒有創建自己的虛擬環境,因此在Java中調用python文件時,就會找不到各種第三方庫的問題。
問題2 BufferedReader的readline函數無任何輸出結果,進程長時間卡住
這個問題我也找了很多方法才解決,產生這個問題的主要原因是:
命令在運行的過程中會向標准輸出或者標准錯誤輸出寫出數據,但JVM又沒有去讀,導致緩沖區滿,進而導致進程阻塞。因為我的python文件調用的tensorflow,他會輸出一堆警告信息,這些警告信息屬於錯誤流(ErrorStream),如果我們沒有及時處理,就會導致緩沖區占滿,從而導致進程阻塞。
Process.getInputStream()和Process.getErrorStream()分別返回Process的標准輸出流和錯誤流,兩個流如果處理不當,其緩沖區不能被及時清除而被塞滿,則進程被阻塞,即使調用Process.destory()也未必能銷毀被阻塞的子進程。
這個問題的解決的方法比較簡單,既然問題是緩沖區滿之後沒有及時清理,那麼只要在Java代碼里去讀一下數據,保證緩沖區不會滿即可.
⑦ java運行python程序,.readLine()讀出的內容為null, proc.waitFor();返回值為1,求解
你在while循環里加行代碼
line = decodeUnicode(line);