La API reflection se trata de leer y modificar el contenido de un objeto en tiempo de ejecución sin ningún conocimiento previo de la estructura o incluso la clase de ese objeto en el momento de la compilación. De hecho, la API de reflection trata de descubrir el contenido de un objeto y la estructura de la clase en tiempo de ejecución, que es una funcionalidad muy potente.
Hay varias clases fundamentales que se deben conocer en está API y el nombre esas clases es el siguiente, hay una clase llamada ‘Class’, que puede ser confuso, pero este es el caso. También una clase llamada ‘Field’ para modelar los campos dentro de una clase y una clase llamada ‘Method’ para modelar métodos, nuevamente dentro de una clase y hay una clase adicional llamada la clase ‘Constructor’, un constructor en la clase no es realmente un método, incluso si parece un método y los constructores se modelan utilizando su propia clase. También se usaría la clase ‘Annotation’ que modela las anotaciones.
String hello = "Hello";
Class helloClass = hello.getClass();
Class<?> helloClass = "Hello".getClass();
// la razón de esto radica en la forma en que los genéricos funcionan en java. De echo Class of String o Class of Object no son extensiones de Class of question mark.
Class<String> helloClass = "Hello".getClass(); // compile error.
Class<? extends String> helloClass = "Hello".getClass(); // is correct.
// obtiene una referencia de la clase misma.
Class<?> stringClass = String.class;
// recomienda porque Brinda muchas oportunidades de la reflection.
String className = "java.lang.String";
Class<?> stringClass = Class.forName(className);
// forName puede producir varias excepciones por casos de seguridad y también puede ser por falta de derechos.
Hay muchos métodos en la clase llamada ‘Class’, son métodos para obtener referencias en los campos, para obtener referencias en los métodos, en constructores y mucha más información.
Class<?> getClass = Person.class;
Field field = getClass.getField("age");
// los campos declarados son los campos declarados dentro de esa clase, ya sean privados, protegidos, protegidos por paquetes o públicos.
Field[] declaredFields = getClass.getDeclaredFields();
// Son los campos públicos de la clase, incluidos de la super clase.
Field[] fields = getClass.getFields();
Una de las formas de obtener los métodos es usando el método getMethod(), devolverá un solo método, pasamos el nombre del método y luego los tipos de parámetros en el orden correcto. Si este método toma varios parámetros, luego agregamos los otros tipos de los otros parámetros después del primero. También hay un método GetDeclaredMethod() que devolverá todos los métodos dentro de esa clase ya sean públicos, protegidos o privados y un método getMethods() que devolverá todos los métodos públicos declarados en esa clase y todos los super clases de esa clase.
Class<?> getClass = Person.class;
Method method = getClass.getMethod("setName", String.class);
Method[] declaredMethods = getClass.getDeclaredMethods();
Method[] methods = getClass.getMethods();
Para obtener los constructores de una clase, un constructor puede ser visto como una especie de método, pero de hecho no lo es y dentro del contexto de la API de reflection, esta modelado por una clase que es la clase constructora ‘Constructor’. El primer método es el mismo que el método o los campos. Es solo un método getConstructor() que toma los tipos de parámetros tomados por ese constructor.
Class<?> getClass = Person.class;
Constructor constructor = getClass.getConstructor(Class<?>... types);
Constructor[] declaredConstructors = getClass.getDeclaredConstructors();
//Los constructores de la super clase no estarán incluidos, es la diferencia con los métodos y campos.
Constructor[] constructors = getClass.getConstructos();
Si el campo dado es privado entonces habría una excepción que es una excepción IllegalAccessException porque no hay permiso para leer un campo privado desde fuera de la clase ‘Person’, entonces la encapsulación no está rota y si hay una verificación de seguridad para acceder a un miembro privado a una clase Reflection, hay un método en la clase ‘Field’ que se llama setAccessible(). La llamada a setAccessible() en true realmente hace que suprima el control de acceso en ese campo.
Person o = ...;
Class<?> getClass = o.getClass();
Field field = getClass.getDeclaredField("name");
field.setAccessible(true);
field.setValue(o, "Sarah");
String name = (String)field.getValue(o);
public class Person {
@PrimaryKey
private long id;
@Column
private int age;
@Column
private String name;
// getters and setters
}
public @interface PrimaryKey { }
public @interface Column { }
En la forma como funcionan las anotaciones en java, el compilador primero maneja una anotación y el compilador debe decidir en que momento del ciclo de vida de una clase estará disponible una anotación. Hay tres etapas cuando una anotación puede estar disponible.
- La primera etapa es la propia etapa de compilación, por lo que solo el compilador podrá ver esa anotación si elegimos hacerlo.
- La segunda etapa es la carga de clases por lo que la anotación solo verá el cargador de clases.
- La tercera es el propio tiempo de ejecución.
Si se está diseñando un objeto que necesita otro objeto para funcionar, un objeto delegado, entonces este primer objeto no debe intentar crear el objeto delegado, en lugar de eso cuando se construye el objeto, debería intentar pedir algún otro mecanismo para inyectar ese objeto delegado en el primer objeto. Entonces una forma de implementar esto es trabajar con un framework de inyección de dependencias, no es la única forma de implementar, pero probablemente sea la más clásica en las aplicaciones empresariales.
BeanManager beanManager = BeanManager.getInstance();
EntityManager entityManager = beanManager.get(EntityManager.class);
public class ConnectionProvider {
Connection createConnection(String url) { ... }
}
Class<?> connectionType = ConnectionProvider.class;
Object connectionProvider = connectionType.getConstructor().getInstance();
Method method = connectionType.getMethod("createConnection", String.class);
method.invoke(connectionProvider, "jdbc:h2:mem:db_reflection");
Cada vez que se llama a un método desde la API de reflection, se realizan varias comprobaciones de seguridad para verificar si el código puede acceder a una clase mediante reflection y todas esas comprobaciones se verifican nuevamente cada vez que se realiza un acceso, entonces si el mismo código se llama una y otra vez al mismo método de reflection, esas comprobaciones de seguridad se realizan una y otra vez solo para devolver el mismo resultado. Esto ha sido visto hace mucho tiempo, todos esos controles son costosos y tienen un impacto notable en el rendimiento.
Es un objeto de búsqueda porque encapsula toda la información de seguridad, ya que se comprueba una vez y como esa información no va a cambiar se usará todo el tiempo cuando se acceda a una clase, un campo o un método. Desde el objeto de búsqueda se puede crear instancias de MethodHandle para acceder a la clase, los métodos dentro de esa clase y los campos dentro de esa clase, por lo tanto, este único identificador de método le dará acceso a todo el contenido de la clase que se distribuyó entre varios objetos en la API de reflection de java, es decir, objetos de método, constructor y campo.
// Por cada subproceso con su rol de seguridad el objeto de
// búsqueda será diferente.
Lookup lookup = MethodHandles.lookup(); // factory method
// no debe compartirse con algún código no confiable,
// porque se tendrá acceso a los objetos utilizando reflection
// con la seguridad de otra persona.
Person person = ...;
// devuelve una instancia de Lookup
Class<?> personClass = MethodHandles.lookup()
.findClass(Person.class.getName());
Un tipo de método para un método que devuelve una cadena y no toma ningún argumento.
Lookup lookup = MethodHandles.lookup();
// public String getName() { ... }
MethodType getterType = MethodType.methodType(String.class);
Un tipo de método para un método que devuelve void y toma un argumento String.
// public void setName(String name) { … }
MethodType setterType = MethodType.methodType(void.class, String.class);
En el caso de un constructor, el tipo devuelto es void.class
// public Person() { ... }
MethodType emptyConstructorType = MethodType.methodType(void.class);
// public Person(String name, int age) { … }
MethodType constructorType = MethodType.methodType(void.class,
String.class, int.class);
Obtener un controlador en un método de instancia regular
Lookup lookup = MethodHandles.lookup();
// public String getName() { ... }
MethodType getterType = MethodType.methodType(String.class);
MethodHandle getterHandle = lookup.findVirtual(Person.class, "getName", getterType);
// public void setName(String name) { ... }
MethodType setterType = MethodType.methodType(void.class, String.class);
MethodHandle setterHandle = lookup.findVirtual(Person.class, "setName", setterType);
// public Person() { ... }
MethodType emptyConstructorType = MethodType.methodType(void.class); // not Void.class
MethodHandle emptyConstructorHandle = lookup.findConstructor(Person.class, emptyConstructorType);
// public Person(String name, int age) { ... }
MethodType constructorType = MethodType.methodType(void.class, String.class, int.class);
MethodHandle constructorHandle = lookup.findConstructor(Person.class, constructorType);
Un identificador devuelto por un findGetter da acceso de lectura en un campo y no llama al captador de ese campo.
Lookup lookup = MethodHandles.lookup();
// reads name
MethodHandle nameReader = lookup.findGetter(Person.class, "name", String.class);
Un identificador devuelto por un findSetter da acceso de escritura en un campo y no llama al configurador de ese campo.
Lookup lookup = MethodHandles.lookup();
// writes name
MethodHandle nameSetter = lookup.findSetter(Person.class, "name", String.class);
Invocar a un getter no requiere ningún argumento y devuelve una cadena.
Person person = ...;
MethodHandle nameGetter = ...;
String name = (String)nameGetter.invoke(person);
Invocar a un definidor requiere un argumento String y no devuelve nada.
Person person = ...;
MethodHandle nameSetter = ...;
nameSetter.invoke(person, "John");
Una solución pre-Java 9 es usar los métodos unreflect() para accesos privados.
Person person = ...;
Field nameField = Person.class.getDeclaredField("name");
nameField.setAccessible(true);
MethodHandle privateNameReader = lookup.unreflectGetter(field);
String name = (String)privateNameReader.invoke(person);
Java 9 trae una solución mucho más limpia, usando un objeto de búsqueda para los elementos privados de una clase.
Person person = ...;
Lookup privateLookup = MethodHandles.privateLookupIn(Person.class, lookup);
MethodHandle privateNameReader = privateLookup.findGetter(Person.class, "name", String.class);
String name = (String)privateNameReader.invoke(person);
MethodHandle privateNameWriter = privateLookup.findSetter(Person.class, "name", String.class);
privateNameWriter.invoke(person, "John");
Tiene una funcionalidad especial que fue agregada en java 9, parece un MethodHandle para los campos, el MethodHandle ya puede acceder a un campo, dándole acceso completo e incluso a los campos privados ¿por qué se han agregado VarHandles a Java 9? Es porque VarHandle da tres tipos de acceso a los campos.
- Es solo el acceso sencillo, regular, leer y escribir en campos públicos y privados, esto se hace en MethodHandles
- También brinda acceso volátil
- Compara y configura el acceso
El método get() invoca un manejador var en modo normal.
Lookup lookup = ...;
Person person = ...;
VarHandle nameVarHandle = MethodHandles.privateLookupIn(Person.class, lookup)
.findVarHandle(Person.class, "name", String.class);
String name = (String)nameVarHandle.get(person);
El método getVolatile () invoca un manejador var en modo volátil.
String name = (String)nameVarHandle.getVolatile(person);
El método getAndAdd () agrega atómicamente el valor pasado y devuelve el valor anterior.
Int newAge = (int)ageVarHandle.getAndAdd(person, 1);