Как устранить эту ошибку отражения? java.lang.reflect.Field.get(Field.java:393)

Я пытаюсь изменить значение @ColumnTransformer во время выполнения, чтобы я мог хранить свои ключи шифрования где-нибудь еще на сервере. Как я могу этого добиться?

Ниже мой основной код:

      List<Field> fields = Arrays.asList(User.class.getDeclaredFields());
for(Field field:fields)
{
    if(field.toString().endsWith("secretfield"))
    {
        System.out.println(field);
        List<Annotation> annotations = Arrays.asList(field.getDeclaredAnnotations());
        for(Annotation annotation : annotations)
        {
            if(annotation.annotationType().toString().endsWith("ColumnTransformer"))
            {
                Annotation newAnnotation = new ColumnTransformer(){
                    @Override
                    public String forColumn() {
                        return "secretfield";
                    }

                    @Override
                    public String read(){
                        return "This is Sparta";
                    }

                    @Override
                    public String write(){
                        return "This is also sparta";
                    }

                    @Override
                    public Class<? extends Annotation> annotationType() {
                        return annotation.annotationType();
                    }
                };
                Field field1 = User.class.getDeclaredField("secretfield");
                field1.setAccessible(true);
                Map<Class<? extends Annotation>, Annotation> annotations1 = (Map<Class<? extends Annotation>, Annotation>) field1.get(User.class);
                annotations1.put(ColumnTransformer.class, newAnnotation);
            }
        }
    }
}

А ниже мой Pojo:

      package Model;

import org.hibernate.annotations.ColumnTransformer;


import javax.persistence.*;
import javax.sql.rowset.serial.SerialBlob;
import java.sql.Blob;
import java.io.Serializable;

@Entity(name = "user")
@Table(name = "user")
public class User implements Serializable {

    @Column(name = "secretfield", nullable = true)
    @ColumnTransformer(read = "TEST")
    private String secretfield;

    public String getSecretfield() {
        return secretfield;
    }

    public void setSecretfield(String secretfield) {
        this.secretfield = secretfield;
    }
}

Я получаю эту ошибку:

      Exception in thread "main" java.lang.IllegalArgumentException: Can not set java.lang.String field Model.User.secretfield to null value
    at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:167)
    at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:171)
    at sun.reflect.UnsafeFieldAccessorImpl.ensureObj(UnsafeFieldAccessorImpl.java:58)
    at sun.reflect.UnsafeObjectFieldAccessorImpl.get(UnsafeObjectFieldAccessorImpl.java:36)
    at java.lang.reflect.Field.get(Field.java:393)
    at Runner.main(Runner.java:65)

1 ответ

Эта линия

      Map<Class<? extends Annotation>, Annotation> annotations1 = (Map<Class<? extends Annotation>, Annotation>) field1.get(User.class);

это неверно. field1.get извлекает значение поля вместо карты аннотации.

Поскольку вам нужно только изменить параметр внутри аннотации, вам действительно не нужно создавать новый экземпляр. Дополнительные сведения см. В разделе «Изменение параметра строки аннотации определения класса во время выполнения» .

changeAnnotationValue немного изменен, чтобы исправить NPE из-за ColumnTransformer отсутствует значение по умолчанию, как показано ниже.

      import org.hibernate.annotations.ColumnTransformer;

import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Proxy;
import java.util.Map;

public class FieldGetAnnotation {
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
        Field field = User.class.getDeclaredField("secretfield");
        field.setAccessible(true);
        final ColumnTransformer fieldAnnotation = field.getAnnotation(ColumnTransformer.class);
        System.out.println("old FieldAnnotation = " + fieldAnnotation.forColumn());
        changeAnnotationValue(fieldAnnotation, "forColumn", "secretfield");
        System.out.println("modified FieldAnnotation = " + fieldAnnotation.forColumn());
        System.out.println("old FieldAnnotation = " + fieldAnnotation.read());
        changeAnnotationValue(fieldAnnotation, "read", "This is Sparta");
        System.out.println("modified FieldAnnotation = " + fieldAnnotation.read());
        System.out.println("old FieldAnnotation = " + fieldAnnotation.write());
        changeAnnotationValue(fieldAnnotation, "write", "This is also sparta");
        System.out.println("modified FieldAnnotation = " + fieldAnnotation.write());
    }

    @SuppressWarnings("unchecked")
    public static Object changeAnnotationValue(Annotation annotation, String key, Object newValue) {
        Object handler = Proxy.getInvocationHandler(annotation);
        Field f;
        try {
            f = handler.getClass().getDeclaredField("memberValues");
        } catch (NoSuchFieldException | SecurityException e) {
            throw new IllegalStateException(e);
        }
        f.setAccessible(true);
        Map<String, Object> memberValues;
        try {
            memberValues = (Map<String, Object>) f.get(handler);
        } catch (IllegalArgumentException | IllegalAccessException e) {
            throw new IllegalStateException(e);
        }
        Object oldValue = memberValues.get(key);
        if (oldValue != null && oldValue.getClass() != newValue.getClass()) {
            throw new IllegalArgumentException();
        }
        memberValues.put(key, newValue);
        return oldValue;
    }
}
Другие вопросы по тегам