Python中使用jpype調(diào)用Jar包中的實現(xiàn)方法
使用jpype調(diào)用Jar包中的實現(xiàn)方法
安裝
pip install jpype1(注意要加后邊這個1)
使用
基本流程如下:
- 使用jpype開啟jvm
- 加載java類
- 調(diào)用java方法
- 關閉jvm
說明
我這里是在Python中使用Java的第三方抽象語法樹包JavaParser(Python中的javalang實在太難用了),實現(xiàn)得到一個類文件中的所有的方法的功能
代碼
Python代碼:
import jpype import os import json if __name__ == '__main__': ? ? # 加載jar包 ? ? jarpath = os.path.join(os.path.abspath('.'),'D:/study/hdu-cs-learning/Paper/TD/BuildDataset/target/BuildDataset.jar') ? ? # 獲取jvm.dll的默認文件路徑 ? ? jvmPath = jpype.getDefaultJVMPath() ? ? # 開啟虛擬機 ? ? jpype.startJVM(jvmPath, '-ea', '-Djava.class.path=%s' % (jarpath)) ? ? # 加載java類(參數(shù)名是java的長類名) ? ? javaClass = jpype.JClass('scluis.API') ? ? # 調(diào)用類中的方法 ? ? class_file_path = 'D:/study/TD/OpenSourceProject/JFreeChart/source/org/jfree/chart/fx/ChartViewer.java' ? ? # 這里是調(diào)用scluis.API類的靜態(tài)方法getMethods,如果調(diào)用實例方法,則需要先實例化java對象,即javaInstance = javaClass(),在使用實例調(diào)用 ? ? file_methods_str = javaClass.getMethods(class_file_path) ? ? # 解析返回值,這里的返回值是json數(shù)組 ? ? methods_list = "" ? ? if file_methods_str != "": ? ? ? ? methods_list = json.loads(str(file_methods_str)) ? ? # 關閉虛擬機 ? ? jpype.shutdownJVM()
API.java:
package scluis; import com.github.javaparser.ParseException; import java.io.IOException; import java.security.SecureRandom; import java.util.Random; public class API { ? ? public static void main(String[] args) { ? ? ?? ?//main方法是為了測試Java代碼即,getMethods,main方法不是必須的 ? ? ? ? String filePath="D:/study/OpenSourceProject/SQuirrel/app/src/net/sourceforge/squirrel_sql/client/gui/HelpViewerWindow.java"; ? ? ? ? String methods=getMethods(filePath); ? ? ? ? System.out.println(methods); ? ? } ? ? public static String getMethods(String filePath){ ? ? ? ? String res=""; ? ? ? ? try { ? ? ? ? ? ? Parser fileParser = new Parser(); ? ? ? ? ? ? res= fileParser.getFileMethods(filePath); ? ? ? ? } catch (ParseException | IOException e) { ? ? ? ? ? ? e.printStackTrace(); ? ? ? ? } ? ? ? ? return res; ? ? } }
Parser.java(內(nèi)含Java返回值格式)
package scluis; import com.alibaba.fastjson.JSON; import com.github.javaparser.*; import com.github.javaparser.ast.CompilationUnit; import com.github.javaparser.ast.Node; import com.github.javaparser.ast.NodeList; import com.github.javaparser.ast.body.MethodDeclaration; import com.github.javaparser.ast.comments.Comment; import com.github.javaparser.ast.expr.*; import com.github.javaparser.ast.stmt.BlockStmt; import com.github.javaparser.ast.stmt.ExpressionStmt; import com.github.javaparser.ast.stmt.Statement; import com.github.javaparser.printer.DotPrinter; import java.io.File; import java.io.IOException; import java.util.ArrayList; import java.util.List; import java.util.Optional; import java.util.stream.Stream; public class Parser { ? ? private CompilationUnit m_CompilationUnit; ? ? public CompilationUnit getParsedFile() { ? ? ? ? return m_CompilationUnit; ? ? } ? ? public String getFileMethods(String filePath) throws ParseException, IOException { ? ? ? ? String methodJson = ""; ? ? ? ? try { ? ? ? ? ? ? m_CompilationUnit = StaticJavaParser.parse(new File(filePath)); ? ? ? ? ? ? FunctionVisitor functionVisitor = new FunctionVisitor(); ? ? ? ? ? ? functionVisitor.visit(m_CompilationUnit, null); ? ? ? ? ? ? ArrayList<MethodDeclaration> nodes = functionVisitor.getMethodDeclarations(); ? ? ? ? ? ? ArrayList<Method> methodList = new ArrayList<>(); ? ? ? ? ? ? for (int i = 0; i < nodes.size(); i++) { ? ? ? ? ? ? ? ? MethodDeclaration methodDeclaration = nodes.get(i); ? ? ? ? ? ? ? ? //起止行 ? ? ? ? ? ? ? ? int startLine = methodDeclaration.getRange().get().begin.line; ? ? ? ? ? ? ? ? int endLine = methodDeclaration.getRange().get().end.line; ? ? ? ? ? ? ? ? //方法代碼 ? ? ? ? ? ? ? ? String method = methodDeclaration.removeComment().toString(); ? ? ? ? ? ? ? ? Method m = new Method(startLine, endLine, method); ? ? ? ? ? ? ? ? methodList.add(m); ? ? ? ? ? ? } ? ? ? ? ? ? methodJson = JSON.toJSONString(methodList); ? ? ? ? } catch (Exception e) { ? ? ? ? ? ? e.printStackTrace(); ? ? ? ? } ? ? ? ? return methodJson; ? ? } }
jpype調(diào)用jar包“Class xx not found“問題
環(huán)境
- Java 1.8 (64位)
- Python 2.7.9 (32位)
- Jpype 0.5.4.2
- Pycharm
代碼
import jpype from jpype import * # 該目錄下有需要調(diào)用的jar包 pak.jar jvmArg = "-Djava.ext.dirs=D:/1_Workspace/jpype_test/" if not jpype.isJVMStarted(): jvmPath = jpype.getDefaultJVMPath() jpype.startJVM(jvmPath,"-ea", jvmArg) # pak.jar中包含類com.abc.EFG jd = JClass("com.abc.EFG") instance = jd() print(instance.getName()) jpype.shutdownJVM()
問題
執(zhí)行以上代碼,報錯如下。Class com.abc.EFG not found
檢查點
1.在pak.jar包中有com.abc.EFG
類
2.EFG
依賴的jar包都在同級目錄下
3.Jpype使用沒有問題:調(diào)用System.out.println
可以打印
jprint = java.lang.System.out.println jprint("xxx") #輸出xxx
試著從環(huán)境上找原因,也許和Java版本有關。
首先查找pak.jar包的編譯jdk版本
使用以上方法獲得輸出是 52,即對應java 1.8,與本機Java版本相符。
但是回頭一想其實在Pycharm中并沒有配置過Java的版本。
那么打印一下jvmPath
,獲得的路徑是C:\Program Files (x86)\Java\jre6\bin\client\jvm.dll
,=> 使用的是 jre1.6(32位)。
此時去Java1.8目錄找,發(fā)現(xiàn)server
目錄下有jvm.dll
,不是在client
目錄。管他的,嘗試直接指定存在的jvm.dll路徑。
# ... # jvmPath = jpype.getDefaultJVMPath() jvmPath = r'D:\java\jdk1.8\jre\bin\server\jvm.dll' jpype.startJVM(jvmPath,"-ea", jvmArg) # ...
到這里也許有人的問題可以解決了,但我的情況是還是報錯…
解決
最后裝了jdk1.8 32位?。。?,發(fā)現(xiàn)在client
目錄下有jvm.dll
文件了
再次修改代碼,執(zhí)行通過。
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
python中l(wèi)ower函數(shù)實現(xiàn)方法及用法講解
在本篇文章里小編給大家整理的是一篇關于python中l(wèi)ower函數(shù)實現(xiàn)方法及用法講解內(nèi)容,有需要的朋友們可以學習參考下。2020-12-12詳解使用python3.7配置開發(fā)釘釘群自定義機器人(2020年新版攻略)
這篇文章主要介紹了詳解使用python3.7配置開發(fā)釘釘群自定義機器人(2020年新版攻略),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-04-04使用Python FastAPI構建Web服務的實現(xiàn)
這篇文章主要介紹了使用Python FastAPI構建Web服務的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-06-06將Python的Django框架與認證系統(tǒng)整合的方法
這篇文章主要介紹了將Python的Django框架與認證系統(tǒng)整合的方法,包括指定認證后臺和編寫認證后臺等內(nèi)容,需要的朋友可以參考下2015-07-07基于tkinter中ttk控件的width-height設置方式
這篇文章主要介紹了基于tkinter中ttk控件的width-height設置方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-05-05