当前位置: 首页 > 知识库问答 >
问题:

使用字节伙伴的方法委托

宁良平
2023-03-14

我在使用Byte Buddy时遇到了一个简单的问题,下面是我的代码:

import static java.util.Arrays.asList;
import java.util.stream.Stream;
import net.bytebuddy.ByteBuddy;
import net.bytebuddy.dynamic.loading.ClassLoadingStrategy;
import net.bytebuddy.implementation.MethodDelegation;

public class Foo {
  public static class Bar {
    public void bar (String s, Integer i) {
      System.out.println ("s:" + s + " i:" + i);
    }
  }
  public static class Baz {
    public void baz (Integer i, String s) {
      System.out.println ("i:" + i + " s:" + s);
    }
  }

  public static void main (String[] args) {
    try {
      Class<?> c = new ByteBuddy ()
        .subclass (Object.class)
        .defineMethod ("bar", Void.class, asList (String.class, Integer.class), 1)
        .intercept (MethodDelegation.to (new Bar ()))
        .defineMethod ("baz", Void.class, asList (Integer.class, String.class), 1)
        .intercept (MethodDelegation.to (new Baz ()))
        .make ()
        .load (Foo.class.getClassLoader (), ClassLoadingStrategy.Default.WRAPPER)
        .getLoaded ();
      Stream.of (c.getDeclaredMethods ()).forEach (System.out::println);
    } catch (Exception e) {
      e.printStackTrace ();
    }
  }
}

我希望工作并将调用转发到受人尊敬的拦截器实例,但是我得到这个异常

java.lang.IllegalArgumentException: None of [protected void java.lang.Object.finalize() throws java.lang.Throwable, public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException, public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException, public final void java.lang.Object.wait() throws java.lang.InterruptedException, public boolean java.lang.Object.equals(java.lang.Object), public java.lang.String java.lang.Object.toString(), public native int java.lang.Object.hashCode(), public final native java.lang.Class java.lang.Object.getClass(), protected native java.lang.Object java.lang.Object.clone() throws java.lang.CloneNotSupportedException, public final native void java.lang.Object.notify(), public final native void java.lang.Object.notifyAll(), public void Foo$Bar.bar(java.lang.String,java.lang.Integer)] allows for delegation from public java.lang.Void net.bytebuddy.renamed.java.lang.Object$ByteBuddy$P87HQ3lQ.bar(java.lang.String,java.lang.Integer)
at net.bytebuddy.implementation.bind.MethodDelegationBinder$Processor.process(MethodDelegationBinder.java:881)
at net.bytebuddy.implementation.MethodDelegation$Appender.apply(MethodDelegation.java:1218)
at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod$WithBody.applyBody(TypeWriter.java:510)
at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod.apply(TypeWriter.java:444)
at net.bytebuddy.dynamic.scaffold.TypeWriter$Default$ForCreation.create(TypeWriter.java:3193)
at net.bytebuddy.dynamic.scaffold.TypeWriter$Default.make(TypeWriter.java:1481)
at net.bytebuddy.dynamic.scaffold.subclass.SubclassDynamicTypeBuilder.make(SubclassDynamicTypeBuilder.java:234)
at net.bytebuddy.dynamic.DynamicType$Builder$AbstractBase$AbstractDelegatingBuilder.make(DynamicType.java:2177)
at Foo.main(Foo.java:55)

最奇怪的部分在异常消息的最后:

[...] public void Foo$Bar.bar(java.lang.String,java.lang.Integer)] allows for delegation from public java.lang.Void net.bytebuddy.renamed.java.lang.Object$ByteBuddy$P87HQ3lQ.bar(java.lang.String,java.lang.Integer)

我做错了什么?

共有2个答案

归星驰
2023-03-14

当我提交并回顾我的问题时,我认为问题可能是void返回,而不是void

阎星华
2023-03-14

你的假设是对的。Byte Buddy无法将void非类型分配给void类型。然而,当您使用@RuntimeType注释拦截器时,Byte Buddy可以将null分配给void方法的引用类型:

public class Bar {
  @RuntimeType
  public void bar (String s, Integer i) {
    System.out.println ("s:" + s + " i:" + i);
  }
}
public class Baz {
  @RuntimeType
  public void baz (Integer i, String s) {
    System.out.println ("i:" + i + " s:" + s);
  }
}
 类似资料:
  • 我正在尝试用Byte Buddy拦截构造函数调用,这是我的示例代码: 我看到了这个相关的问题,但是,我得到了一个冲突异常(就好像构造函数被定义了两次)。

  • 京东金融 App介绍 [京东金融](jr.jd.com)京东金融是京东数字科技集团旗下专注于金融科技服务的重要业务板块。 京东金融始终基于强大的数字科技能力,致力于为让消费者享受专业、安全的数字金融服务,旗下包含个人和企业两大服务体系。 在个人金融领域,“京东金融App”作为载体,通过独有的大数据技术及人工智能风控能力,携手400多家银行、120余家保险公司、110余家基金公司,已为过亿的消费者精

  • 如何在构造函数拦截器中启动对象字段? 我在Buddy代码中创建了一个字节构造函数。 构造函数首先将参数保存到私有字段。然后它创建集合。然后它调用以下拦截器来填充该集合。 最好在拦截器中实例化variableNamedField字段,因为事实证明,每次创建新的类实例时,variableNamedField字段都是用相同的HashMap对象实例化的。但是,我只能通过@FieldValue注释将现有字段

  • 「Container+」联盟是由 DaoCloud 于今年6月份发起的开放性组织。联盟以打造立体生态、共赢容器市场为目标,旨在推动容器技术在企业落地。 联合产业链上下游的合作伙伴 共同推动企业级容器云平台发展 帮助企业实现互联网的架构和业务双转型 如您有意成为 DaoCloud 「Container+」联盟成员,请与我们联系。

  • 4.4.1.3 创建/使用伙伴服务 伙伴服务是只能由特定应用使用的服务。 系统由伙伴公司的应用和内部应用组成,用于保护在伙伴应用和内部应用之间处理的信息和功能。 以下是 AIDL 绑定类型服务的示例。 要点(创建服务): 不要定义意图过滤器,并将导出属性显式设置为true。 验证请求应用的证书是否已在自己的白名单中注册。 请勿(无法)通过onBind(onStartCommand, onHandl

  • 4.1.1.3 创建/使用伙伴活动 伙伴活动是只能由特定应用程序使用的活动。 它们在想要安全共享信息和功能的伙伴公司之间使用。 第三方应用程序可能会读取用于启动活动的意图。 因此,如果你将敏感信息放入用于启动活动的意图中,有必要采取对策来确保其无法被恶意第三方读取。 创建伙伴活动的示例代码如下所示。 要点(创建活动): 不要指定taskAffinity。 不要指定launchMode。 不要定义意