简体   繁体   English

使用Bytebuddy拦截二传手

[英]Using Bytebuddy to intercept setter

lets asume i have a Interface like that: 假设我有一个像这样的接口:

public interface User extends Element {

    String getName();

    String getPassword();

}

and a implementing class like that: 和这样的实现类:

public class BaseUser implements User {

    @Override
    public String getId() {
        return id;
    }

    @Override
    public String getName() {
        return name;
    }

    @Override
    public String getPassword() {
        return password;
    }

    public void setId(String id) {
        this.id = id;
    }

    public void setName(String name) {
        this.name = name;
        System.out.println("Set name to " + name);
    }

    public void setPassword(String password) {
        this.password = password;
    }

    private String id;
    private String name;
    private String password;

}

Now i want to use bytebuddy to create a interceptor/proxy which catches the call onto the setter, store the changed value and call the real method also. 现在,我想使用bytebuddy创建一个拦截器/代理,以将调用捕获到setter上,存储更改的值并还调用real方法。

At the end i want to "ask" the interceptor/proxy for the called setter and the changed values. 最后,我想“询问”被调用的setter和更改后的值的拦截器/代理。

I tried a lot considering also the tutorials but up to now i found no working solution. 我也考虑了教程,尝试了很多,但是到目前为止,我还没有找到可行的解决方案。 Maybe someone could help me pls. 也许有人可以帮助我。

And here is the Interceptor: 这是拦截器:

public class GenericInterceptor implements InvocationHandler {

    @Override
    @RuntimeType
    public Object invoke(@This Object proxy, @Origin Method method, @AllArguments Object[] args) throws Throwable {
        if (isSetter(method, args)) {
            intercept(proxy, method, args);
        }
        return method.invoke(proxy, args);
    }

}

Here is my current 'test' code: 这是我当前的“测试”代码:

public static void main(String[] args) {
    final ByteBuddy bb = new ByteBuddy();
    final GenericInterceptor interceptor = new GenericInterceptor();

    bb.subclass(BaseUser.class)
            .method(isDeclaredBy(BaseUser.class).and(isSetter()))
            .intercept(MethodDelegation.to(interceptor))
            .make()
            .load(BaseUser.class.getClassLoader(), ClassLoadingStrategy.Default.WRAPPER);

    final BaseUser user = new BaseUser();
    user.setName("my name");
}

EDIT: 编辑:

public interface Element {
    String getId();
}

public class GenericInterceptor<T extends Element> {

    public GenericInterceptor(Class<T> type) {
        this.type = type;
    }

    public Map<String, Object> getChanges(T obj) {
        final String id = obj.getId();
        return changes.get(id);
    }

    @RuntimeType
    public void invoke(@This T proxy, @Origin Method method, @AllArguments Object[] args) throws Throwable {
        System.out.println("invoke " + method.getName() + " " + Arrays.toString(args));

        intercept(proxy, method, args);
    }

    private Object getCurrentValue(T proxy, final Field field) {
        try {
            return field.get(proxy);
        } catch (IllegalArgumentException | IllegalAccessException e) {
            return null;
        }
    }

    private Field getSetterField(Method setter) {
        final String setterName = setter.getName();

        Field f = assignedFields.get(setterName);
        if (f != null) return f;

        final String fieldName = Character.toLowerCase(setterName.charAt(3)) + setterName.substring(4);
        try {
            f = type.getDeclaredField(fieldName);
            if (f == null) return null;
            f.setAccessible(true);
            assignedFields.put(setterName, f);
            return f;
        } catch (NoSuchFieldException | SecurityException e) {
            return null;
        }
    }

    private void intercept(T proxy, Method setter, Object[] args) {
        final Field field = getSetterField(setter);
        if (field == null)
            return;

        final Object currentValue = getCurrentValue(proxy, field);
        final Object newValue = args[0];

        System.out.println("Set from " + currentValue + " to " + newValue);

        final String id = proxy.getId();
        Map<String, Object> changeMap = changes.get(id);
        if (changeMap == null) {
            changeMap = new HashMap<>();
        }
        changeMap.put(field.getName(), currentValue);
        changes.put(id, changeMap);
    }

