Java調(diào)用Python腳本傳遞數(shù)據(jù)并返回計算結(jié)果
需求:最近在使用基于Java編寫的Cloudsim 4.0云仿真平臺進行虛擬機動態(tài)遷移實驗,由于中間有需要用到深度強化學(xué)習(xí)算法,因此需要將集群的狀態(tài)表示為二維數(shù)組,比如物理機的計算能力Mips,RAM,帶寬等等。希望將這樣的二維數(shù)組傳入到帶torch等第三方庫的Python腳本進行深度強化學(xué)習(xí)訓(xùn)練,所以就有二維int數(shù)組的傳入,和從Python計算后的結(jié)果返回讀取這個需求
一、實現(xiàn)思路:將Java中的data結(jié)構(gòu)化為字符串,以命令行參數(shù)的形式傳入Python中
目前有幾種Java調(diào)用Python的方法,不過能良好兼容Python第三方庫的方法通常是使用
Process proc = Runtime.getRuntime().exec(args1); // 執(zhí)行Python腳本并傳參數(shù)
如果只是簡單傳入幾個數(shù)字,或者幾個URL,比如可直接寫為
int num1 = 5; int num2 = 10; Process proc = Runtime.getRuntime().exec(args1, String.valueOf(num1), String.valueOf(num2));
而如果要傳入多維數(shù)組且每次傳遞時數(shù)組的大小會變,比如
int[][] stateInt = new int[][]{{2500, 5, 2610, 2620, 2630, 2640, 2650, 2660}, // Mips
{870, 5, 4091, 4092, 4093, 4094, 4095, 4096}}; // RAM
則需要把要傳入的多維數(shù)據(jù),結(jié)構(gòu)化為可分割的字符串,上述二維數(shù)組就可轉(zhuǎn)變?yōu)槿缦伦址?/p>
"2500 5 2610 2620 2630 2640 2650 2660;870 5 4091 4092 4093 4094 4095 4096"
這樣傳入到Python中就可以根據(jù);和<空格>通過split()將二維數(shù)組恢復(fù)出來
二、Python實現(xiàn)代碼
import sys
from selenium import webdriver
import torch
def policy(state):
action = [2, 1, 0, 0]
action[0] += state[0][1]
return action
def str2int(stateStr):
'''將完整字符串轉(zhuǎn)換為二維數(shù)組'''
stateList = []
multiVimState = stateStr.split(';')
for singleVimState in multiVimState:
elements = singleVimState.split(' ')
singleVimList = []
for e in elements:
singleVimList.append(int(e))
stateList.append(singleVimList)
return stateList
def int2str(actionIntArr):
'''將形如[0,1,0,0,0]的int動作向量轉(zhuǎn)化為01000字符串,方便Java處理'''
actionStr = '';
for e in actionIntArr:
actionStr += str(e)
return actionStr
if __name__ == '__main__':
state = []
stateStr = sys.argv[1];
stateIntArr = str2int(stateStr)
actionIntArr = policy(stateIntArr)
actionStr = int2str(actionIntArr)
# [2+5=7, 1, 0, 0] => 7100
print(actionStr)
三、Java實現(xiàn)代碼
import java.io.BufferedReader;
import java.io.InputStreamReader;
public class testPython {
/**
* 將整型state數(shù)組轉(zhuǎn)換為帶分隔符的字符串,方便以命令方式傳遞給Python文件以進行計算
* @param stateInt 當(dāng)前虛機 + 所有主機狀態(tài)向量
* @return
*/
public static String state2str(int[][] stateInt){
String stateStr = "";
for (int i = 0; i < stateInt.length; i++) {
for (int j = 0; j < stateInt[0].length; j++) {
if (j == 0) stateStr += String.valueOf(stateInt[i][j]);
else stateStr += " " + String.valueOf(stateInt[i][j]);
}
if (i != stateInt.length - 1) stateStr += ";";
}
return stateStr;
}
public static void main(String[] args) throws Exception {
// Python文件地址(Linux)
String pyPath = "/XXXX/XXXX.py";
int[][] stateInt = new int[][]{{2500, 5, 2610, 2620, 2630, 2640, 2650, 2660}, // Mips
{870, 5, 4091, 4092, 4093, 4094, 4095, 4096}}; // RAM
// 將整型state數(shù)組轉(zhuǎn)換為帶分隔符的字符串,方便以命令方式傳遞給Python文件以進行計算
String stateStr = state2str(stateInt);
String[] args1 = new String[] {"python", pyPath, stateStr};
// 執(zhí)行Python文件,并傳入?yún)?shù)
Process proc = Runtime.getRuntime().exec(args1);
// 獲取Python輸出字符串作為輸入流被Java讀取
BufferedReader in = new BufferedReader(new InputStreamReader( proc.getInputStream() ));
String actionStr = in.readLine();
if (actionStr != null)
System.out.println(actionStr);
in.close();
proc.waitFor();
// 將獲取的字符串分割為字符串?dāng)?shù)組,然后逐個元素轉(zhuǎn)換為int并求和
String nums[] = actionStr.split("");
int sum = 0;
for (int i = 0; i < nums.length; i++)
sum += Integer.valueOf(nums[i]);
System.out.println("求和為:" + sum);
}
}
運行Java代碼后得到如下結(jié)果

到此這篇關(guān)于Java調(diào)用Python腳本傳遞數(shù)據(jù)并返回計算結(jié)果的文章就介紹到這了,更多相關(guān)Java調(diào)用Python腳本內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用try-with-resource的輸入輸出流自動關(guān)閉
這篇文章主要介紹了使用try-with-resource的輸入輸出流自動關(guān)閉方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
Spring Cloud Zipkin服務(wù)端追蹤服務(wù)
這篇文章主要介紹了Spring Cloud Zipkin服務(wù)端追蹤服務(wù),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-04-04
使用Springboot根據(jù)配置文件動態(tài)注入接口實現(xiàn)類
這篇文章主要介紹了使用Springboot根據(jù)配置文件動態(tài)注入接口實現(xiàn)類,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
SpringBoot中使用Servlet的兩種方式小結(jié)
這篇文章主要介紹了SpringBoot中使用Servlet的兩種方式小結(jié),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-07-07
使用Java WebSocket獲取客戶端IP地址的示例代碼
在開發(fā)Web應(yīng)用程序時,我們通常需要獲取客戶端的 IP 地址用于日志記錄、身份驗證、限制訪問等操作,本文將介紹如何使用Java WebSocket API獲取客戶端IP地址,以及如何在常見的WebSocket框架中獲得客戶端 IP地址,需要的朋友可以參考下2023-11-11
json-lib將json格式的字符串,轉(zhuǎn)化為java對象的實例
下面小編就為大家?guī)硪黄猨son-lib將json格式的字符串,轉(zhuǎn)化為java對象的實例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-03-03
關(guān)于springboot2整合lettuce啟動卡住問題的解決方法
Lettuce和Jedis的都是連接Redis Server的客戶端程序,下面這篇文章主要給大家介紹了關(guān)于springboot2整合lettuce啟動卡住問題的解決方法,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下2021-12-12

