现象
jython脚本
void testPythonCallJava2(){
System.out.println("testPythonCallJava");
PythonInterpreter pythonInterpreter = new PythonInterpreter();
StringBuilder builder = new StringBuilder();
builder.append("from com.zp.learn.proxy import Hello\n");
builder.append("h = Hello()\n");
builder.append("h.sayHello()\n");
builder.append("h.sayHello('word')\n");
builder.append("print('中国')\n");
builder.append("h.sayHello('中国')\n");
builder.append("data = h.getData(3)\n");
builder.append("print 'jython %s' % data\n");
builder.append("data2 = h.getMutilData(2)\n");
builder.append("print 'jython %s' % data2\n");
PyCode pyCode = pythonInterpreter.compile(builder.toString(),"testPythonCallJava");
pythonInterpreter.exec(pyCode);
List data = pythonInterpreter.get("data", List.class);
System.out.println("main "+data);
List<Map> data2 = pythonInterpreter.get("data2", List.class);
System.out.println("main data2 "+data2);
}
Hello.java
public class Hello {
public void sayHello(){
System.out.println("Hello python!");
}
public void sayHello(String msg){
System.out.println("Hello "+msg);
}
/**
* 获取指定长度的列表,供jypthon调用
* @param length
* @return
*/
public List<Map<String,Object>> getData(int length){
if(length<1){
return Collections.emptyList();
}
List data = new ArrayList();
for (int i = 0; i< length;i++){
Map<String,Object> map = new HashMap<>();
map.put("k_"+i,"v_"+i);
data.add(map);
}
System.out.println("java"+data);
return data;
}
/**
* 获取指定长度的稍微复查的列表,供jypthon调用
* @param length
* @return
*/
public List<Map<String,Object>> getMutilData(int length){
if(length<1){
return Collections.emptyList();
}
List data = new ArrayList();
for (int i = 0; i< length;i++){
Map<String,Object> map = new HashMap<>();
map.put("k_"+i,getData(i+1));
data.add(map);
}
System.out.println("java m "+data);
return data;
}
}
执行结果
testPythonCallJava
Hello python!
Hello word
中国
Hello ä¸å½
java[{k_0=v_0}, {k_1=v_1}, {k_2=v_2}]
jython [{k_0=v_0}, {k_1=v_1}, {k_2=v_2}]
java[{k_0=v_0}]
java[{k_0=v_0}, {k_1=v_1}]
java m [{k_0=[{k_0=v_0}]}, {k_1=[{k_0=v_0}, {k_1=v_1}]}]
jython [{k_0=[{k_0=v_0}]}, {k_1=[{k_0=v_0}, {k_1=v_1}]}]
main [{k_0=v_0}, {k_1=v_1}, {k_2=v_2}]
main data2 [{k_0=[{k_0=v_0}]}, {k_1=[{k_0=v_0}, {k_1=v_1}]}]
Process finished with exit code 0
看到了输出的结果里的第5行取到的结果是乱码的。
分析原因
弯路(分析过程)
在网上搜到解决问题方法:
1、有的说要设置python脚本的编码格式,在脚本第一行添加注释“#-*- encoding: utf-8 -*-”告诉解析器,统一的编码,结果是失败的。
2、有个方案是PyString codeStr = Py.newStringUTF8(code);任然是对脚本进行编码,如图:
但是我尝试的结果还是失败的,他的这个方案没有从python脚本里再次调用java方法。但是发现了一个现象。在python脚本里如果通过json.dumps(中文)后再传给java方法,java中取到数据尽然不乱码了,但是确实变成了unicode编码了。
PythonInterpreter pythonInterpreter = new PythonInterpreter();
StringBuilder builder = new StringBuilder();
builder.append("from com.zp.learn.proxy import Hello\n");
builder.append("import json\n");
builder.append("h = Hello()\n");
builder.append("h.sayHello()\n");
builder.append("h.sayHello('word')\n");
builder.append("print('中国')\n");
builder.append("h.sayHello(json.dumps('中国'))\n");
PyCode pyCode = pythonInterpreter.compile(builder.toString(),"testPythonCallJava");
pythonInterpreter.exec(pyCode);
执行结果:
testPythonCallJava
Hello python!
Hello word
中国
Hello "\u4e2d\u56fd"
推测原因
原因就是pyhton脚本对中文字符进行了编码,在python脚本内执行时有进行了自动的解码处理,所有pyhton内置的函数print(中文)能正常显示。
解决方法
大概指定原因后,我就在脚本上做了些调整,将python脚本里要传给java方法的字符串做个解码处理 decode('UTF-8');
System.out.println("testPythonCallJava");
PythonInterpreter pythonInterpreter = new PythonInterpreter();
StringBuilder builder = new StringBuilder();
builder.append("from com.zp.learn.proxy import Hello\n");
builder.append("h = Hello()\n");
builder.append("h.sayHello()\n");
builder.append("h.sayHello('word')\n");
builder.append("print('中国')\n");
builder.append("h.sayHello('中国'.decode('UTF-8'))\n");
builder.append("h.sayHello('u8932'.decode('UTF-8'))\n");
builder.append("data = h.getData(3)\n");
builder.append("print 'jython %s' % data\n");
builder.append("data2 = h.getMutilData(2)\n");
builder.append("print 'jython %s' % data2\n");
PyCode pyCode = pythonInterpreter.compile(builder.toString(),"testPythonCallJava");
pythonInterpreter.exec(pyCode);
List data = pythonInterpreter.get("data", List.class);
System.out.println("main "+data);
List<Map> data2 = pythonInterpreter.get("data2", List.class);
System.out.println("main data2 "+data2);
执行结果
testPythonCallJava
Hello python!
Hello word
中国
Hello 中国
Hello u8932
java[{k_0=v_0}, {k_1=v_1}, {k_2=v_2}]
jython [{k_0=v_0}, {k_1=v_1}, {k_2=v_2}]
java[{k_0=v_0}]
java[{k_0=v_0}, {k_1=v_1}]
java m [{k_0=[{k_0=v_0}]}, {k_1=[{k_0=v_0}, {k_1=v_1}]}]
jython [{k_0=[{k_0=v_0}]}, {k_1=[{k_0=v_0}, {k_1=v_1}]}]
main [{k_0=v_0}, {k_1=v_1}, {k_2=v_2}]
main data2 [{k_0=[{k_0=v_0}]}, {k_1=[{k_0=v_0}, {k_1=v_1}]}]
看到效果是成功的,只要在python脚本里做了解码处理,通过jython解析器执行后,在java代码中就可以正常获取的中文数据了。