解决jython脚本中含有中文参数,在java中获取的参数乱码

现象

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代码中就可以正常获取的中文数据了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java调用Python代码有多种方法,其一种是使用JythonJython是一个Java平台上的Python解释器,它允许在Java程序嵌入Python代码,并且可以在Java代码直接调用Python模块和函数。 以下是使用JythonJava调用Python代码的基本步骤: 1. 下载并安装Jython 你可以从Jython官方网站下载Jython安装包,并按照安装向导进行安装。 2. 创建Python脚本 编写一个Python脚本,例如hello.py: ``` def say_hello(name): print("Hello, " + name + "!") ``` 3. 在Java代码加载并调用Python模块 在Java代码使用JythonPythonInterpreter类加载并调用Python模块和函数,例如: ``` import org.python.util.PythonInterpreter; public class Main { public static void main(String[] args) { PythonInterpreter interpreter = new PythonInterpreter(); interpreter.execfile("hello.py"); interpreter.get("say_hello").__call__("John"); } } ``` 在上面的代码,我们首先创建了一个PythonInterpreter对象,然后使用它的execfile方法加载Python脚本文件。接着,我们使用interpreter.get方法获取Python模块的say_hello函数,并使用__call__方法调用该函数,并传递一个参数。 当你运行Java程序时,你应该能够在控制台上看到输出:Hello, John! 需要注意的是,Jython并不是Python的完全实现,它只支持Python 2.7的语法和一些标准库。如果你的Python代码使用了不受支持的语法或库,那么你需要考虑其他的方法来在Java调用Python代码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值