Требуется ли Assigner.Typing.DYNAMI C Assigner при возврате подтипа? - PullRequest
1 голос
/ 13 июля 2020

Предположим, я использую sh для реализации интерфейса Function<Object, Object>, делая что-то очень глупое вроде этого:

final ParameterizedType whatTheUserWillSee = functionObjectObject(); // e.g. Function<Object, Object>
DynamicType.Builder<?> builder = builder
  .subclass(Object.class, ConstructorStrategy.Default.NO_CONSTRUCTORS)
      .merge(Visibility.PUBLIC, SyntheticState.SYNTHETIC, TypeManifestation.FINAL)
      .defineConstructor(Visibility.PUBLIC, SyntheticState.SYNTHETIC)
      .intercept(MethodCall.invoke(OBJECT_CONSTRUCTOR).onSuper()) // OBJECT_CONSTRUCTOR is just new Object()
      .implement(whatTheUserWillSee)
      .intercept(MethodCall.invoke(new MethodDescription.ForLoadedMethod(Object.class.getMethod("toString"))
                 .withAssigner(Assigner.DEFAULT, Assigner.Typing.DYNAMIC));

То есть я создаю что-то вроде этого:

public Foo() {
  super();
}
public Object apply(Object x) {
  return toString();
}

Без вызова .withAssigner я получаю:

java.lang.IllegalStateException: Cannot return class java.lang.String from public java.util.function.Function java.util.function.Function.compose(java.util.function.Function)
    at net.bytebuddy.implementation.MethodCall$TerminationHandler$Simple$1.toStackManipulation(MethodCall.java:3005)
    at net.bytebuddy.implementation.MethodCall$Appender.toStackManipulation(MethodCall.java:3551)
    at net.bytebuddy.implementation.MethodCall$Appender.apply(MethodCall.java:3509)
    at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod$WithBody.applyCode(TypeWriter.java:708)
    at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod$WithBody.applyBody(TypeWriter.java:693)
    at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod.apply(TypeWriter.java:600)
    at net.bytebuddy.dynamic.scaffold.TypeWriter$Default$ForCreation.create(TypeWriter.java:5660)
    at net.bytebuddy.dynamic.scaffold.TypeWriter$Default.make(TypeWriter.java:2166)
    at net.bytebuddy.dynamic.scaffold.subclass.SubclassDynamicTypeBuilder.make(SubclassDynamicTypeBuilder.java:232)
    at net.bytebuddy.dynamic.scaffold.subclass.SubclassDynamicTypeBuilder.make(SubclassDynamicTypeBuilder.java:204)
    at net.bytebuddy.dynamic.DynamicType$Builder$AbstractBase.make(DynamicType.java:3595)

Я не эксперт во внутреннем устройстве ByteBuddy. Почему для возврата подтипа Object необходим назначитель типа Dynami c?

1 Ответ

1 голос
/ 16 июля 2020

Вы перехватываете все методы Function. Пока вы думаете, что перехватываете apply, если вы посмотрите внимательно, Byte Buddy жалуется на метод compose. Чтобы этого избежать, укажите явное сопоставление.

...