Jython, использовать только метод из Python из Java?

При чтении и использовании этой статьипредполагается, что у нас есть полное определение объекта с классом и отображением (proxy) объекты от python до java.

Можно ли импортировать метод (не определенный внутри класса, а использующий внутренний класс python) из фрагмента кода в python, не обернув его в определение класса (без использования описанной выше парадигмы фабрики).

Я хотел бы сделать какой-то из myPyFile импорт myMethodиз java, а затем использовать myMethod, непосредственно из java (может быть, как статический метод?) ? Но если это возможно, я не нашел никакого понятия о том, как это сделать (интерфейс, описанный в статье, может быть все еще необходим, чтобы рассказать Java, как использовать myMethod?)

С наилучшими пожеланиями.

EDIT: Сейчас я имею дело сJython 2.5.2 , так что это может быть зависит от версии и намного проще в будущем?

EDIT :Ниже в ответ Даниилу :

Вот пример кода, чтобы воспроизвести ошибку, которую я получаю, а также получить рабочий пример из вашего полезного ответа!

(Ну и добавьте немного другого вопроса о сопоставлении с объектами Java из yield-ed Python/Jython result)

(@Joonas, Извините, я изменил свой код, и теперь я не могу вернуться к ошибке, которая у меня была раньше)

import org.python.core.Py;
import org.python.core.PyList;
import org.python.core.PyTuple;
import org.python.core.PyObject;
import org.python.core.PyString;
import org.python.core.PySystemState;
import org.python.util.PythonInterpreter;

interface MyInterface {
    public PyList getSomething(String content, String glue, boolean bool);
}
class MyFactory {

    @SuppressWarnings("static-access")
    public MyFactory() {
        String cmd = "from mymodule import MyClass";
        PythonInterpreter interpreter = new PythonInterpreter(null, new PySystemState());

        PySystemState sys = Py.getSystemState();
        sys.path.append(new PyString("C:/jython2.5.2/Lib"));

        interpreter.exec(cmd);
        jyObjClass = interpreter.get("MyClass");
    }

    public MyInterface createMe() {
        PyObject myObj = jyObjClass.__call__();
        return (MyInterface)myObj.__tojava__(MyInterface.class);
    }

    private PyObject jyObjClass;
}


public class Main {

    public static void main(String[] args) {

    /*
// with only :
    PythonInterpreter interpreter = new PythonInterpreter();

     i get : 
Exception in thread "main" Traceback (most recent call last):
  File "", line 1, in 
LookupError: no codec search functions registered: can't find encoding 'iso8859_1'

which is probably due to the : 
#!/usr/bin/env python
# -*- coding: latin-1 -*-

// yes i am from France, so my - sorry for that - bad english ;) and have to deal with non 127 ascii chars :)
     */

    PythonInterpreter interpreter = new PythonInterpreter(null, new PySystemState());

    PySystemState sys = Py.getSystemState();
    sys.path.append(new PyString("C:/jython2.5.2/Lib"));

    interpreter.exec("from mymodule import getSomething"); 
    PyObject tmpFunction = interpreter.get("getSomething"); 
    System.err.println(tmpFunction.getClass()); 
    MyInterface i = (MyInterface) tmpFunction.__tojava__(MyInterface.class); 
    System.err.println(i.getSomething("test", " - ", true));
    for (Object x : i.getSomething("test", " - ", true)) {
        System.out.println(x);
        // How can i get back an equivallent of the Python _"for (a, b) in getSomething:"_ 
        // with _"a"_ being PyUnicode or better String, and _"b"_ being boolean ?
    }

    // ^^ so adapting Daniel Teply solution works ! Thanks to him... 
    // BTW the part below did not work : but i may have missed or/and also mixed many things :/
    // i feel Jython damned hard to dive in :/ 
    // and really hope that the sample that i post and answers that i get will help others to swim!

    try {
        MyFactory factory = new MyFactory();
        MyInterface myobj = factory.createMe();

        PyList myResult = myobj.getSomething("test", " - ", true);
        System.out.println(myResult);
    }
    catch (Exception e) {
        System.out.println(e);
        // produce a : java.lang.ClassCastException: org.python.core.PySingleton cannot be cast to MyInterface
        // EDIT : see below last edit, this error may be due to my forgotten heritage from interface in the python code!
    }

    System.exit(-1);
    }
}

Python часть: (mymodule.py)

#!/usr/bin/env python
# -*- coding: latin-1 -*-

class MyClass:
    def __init__(selfself):
        pass
    def getSomething(self, content, glue = '', bool = True):
        for x in range(5):
            yield (glue.join(map(str, (content, x, chr(97 + x))))), bool
        #return list()

def getSomething(content, glue = '', bool = True):
    print "test"
    myclass = MyClass()
    return list(myclass.getSomething(content, glue, bool))

def main():
    test()

if __name__ == "__main__":
    main()

EDIT:

Частично отвечая себе, для внутреннего вопроса (внутренние комментарии):
(на самом деле я чувствую, что мой ответ и код уродливы, но это работает и, кажется, нормально для отмены кортежа Я не знаю, есть ли лучший Jythonic-способ сделать это, если это так, я действительно заинтересован :))

for (Object x : i.getSomething("test", " - ", true)) {
    System.out.println(x);
    // How can i get back an equivallent of the Python _"for (a, b) in getSomething:"_ 
    // with _"a"_ being PyUnicode or better String, and _"b"_ being boolean ?

    // answering myself here :
    PyTuple mytuple = (PyTuple) x; // casting back x as PyTuple, can we have a java equivalent to _`(a, b) = x_ ? not sure...
    PyObject a = mytuple.__getitem__(0);
    PyObject b = mytuple.__getitem__(1);
    String aS = a.toString(); // mapping a unicode python string to java is as simple?
    boolean bB = b.toString().toLowerCase().equals("true");
    System.out.println(mytuple + "[" + aS + "][" + b + "][" + bB + "]");


ПРАВКА:

Отвечая себе на часть про "produce a : "java.lang.ClassCastException: org.python.core.PySingleton не может быть приведен к MyInterface"... большинство моих недоразумений и ошибок где из-за того, что я забыл обрабатывать Java из части Python! (см. мой код выше, я оставляю его неисправленным по поводу этого факта, потому что это был не мой главный вопрос, а в его фактическом виде, это рабочий ответ на этот главный вопрос, большое спасибо Даниэлю и Йоонасу, которые помогли мне понять). Таким образом, для фабричной парадигмы следует NOTзабыть добавить адекватный импорт в свой файл Python :

from testjython.interfaces import MyInterface #// defining method inside a MyInterface.java

class MyClass(MyInterface):
    [...]

Еще одна трудность, с которой я столкнулся, заключалась в том, чтобы правильно обрабатывать импорт и пакеты. Кстати, добавление этого кода в верхний код приведет к появлению TypeError: не удается преобразовать в org.python.core.PyList, но это еще одна проблема...

14
задан user369450 5 September 2014 в 22:40
поделиться