El contenedor IoC de Spring gestiona no solo la creación de instancias de sus objetos (beans), sino también la vinculación de objetos que interactúan (o dependencias). Si necesita inyectar (por ejemplo) un bean que está dentro del alcance de una solicitud HTTP en otro bean con un alcance a largo plazo, puede inyectar un proxy AOP en lugar del bean que está dentro del alcance. Es decir, necesita implementar un objeto proxy que exponga la misma interfaz pública que el objeto dentro del alcance, pero que también pueda recuperar el objeto de destino real del alcance correspondiente (por ejemplo, solicitud HTTP) y delegar llamadas a métodos al objeto real. .

También puede usar <aop:scoped-proxy/> entre beans definidos como singleton, con la referencia pasando a través de un proxy intermedio que es serializable. y, por lo tanto, puede volver a obtener el bean singleton objetivo tras la deserialización.

Cuando declaras <aop:scoped-proxy/> para un bean de alcance de visibilidad prototype, cada llamada a un método en el proxy compartido da como resultado la creación de un nueva instancia de destino, a la que luego se transfiere la llamada.

Además, los proxies con ámbito no son la única forma de acceder a beans desde ámbitos más cortos de una manera segura para el ciclo de vida. También es posible declarar el punto de inyección (es decir, un argumento constructor o definidor o un campo enlazado automáticamente) como ObjectFactory<MyTargetBean>, lo que permite la llamada a getObject(). para recuperar la instancia actual a pedido cada vez que sea necesario, sin tener que conservar la instancia ni almacenarla por separado.

Como opción extendida, puede declarar un ObjectProvider<MyTargetBean>, que proporciona varias opciones de acceso adicionales, incluidas getIfAvailable y getIfUnique.

La opción JSR-330 se llama Provider y se usa con una declaración Provider<MyTargetBean> y una llamada get() correspondiente para cada intento de obtención.

La configuración en el siguiente ejemplo es solo una línea, pero es importante entender por qué y cómo funciona:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
    <!-- Bean con ámbito de sesión HTTP expuesto como proxy -->
    <bean id="userPreferences" class="com.something.UserPreferences" alcance="sesión">
        <!-- indica al contenedor que actúe como proxy del bean circundante -->
        <aop:scoped-proxy/> 
    </frijol>
    <!-- un bean en el alcance de un singleton, inyectado por proxy al bean anterior -->
    <bean id="userService" class="com.something.SimpleUserService">
        <!-- enlace al bean proxy Preferencias de usuario -->
        <nombre de propiedad="preferencias de usuario" ref="Preferencias de usuario"/>
    </frijol>
</beans>
  1. String defining the proxy

Para crear un proxy de este tipo, debe insertar un elemento secundario <aop:scoped-proxy/> en la definición del bean dentro del alcance. ¿Por qué las definiciones de beans que están dentro del alcance de solicitud, sesión o niveles de alcance especiales requieren un <aop:scoped-proxy/> de elemento? Considere la siguiente definición de un bean singleton y compárela con lo que necesita definir para los ámbitos anteriores (tenga en cuenta que la siguiente definición del bean userPreferences tal como está está incompleta):

<bean id="userPreferences" class="com.something.UserPreferences" alcance="session"/>
<bean id="userManager" class="com.something.UserManager">
    <nombre de propiedad="preferencias de usuario" ref="Preferencias de usuario"/>
</bean>

En el ejemplo anterior, el bean único (userManager) se implementa utilizando una referencia al bean (userPreferences), ubicado en el área de visibilidad de HTTP. Sesión. El punto importante aquí es que el bean userManager es un objeto singleton: se crea una instancia exactamente una vez para cada contenedor y sus dependencias (en este caso, solo una, el bean userPreferences ) ) también se implementan solo una vez. Esto significa que el bean userManager solo funciona con el mismo objeto userPreferences (es decir, aquel con el que se implementó originalmente).

Este no es el comportamiento apropiado al inyectar un bean con un ciclo de vida más corto que está dentro del alcance en un bean con un ciclo de vida más largo que está dentro del alcance (por ejemplo, inyectar un bean interoperativo que está dentro del alcance del código HTTP Sesión, como una dependencia en un único bean). Más bien, requiere un objeto userManager y, durante la duración de la sesión HTTP, un objeto userPreferences específico para esa sesión . Entonces, el contenedor crea un objeto que expone exactamente la misma interfaz pública que la clase UserPreferences (idealmente un objeto que es una instancia de UserPreferences) que puede recibir el real. UserPreferences object.code>UserPreferences del mecanismo de definición (solicitud HTTP, Sesión, etc.). El contenedor inyecta este objeto proxy en el bean userManager, que no sabe que esta referencia UserPreferences es un proxy. En este ejemplo, cuando una instancia de UserManager inicia un método en un objeto UserPreferences con una dependencia inyectada, en realidad llama a un método en el proxy. Luego, el proxy recupera el objeto UserPreferences real de (en este caso) la Session HTTP y delega la llamada al método al objeto UserPreferences real resultante.

Por lo tanto, se requiere la siguiente configuración (correcta y completa) al inyectar beans dentro del alcance de request y session en objetos colaboradores, como se muestra en el siguiente ejemplo:

<bean id="userPreferences" class="com.something.UserPreferences" alcance="sesión">
    <aop:scoped-proxy/>
</frijol>
<bean id="userManager" class="com.something.UserManager">
    <nombre de propiedad="preferencias de usuario" ref="Preferencias de usuario"/>
</bean>

Seleccionar el tipo de proxy a crear

De forma predeterminada, cuando un contenedor Spring crea un proxy para un bean marcado con el elemento <aop:scoped-proxy/>, se crea un proxy de clase basado en CGLIB.

¡Los proxies CGLIB sólo interceptan llamadas a métodos públicos! No llame a métodos no públicos en dicho proxy. No se delegan al objeto de destino real que está dentro del alcance.

Como alternativa, puede configurar el contenedor Spring para crear servidores proxy estándar basados en JDK para dichos beans dentro del alcance especificando false para el valor del atributo proxy-target-class. elemento <aop:scoped-proxy/>. El uso de un proxy basado en la interfaz JDK significa que no necesitará bibliotecas adicionales en la ruta de clase de su aplicación para afectar dicho proxy. Sin embargo, esto también significa que la clase de un bean con ámbito debe implementar al menos una interfaz, y que todos los objetos de comunicación en los que está incrustado el bean con ámbito deben hacer referencia a él a través de una de sus interfaces. El siguiente ejemplo muestra un proxy basado en interfaz:

<!-- DefaultUserPreferences implementa la interfaz UserPreferences -->
<bean id="userPreferences" class="com.stuff.DefaultUserPreferences" alcance="sesión">
    <aop:scoped-proxy proxy-target-class="false"/>
</frijol>
<bean id="userManager" class="com.stuff.UserManager">
    <nombre de propiedad="preferencias de usuario" ref="Preferencias de usuario"/>
</bean>