[轉]java調用python腳本以及通過Process.waitFor()直接調用python模塊返回錯誤代碼1的一種解決辦法


常見的java調用python腳本方式

  • 通過jython提供的類庫實現
  • 通過Runtime.getRuntime()開啟進程來執行腳本文件

通過jython提供的類庫實現

  通過jython實現的話,我們需要引入jar包(jython官網:https://www.jython.org/),具體我寫了一個demo,假設你的python代碼為test.py:

public static void main(String[] args) {
      PythonInterpreter interpreter = new PythonInterpreter();  
      interpreter.execfile("E:\\workspace\\pycharm_workspace\\weixincrawer\\test.py");  
      PyFunction function = (PyFunction)interpreter.get("my_test",PyFunction.class);  
      PyObject pyobject = function.__call__(new PyString("huzhiwei"),new PyString("25")); 
      System.out.println("anwser = " + pyobject.toString());  
    }

輸出結果:

name: huzhiwei
age: 25
anwser = success

  到此是沒有什么問題的,我們使用function.call方法傳入參數調用python函數,使用pyobject.toString()方法拿到python中my_test函數的返回值,但是如果你把test.py稍微做下修改如下:

import requests

def my_test(name, age):
    response = requests.get("http://www.baidu.com")
    print("name: "+name)
    print("age: "+age)
    return "success"

  不修改java調用代碼的情況下,你會得到下面異常信息:

  ImportError: No module named requests

  沒錯,這就是我要討論的問題,因為jython不可能涵蓋所有python第三方類庫的東西,所以在我們得python文件中用到requests類庫的時候,很顯然會報找不到模塊的錯誤,這個時候我們是可以通過Runtime.getRuntime()開啟進程來執行python腳本文件的。

通過Runtime.getRuntime()開啟進程來執行腳本文件

使用這種方式需要同時修改python文件以及java調用代碼,在此我同樣在上面test.py的基礎上進行修改:

import requests
import sys

def my_test(name, age):
    response = requests.get("http://www.baidu.com")
    print("url:"+response.url)
    print("name: "+name)
    print("age: "+age)
    return "success"

my_test(sys.argv[1], sys.argv[2])

和上面test.py代碼最大的區別在於,我們此處開啟進程的方式實際上是在隱形的調用dos界面進行操作,因此在python代碼中我們需要通過sys.argv的方式來拿到java代碼中傳遞過來的參數。

java調用代碼部分:

public static void main(String[] args) {
      String[] arguments = new String[] {"python", "E:\\workspace\\pycharm_workspace\\weixincrawer\\test.py", "huzhiwei", "25"};
        try {
            Process process = Runtime.getRuntime().exec(arguments);
            BufferedReader in = new BufferedReader(new InputStreamReader(process.getInputStream()));
            String line = null;  
          while ((line = in.readLine()) != null) {  
              System.out.println(line);  
          }  
          in.close();  
          int re = process.waitFor();  
          System.out.println(re);
        } catch (Exception e) {
            e.printStackTrace();
        }  
    }

結果輸出:

url:http://www.baidu.com/
name: huzhiwei
age: 25
0

在此需要注意的一點,java代碼中的process.waitFor()返回值為0表示我們調用python腳本成功,返回值為1表示調用python腳本失敗,這和我們通常意義上見到的0與1定義正好相反。

java通過Process.waitFor()調用python模塊返回錯誤代碼1的一種解決辦法

  在本人的實際項目需求開發時,基本上也是模仿上面的思路進行的,python腳本成功在PyCharm Community中也執行也成功,獲得了所請求的網頁數據,但是在java環境中通過Process.waitFor()調用python模塊返回錯誤代碼1。

  經過多次嘗試和分析,發現

  問題原因:我的請求的URL設置為安全Http了,即https了,在PyCharm Community IDE可能存在HTTPS相關的處理模塊所以執行成功。但是在java環境中通過Process.waitFor()調用python模塊,實質上是通過dos/cmd命令行執行python.exe *.py命令的。而在這種情況下,由於環境中缺少HTTPS相關的處理模塊所以執行不成功。

  解決辦法:將請求的URL由安全Https修改為http即可

下面附上我的示例代碼:

被調用的python代碼模塊為:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2020/3/6 23:41
# @Author  : 
# @File    : URLibTest.py

import urllib.request
import sys

def WebSpider(name, age):
    response = urllib.request.urlopen('http://celestrak.com/satcat/tle.php?CATNR=25994')
    print(response.read().decode('utf-8'))


WebSpider(sys.argv[1], sys.argv[2])
# WebSpider("huzhiwei", "25")

主調的java代碼模塊為:

package cn.cetc;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;

public class JavaInvokePython {
    private static final String pyInterpreterPath = "C:\\Anaconda3\\python.exe";//注意:當命令行參數分開寫的時候,exe后面不用添加一個空格。當命令行參數一起寫的時候,exe后面一定要添加一個空格
    private static final String pyFilePath = "URLibTest.py";//如果未指定.py文件的完全路徑,則默認從工程當前目錄下搜索
    private static Process proc = null;//java進程類
     
    /**
     * 執行*.py文件
     */
    public static void execPy() {
        try {
            String[] arguments = new String[] {pyInterpreterPath, pyFilePath, "huzhiwei", "25"};//實際上后兩個參數傳進入也沒使用。當真正需要有參數傳入時可以是利用這種方式傳參
            proc = Runtime.getRuntime().exec(arguments);
            BufferedReader in = new BufferedReader(new InputStreamReader(proc.getInputStream()));
            String line = null;
            while ((line = in.readLine()) != null) {
                System.out.println(line);
            }
            in.close();
            
            int re = proc.waitFor();//返回0:成功。其余返回值均表示失敗,如:返回錯誤代碼1:操作不允許,表示調用python腳本失敗
            System.out.println(re);
        } catch (IOException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        execPy();
    }

}

注意:上述java代碼運行環境不需要額外安裝Jython-x.x.x.jar,只需要一般的JDK環境和常規的需要的包導入。

參考鏈接:

1、java調用python腳本

2、在Java中調用Python代碼

3、java調用python的幾種用法(看這篇就夠了)

4、Java調用Python程序方法總結

5、java調用python腳本無法正確獲取返回值

6、在Java中調用Python


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM