Как сериализовать лямбда?

Как я могу элегантно сериализовать лямбда?

Например, приведенный ниже код бросает a NotSerializableException. Как я могу исправить это, не создавая интерфейс SerializableRunnable "dummy"?

public static void main(String[] args) throws Exception {
    File file = Files.createTempFile("lambda", "ser").toFile();
    try (ObjectOutput oo = new ObjectOutputStream(new FileOutputStream(file))) {
        Runnable r = () -> System.out.println("Can I be serialized?");
        oo.writeObject(r);
    }

    try (ObjectInput oi = new ObjectInputStream(new FileInputStream(file))) {
        Runnable  r = (Runnable) oi.readObject();
        r.run();
    }
}

Ответ 2

Такая же конструкция может использоваться для ссылок на методы. Например, этот код:

import java.io.Serializable;

public class Test {
    static Object bar(String s) {
        return "make serializable";
    }

    void m () {
        SAM s1 = (SAM & Serializable) Test::bar;
        SAM s2 = (SAM & Serializable) t -> "make serializable";
    }

    interface SAM {
        Object action(String s);
    }
}

определяет лямбда-выражение и ссылку на метод с сериализуемым типом цели.

Ответ 3

Очень уродливый актерский состав. Я предпочитаю определять расширение Serializable для функционального интерфейса, который я использую

Например:

interface SerializableFunction<T,R> extends Function<T,R>, Serializable {}
interface SerializableConsumer<T> extends Consumer<T>, Serializable {}

тогда метод, принимающий лямбда, может быть определен как таковой:

private void someFunction(SerializableFunction<String, Object> function) {
   ...
}

и вызывая функцию, вы можете передать свою лямбду без какого-либо уродливого броска:

someFunction(arg -> doXYZ(arg));

Ответ 4

Если вы хотите переключиться на другую структуру сериализации, например Kryo, вы можете избавиться от множества ограничений или требования, реализованный интерфейс должен реализовывать Serializable. Подход к

  • Измените InnerClassLambdaMetafactory, чтобы всегда генерировать код, необходимый для сериализации
  • Непосредственно вызовите LambdaMetaFactory во время десериализации

Подробнее и код см. в этом сообщении в блоге

Ответ 5

На случай, если кто-то попадет сюда при создании кода Beam/Dataflow: Beam имеет собственный интерфейс SerializableFunction, поэтому не тратьте время на создание собственного интерфейса или подробное приведение типов.