    private final Map<String, Field> assignedFields = new HashMap<>();
    private final Map<String, Map<String, Object>> changes = new LinkedHashMap<>();
    private final Class<T> type;

}

You can call orignal method using MethodDelegation.to(...).andThen(SuperMethodCall.INSTANCE) . 您可以使用MethodDelegation.to(...)。andThen(SuperMethodCall.INSTANCE)调用原始方法。

public class ByteBuddyTest {

  public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, InstantiationException {
      GenericInterceptor interceptor = new GenericInterceptor ();

      Class<?> clazz = new ByteBuddy()
        .subclass(BaseUser.class)
        .method(ElementMatchers.isDeclaredBy(BaseUser.class).and(ElementMatchers.isSetter()))
            .intercept(SuperMethodCall.INSTANCE.andThen(MethodDelegation.to(interceptor))))
        .make()
        .load(ByteBuddyTest.class.getClassLoader())
        .getLoaded();

      BaseUser user1 = (BaseUser) clazz.getConstructors()[0].newInstance();
      BaseUser user2 = (BaseUser) clazz.getConstructors()[0].newInstance();
      user1.setName("user1");
      user1.setPassword("password1");
      user2.setName("user2");
      user2.setPassword("password2");

      System.out.println(interceptor.getInterceptedValue("user1", "name"));
      System.out.println(interceptor.getInterceptedValue("user1", "password"));
      System.out.println(interceptor.getInterceptedValue("user2", "name"));
      System.out.println(interceptor.getInterceptedValue("user2", "password"));

      user1.setPassword("password2");
      user1.setPassword("password3");
  }

  public static class GenericInterceptor {
      private Map<String, Object> interceptedValuesMap = new HashMap();

      public void set(String obj, @This User user, @Origin Method setter) {
        // assume that user name is unique so we can use it as a key in values map.
        // or define equals/hashcode in GenericUser object and use it as a key directly
        String setterName = setter.getName();
        String propertyName = setterName.substring(3, setterName.length()).toLowerCase();
        String key = user.getName() + "_" + propertyName;

        System.out.println("Setting " + propertyName + " to " + obj);
        System.out.println("Previous value " + interceptedValuesMap.get(key));

        interceptedValuesMap.put(key, obj);
      }

    public Object getInterceptedValue(String userName, String fieldName) {
        return interceptedValuesMap.get(userName + "_" + fieldName);
    }
  }

  public static interface User {

      String getName();

      String getPassword();

  }

  public static class BaseUser implements User {

      @Override
      public String getName() {
          return name;
      }

      @Override
      public String getPassword() {
          return password;
      }

      public void setName(String name) {
          this.name = name;
      }

      public void setPassword(String password) {
          this.password = password;
      }

      private String name;
      private String password;

  }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

相关问题 如何使用 Bytebuddy 拦截字段访问(没有 getter/setter) - How to intercept field accesses (without getter/setter) using Bytebuddy ByteBuddy 使用 arguments 拦截 setter 和调用方法,具体取决于调用 setter 方法的 object - ByteBuddy intercept setter and calling method with arguments depending on the object on which the setter method is called Bytebuddy拦截异常抛出 - Bytebuddy intercept exception throw 在 ByteBuddy 中使用 @Advice.AllArguments 是否可以拦截没有 arguments 的方法? - Is it possible to intercept a method that has no arguments with using @Advice.AllArguments in ByteBuddy? 在Android上可以使用ByteBuddy进行MethodDelegation拦截吗? - Is MethodDelegation intercept with ByteBuddy possible on Android? 在ByteBuddy中的rebase期间拦截构造函数 - Intercept constructors during rebase in ByteBuddy 是否可以使用 ByteBuddy 拦截数组构造函数? - Is it possible to intercept array constructor with ByteBuddy? 是否可以使用 bytebuddy 拦截在特定 java class 中初始化的执行程序服务的方法 - Is it possible to intercept methods of a Executor Service that is initialised in a specific java class using bytebuddy Bytebuddy - 拦截 java.net.ServerSocket 构造函数 - Bytebuddy - Intercept java.net.ServerSocket constructor ByteBuddy 是否可以拦截由带有注释的接口声明的方法? - Is it possible with ByteBuddy to intercept a method that is declared by an interface with an annotation?
 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM