Noté que hay diferentes ámbitos de frijol como:
@RequestScoped
@ViewScoped
@FlowScoped
@SessionScoped
@ApplicationScoped
¿Cuál es el propósito de cada uno? ¿Cómo elijo un alcance adecuado para mi bean?
jsf
jsf-2
scope
managed-bean
Valter Silva
fuente
fuente
Respuestas:
Introducción
Representa el alcance (la vida útil) del bean. Esto es más fácil de entender si está familiarizado con el funcionamiento "oculto" de una aplicación web de servlet básica: ¿Cómo funcionan los servlets? Instanciación, sesiones, variables compartidas y multihilo .
@Request/View/Flow/Session/ApplicationScoped
Un
@RequestScoped
bean vive tanto como un solo ciclo de solicitud-respuesta de HTTP (tenga en cuenta que una solicitud de Ajax también cuenta como una sola solicitud de HTTP). Un@ViewScoped
bean vive mientras interactúa con la misma vista JSF mediante devoluciones que llaman métodos de acción que regresannull
/void
sin navegación / redirección. Un@FlowScoped
bean vive mientras navega por la colección especificada de vistas registradas en el archivo de configuración de flujo. Un@SessionScoped
bean vive tanto como la sesión HTTP establecida. Un@ApplicationScoped
bean vive mientras se ejecute la aplicación web. Tenga en cuenta que el CDI@Model
es básicamente un estereotipo@Named @RequestScoped
, por lo que se aplican las mismas reglas.El alcance a elegir depende únicamente de los datos (el estado) que el bean contiene y representa. Úselo
@RequestScoped
para formularios / presentaciones simples y no ajax. Úselo@ViewScoped
para vistas dinámicas habilitadas con ajax (validación, renderizado, diálogos, etc.). Se utiliza@FlowScoped
para el patrón de "asistente" ("cuestionario") de recopilación de datos de entrada distribuidos en varias páginas. Utilícelo@SessionScoped
para datos específicos del cliente, como el usuario conectado y las preferencias del usuario (idioma, etc.). Úselo@ApplicationScoped
para datos / constantes de toda la aplicación, como listas desplegables que son iguales para todos, o beans administrados sin ninguna variable de instancia y que solo tienen métodos.Abusar de un
@ApplicationScoped
bean para datos de alcance de sesión / vista / solicitud haría que se compartiera entre todos los usuarios, de modo que cualquier otra persona pueda ver los datos de los demás, lo cual es simplemente incorrecto. Abusar de un@SessionScoped
bean para ver / solicitar datos con alcance haría que se compartiera entre todas las pestañas / ventanas en una sola sesión del navegador, por lo que el usuario final puede experimentar inconsistencias al interactuar con cada vista después de cambiar entre pestañas, lo que es malo para la experiencia del usuario. Abusar de un@RequestScoped
bean para ver los datos de ámbito hará que los datos de ámbito de vista se reinicialicen a los valores predeterminados en cada devolución de datos (ajax), lo que puede causar formas que no funcionen ( ver también los puntos 4 y 5 aquí ). Abusar de un@ViewScoped
bean para datos de ámbito de solicitud, sesión o aplicación, y abusar de un@SessionScoped
Bean para los datos de ámbito de aplicación no afecta al cliente, pero ocupa innecesariamente la memoria del servidor y es ineficiente.Tenga en cuenta que el alcance no debe elegirse en función de las implicaciones de rendimiento, a menos que realmente tenga una huella de memoria baja y desee pasar completamente sin estado; necesitaría usar exclusivamente
@RequestScoped
beans y violín con parámetros de solicitud para mantener el estado del cliente. También tenga en cuenta que cuando tiene una sola página JSF con datos de alcance diferente, entonces es perfectamente válido ponerlos en beans de respaldo separados en un ámbito que coincida con el alcance de los datos. Los beans solo pueden acceder entre sí a través@ManagedProperty
de los beans administrados por JSF o@Inject
en el caso de los beans administrados por CDI.Ver también:
@CustomScoped/NoneScoped/Dependent
No se menciona en su pregunta, pero JSF (heredado) también es compatible
@CustomScoped
y@NoneScoped
, que rara vez se usa en el mundo real. El@CustomScoped
debe referir a un encargo deMap<K, Bean>
la aplicación de alguna alcance más amplio, que ha anuladoMap#put()
y / oMap#get()
fin de tener un control de grano más fino sobre la creación de frijol y / o destruir.El JSF
@NoneScoped
y el CDI@Dependent
básicamente viven tanto tiempo como una única evaluación EL en el bean. Imagine un formulario de inicio de sesión con dos campos de entrada que hacen referencia a una propiedad de bean y un botón de comando que hace referencia a una acción de bean, por lo tanto, con un total de tres expresiones EL, se crearán efectivamente tres instancias. Uno con el nombre de usuario establecido, otro con la contraseña establecida y otro en el que se invoca la acción. Normalmente desea usar este alcance solo en beans que deberían vivir tanto tiempo como el bean donde se está inyectando. Así que si una@NoneScoped
o@Dependent
se inyecta en una@SessionScoped
, entonces se va a vivir siempre y cuando el@SessionScoped
café en grano.Ver también:
Alcance del flash
Como último, JSF también es compatible con el alcance de flash. Está respaldado por una cookie de vida corta que está asociada con una entrada de datos en el alcance de la sesión. Antes de la redirección, se establecerá una cookie en la respuesta HTTP con un valor que está asociado únicamente con la entrada de datos en el alcance de la sesión. Después de la redirección, se comprobará la presencia de la cookie de alcance flash y la entrada de datos asociada con la cookie se eliminará del alcance de la sesión y se colocará en el alcance de la solicitud de la solicitud redirigida. Finalmente, la cookie se eliminará de la respuesta HTTP. De esta forma, la solicitud redirigida tiene acceso a los datos de ámbito de solicitud que se prepararon en la solicitud inicial.
En realidad, esto no está disponible como un ámbito de bean administrado, es decir, no existe tal cosa
@FlashScoped
. El alcance flash solo está disponible como un mapa a travésExternalContext#getFlash()
de beans gestionados y#{flash}
en EL.Ver también:
fuente
@FlowScoped
(no es necesario iniciarlo / detenerlo manualmente).ViewAccesscoped
yWindowScoped
ViewScoped
bean en MyFaces 2.2. Actualmente estoy enfrentando un problema conViewScoped
bean y Ajax, que he publicado aquí . En MyFaces JIRA, también hay una discusión sobre este tema.@RequestScoped
@SessionScoped
@ApplicationScoped
@ConversationScoped
¿por qué los ámbitos que describe son diferentes?Desde JSF 2.3, todos los ámbitos de bean definidos en el paquete del
javax.faces.bean
paquete han quedado en desuso para alinear los ámbitos con CDI. Además, solo son aplicables si su bean usa@ManagedBean
anotaciones. Si está utilizando versiones JSF inferiores a la 2.3, consulte la respuesta heredada al final.Desde JSF 2.3, aquí hay ámbitos que se pueden usar en JSF Backing Beans:
1
@javax.enterprise.context.ApplicationScoped
.: El ámbito de aplicación persiste durante toda la duración de la aplicación web. Ese alcance se comparte entre todas las solicitudes y todas las sesiones. Esto es útil cuando tiene datos para toda la aplicación.2
@javax.enterprise.context.SessionScoped
.: El alcance de la sesión persiste desde el momento en que se establece una sesión hasta su finalización. El contexto de la sesión se comparte entre todas las solicitudes que se producen en la misma sesión HTTP. Esto es útil cuando no va a guardar datos para un cliente específico para una sesión en particular.3
@javax.enterprise.context.ConversationScoped
.: El alcance de la conversación persiste como registro mientras vive el bean. El alcance proporciona 2 métodos:Conversation.begin()
yConversation.end()
. Estos métodos deben llamarse explícitamente, ya sea para iniciar o finalizar la vida de un bean.4
@javax.enterprise.context.RequestScoped
.: El alcance de la solicitud es de corta duración. Comienza cuando se envía una solicitud HTTP y finaliza después de que la respuesta se devuelve al cliente. Si coloca un bean administrado en el alcance de la solicitud, se crea una nueva instancia con cada solicitud. Vale la pena considerar el alcance de la solicitud si le preocupa el costo del almacenamiento del alcance de la sesión.5
@javax.faces.flow.FlowScoped
.: El alcance del flujo persiste mientras viva el flujo. Un flujo puede definirse como un conjunto de páginas (o vistas) que definen una unidad de trabajo. El ámbito de flujo estado ha estado activo mientras el usuario navegue en el flujo.6
@javax.faces.view.ViewScoped
.: Un bean en el alcance de la vista persiste mientras se vuelve a mostrar la misma página JSF. Tan pronto como el usuario navega a una página diferente, el bean queda fuera de alcance.La siguiente respuesta heredada se aplica a la versión JSF anterior a la 2.3
Fuente: Core Java Server Faces 3rd Edition por David Geary y Cay Horstmann [Página no. 51 - 54]
fuente
invalidate()
método o método no válido?FacesContext.getCurrentInstance().getExternalContext().invalidateSession();
lo que quiere decir es que te invoquen en tu "bean de cierre de sesión".