python調用java函數
很多朋友都想知道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代碼
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
❸ python 調用java對象
你使用jython這個解釋器就可以讓python直接調用java, 調用完成後,你用python封裝成一個服務。其它的python程序員就可以間接調用java對象了。
jython調用java這個方式也被eclipse+pydev使用,是目前最直接的方法。
❹ python 調用java 是每一次都啟動jvm嗎
是的
一、JPype簡述
1.JPype是什麼?
JPype是一個能夠讓 python 代碼方便地調用 Java 代碼的工具,從而克服了 python 在某些領域(如伺服器端編程)中的不足。
2.JPype與Jython(JPython後繼者)的區別?
1)運行環境不同:jython運行在jvm上,而JPype的實際運行環境仍然是python runtime,只是在運行期間啟動了一個嵌入的jvm;
2)使用者不同:jython是給java程序玩的,JPype是給python程序員玩的。
二、JPype安裝
1.先安裝Python2.7和JAVA1.6
2.安裝JPype-0.5.4.2.win32-py2.7.exe(http://sourceforge.net/projects/jpype/files/JPype/0.5.4/)
3.Ubuntu12.04安裝命令:sudo apt-get install python-jpype
三、JPype使用說明
1.啟動JVM
JPype 提供的 startJVM() 函數的作用是啟動 JAVA 虛擬機,所以在後續的任何 JAVA 代碼被調用前,必須先調用此方法啟動 JAVA 虛擬機。
jpype.startJVM() 的定義
startJVM(jvm, *args)
jpype.startJVM() 的參數
參數 1: jvm, 描述你系統中 jvm.dll 文件所在的路徑,如「 C:Program FilesIBMJava50jreinj9vmjvm.dll 」。可以通過調用 jpype.getDefaultJVMPath() 得到默認的 JVM 路徑。
參數 2: args, 為可選參數,會被 JPype 直接傳遞給 JVM 作為 Java 虛擬機的啟動參數。此處適合所有合法的 JVM 啟動參數,例如:
-agentlib:libname[=options]
-classpath classpath
-verbose
-Xint在 JVM 啟動時設置系統變數示例:
在 JVM 的啟動參數中加入如下參數:
-Dproperty=valuefromjpypeimport*
importos.path
startJVM("C:/Java/jdk1.6.0_10/jre/bin/client/jvm.dll","-ea")
java.lang.System.out.println("helloWorld")
shutdownJVM()
packagejpype;
publicclassJpypeDemo{
publicStringsayHello(Stringuser){
return"hello"+user;
}
publicintcalc(inta,intb){
returna+b;
}
}
fromjpypeimport*
importos.path
jarpath=os.path.join(os.path.abspath('.'),'F:/sample_Py/')
startJVM("C:/Java/jdk1.6.0_10/jre/bin/client/jvm.dll","-ea","-Djava.class.path=%s"%(jarpath+'jpypedemo.jar'))
#ubuntu 中startJVM("/home/geek/Android/jdk1.6.0_43/jre/lib/i386/server/libjvm.so","-ea","-Djava.class.path=%s"%(jarpath+'XXX.jar'))
JDClass=JClass("jpype.JpypeDemo")
jd=JDClass()
#jd=JPackage("jpype").JpypeDemo()#兩種創建jd的方法
jprint=java.lang.System.out.println
jprint(jd.sayHello("waw"))
jprint(jd.calc(2,4))
shutdownJVM()
2.關閉JVM
當使用完 JVM 後,可以通過 jpype.shutdownJVM() 來關閉 JVM,該函數沒有輸入參數。當 python 程序退出時,JVM 會自動關閉。
3.引用第三方Java擴展包
很多時候,在 python 項目中需要調用第三方的 Java 擴展包,這也是 JPype 的一個重要用途。
通過在 JVM 啟動參數增加:-Djava.class.path=ext_classpath,實現在 python 代碼中調用已有的 Java 擴展包。
4.訪問JAVA的系統屬性
有時,某些 Java 應用需要設置或者獲取 JVM 中的系統屬性。
四、舉例
1.直接調用JAVA API
[java]view plain
2.調用JAVA第三方擴展包
1)JAVA自定義第三方jar包:將JpypeDemo類打包為jpypedemo.jar文件並存儲到F:/sample_Py目錄下
[java]view plain
2)Python調用第三方JAVA jar包程序
[python]view plain
3.訪問JAVA的系統屬性
假設你要設置的屬性名為 yourProperty,屬性值為 yourValue 。
1)JVM啟動時設置系統變數示例
import jpype
jvmPath = jpype.getDefaultJVMPath()
jvmArg = 「 -DyourProperty=yourValue 」
if not jpype.isJVMStarted():
jpype.startJVM(jvmPath,jvmArg)
2)在程序中設置系統變數示例
import jpype
prop = 「 yourProperty 」
value = 「 yourValue 」
system = jpype.JClass('java.lang.System')
system.setProperty(str(prop),str(value))
3)在程序中獲取系統變數示例
import jpype
prop = 「 yourProperty 」
system = jpype.JClass('java.lang.System')
value = system.getProperty(str(prop))
❺ 有沒有從Python調用Java的好方法
[java]view plain
String[]arg=newString[]{"python",types,parameter};//第一個參數是python解釋器位置,第二個參數是執行的python腳本位置,接下來的都是參數
Processprocess=Runtime.getRuntime().exec(arg);
=newInputStreamReader(process.getInputStream());
BufferedReaderbufferedReader=newBufferedReader(inputStreamReader);
Stringline="";
StringBufferstringBuffer=newStringBuffer();
while((line=bufferedReader.readLine())!=null){
stringBuffer.append(line);
stringBuffer.append(" ");
}
bufferedReader.close();
process.waitFor();
❻ python調用的java進程在哪看
python調用的java進程在哪看?
最近在做基於python的locust壓力測試,api介面程序是java版本,導致python無法匹配簽名模式,需要python調用java的簽名程序。
首先需要按照python依賴包
pip install jpype1
jpype調用jar包
jpype的原理是在python進程中嵌入了java虛擬機,並與虛擬機進行通信。
復制代碼
import jpype
# 如果只有java程序,那需要先打成一個jar包來給python引用 ,有java.jar就可以直接使用
def init_jvm():
jpype.startJVM(jpype.getDefaultJVMPath(), "-ea", "-Djava.class.path=%s" % 'java.jar')
def getsign(privateKey,publicKey):
VerCls = jpype.JClass("com.sign") # 引用的java包與類名
sign = VerCls().getSign(privateKey,publicKey)
return sign
def shutdowm():
try:
jpype.shutdownJVM()
except:
pass
if __name__=="__main__":
init_jvm()
print(getsign("priyyyy","alpha1234"))
shutdowm()
復制代碼
代碼說明如下:
需要提供java.jar,如果只有java程序,那需要先打成一個jar包來給python引
init_jvm函數中的 jpype.startJVM用來啟動java虛擬機,-Djava.class.path 用來指定我們要調用的jar包。
在getsign中,jpype.JClass用來引用sign類,通過實例化並調用getSign實現版本號的比較。
jpype.shutdownJVM()是主動關閉java虛擬機,釋放資源。
❼ python 怎麼把一個類傳遞給JAVA,然後Java調用python類的回調函數
走api啊,標准json傳輸,然後分別處理被