Как создать динамический прокси класса без публичного конструктора, используя ByteBuddy
Я хочу создать динамический прокси класса Sample, у которого нет двух открытых конструкторов, он не работает и выдает ошибку. Но если я сделаю конструктор как Public, он будет работать нормально. Возможно ли в байтовом партнере добиться этого?
Также возможно ли сделать метод addToList(..) закрытым?
Образец кода:
public class Sample {
private String name;
private String college;
private String id;
private List<String> fieldList = new LinkedList<>();
Sample() {
//some code
System.out.println("No arg constructor invoked");
}
Sample(String id) {
this();
this.id = id;
}
public String getId() {
return this.id;
}
public String getName() {
return this.name;
}
public void setName(String name) {
System.out.println("Setting name: "+ name);
this.name = name;
}
public String getCollege() {
return college;
}
public void setCollege(String college) {
System.out.println("Setting college: "+college);
this.college = college;
}
public void addToList(String fieldName){
fieldList.add(fieldName);
}
public List<String> getFieldList() {
return Collections.unmodifiableList(fieldList);
}
public static Sample getProxyObject(String id) {
Sample proxyObj = null;
try {
Class<? extends Sample> dynamicType = new ByteBuddy()
.subclass(Sample.class, ConstructorStrategy.Default.IMITATE_SUPER_CLASS )
.method(ElementMatchers.nameStartsWith("set"))
.intercept(MethodDelegation.to(new GreetingInterceptor()))
.make()
.load(Sample.class.getClassLoader())
.getLoaded();
proxyObj = dynamicType.getConstructor(String.class).newInstance(id);
} catch (Exception ex){
ex.printStackTrace();
}
return proxyObj;
}
public static Sample getProxyObject() {
Sample proxyObj = null;
try {
proxyObj = new ByteBuddy()
.subclass(Sample.class)
.method(ElementMatchers.nameStartsWith("setName"))
.intercept(MethodDelegation.to(new GreetingInterceptor()))
.make()
.load(Sample.class.getClassLoader())
.getLoaded().newInstance();
} catch (Exception ex){
ex.printStackTrace();
}
return proxyObj;
}
public static class GreetingInterceptor {
public void abc(@SuperCall Callable<Void> zuper, @Origin Method method, @Super Sample parentObj, @This Object myself, @AllArguments Object[] args) {
try {
parentObj.addToList(method.getName());
zuper.call();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
Основной класс для тестирования:
public class SampleMain {
public static void main(String[] args) {
System.out.println("===Scenario 1=====");
Sample proxyObject = Sample.getProxyObject();
proxyObject.setName("John Doe");
System.out.println("===Scenario 2=====");
proxyObject.getFieldList().stream().forEach(System.out::println);
Sample proxyObject1 = Sample.getProxyObject("id123");
proxyObject1.setName("John Doe");
proxyObject1.setCollege("MIT");
System.out.println("Id is: "+proxyObject1.getId());
proxyObject1.getFieldList().stream().forEach(System.out::println);
}
}
Трассировка ошибок:
=== Сценарий 1=====
Exception in thread "main" java.lang.IllegalAccessError: tried to access method com.algorithm.Sample.<init>()V from class com.algorithm.Sample$ByteBuddy$J74XiIU8
at com.algorithm.Sample$ByteBuddy$J74XiIU8.<init>(Unknown Source)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
at java.lang.Class.newInstance(Class.java:442)
at com.algorithm.Sample.getProxyObject(Sample.java:88)
at com.algorithm.SampleMain.main(SampleMain.java:6)
2 ответа
В случае protected
конструкторы в порядке, тогда:
- + Изменить
Sample()
вprotected Sample()
чтобы сценарий 1 работал: это делает конструктор без аргументов доступным из подкласса, созданного ByteBuddy. + Изменить
Sample(String id)
вprotected Sample(String id)
а затем вgetProxyObject(String id)
менятьproxyObj = dynamicType.getConstructor(String.class).newInstance(id);
в
Constructor<? extends Sample> ctor = dynamicType.getDeclaredConstructor(String.class); ctor.setAccessible(true); proxyObj = ctor.newInstance(id);
Это делает сценарий 2 рабочим.
ConstructorStrategy.Default.IMITATE_SUPER_CLASS
производит тот же защищенный конструктор в подклассе, поэтому вам нужно использоватьgetDeclaredConstructor()
чтобы получить его, а затем сделать его доступным.
Вы также можете сделать addToList(..)
защищены, это будет работать просто отлично, так как GreetingInterceptor
будет иметь доступ к нему.
Обратите внимание, что частный конструктор пакета виден только классу, если он определен в том же пакете времени выполнения. По умолчанию Byte Buddy создает новый загрузчик классов при загрузке класса без указания ClassLoadingStrategy
, Если пакет назван одинаково, но не загружен одним и тем же загрузчиком классов, пакеты времени выполнения будут другими.
Я предполагаю, что, указав:
.load(Sample.class.getClassLoader(), ClassLoadingStrategy.Default.INJECTION)
ваш прокси будет работать как положено, несмотря на нарушенный рефлексивный доступ, о котором говорилось в другом ответе. Однако обратите внимание, что в этой стратегии используется небезопасный API, который может больше не работать в будущей версии JVM.