Autor
|
Tema: Controlar Timeout mediante un EJB (Leído 6,546 veces)
|
routerico
Desconectado
Mensajes: 34
|
Hola qué tal, estoy intentando controlar un TimeOut mediante un EJB.
Ésta es la clase que tiene mi EJB:
import javax.ejb.Stateless; import javax.ejb.Timeout; @Stateless public class TimeoutRespuesta { @Timeout public int sumador(int resultado){ return resultado; } }
Exporto el EJB mediante un .jar, me creo un proyecto Java y me creo una clase de pruebas:
public class Suma { public static void main(String[] args) { int res = (new TimeoutRespuesta()).sumador(sumar(1,2)); System.out.println(res); } public static int sumar(int a, int b){ return a+b; } }
Aunque es un ejemplo muy sencillo, pero mi idea es controlar el Timeout mediante EJB, si fuese un método más complejo que no deje la aplicación bloqueada, que salte el Timeout del EJB y salga del método.
¿Estaría bien así?.
¿Cuál es el tiempo por defecto del Timeout del EJB?.
Un saludo y muchas gracias.
|
|
|
En línea
|
|
|
|
Usuario Invitado
Desconectado
Mensajes: 625
|
Cuando se utiliza un método timeout anotado con @timeout, la especificación nos dice que el método debe retornar void y sólo recibir como parámetro un objeto javax.ejb.TimerService o en su defecto, no debe recibir parámetros. Para registrar un timeout podemos hacerlo de la siguiente manera (de acuerdo a la tercera forma): createTimer(long duracionInicial, long duracionIntervalo, serializable objetoAsociado);
Así que puedes crearlo así: @Resource TimerService timer; // opcional, aquí inyectamos al objeto TimerService public void doSomething(Objeto obj) { timer.createTimer(15*60*1000, 15*60*1000, obj); }
Lo que se ha hecho es crear un TimerService cuya primera véz de ejecución será un timeout de 15 minutos y el intérvalo entre ejecuciones será de 15 minutos también. Así mismo, hemos asociado al TimerService un objeto que debe implementar Serializable. Bien, ahora el método que se ejecutará luego del registro del TimerService será: @timeout doOtherThing (Timer timer ) { Objeto obj = (Objeto) timer.getInfo(); // hacer otras cosas... }
El método getInfo() retorna el objeto asociado al TimerService. Si al momento de crear el TimerService en lugar de un objeto se declara null, el valor devuelto será null también. Ahora, si estoy en lo correcto, podemos cancelar un TimerService. Por ejemplo si en lugar de un objeto le hemos asociado un nombre, podríamos detenerlo así: public void stop (String timerName ) { for(Object obj : timerService. getTimers()) { if (t.getInfo().equals(timerName)) { t.cancel(); } } }
Pero si no hemos asociado un nombre, podemos detenerlos iterando el conjunto de TimerService: public void stop() { List<Timer> timers = timerService.getTimers(); if (timers != null) { t.cancel(); } } }
Salu2. PD: Usa las etiquetas Geshi para colocar código.
|
|
« Última modificación: 31 Enero 2015, 18:03 pm por Gus Garsaky »
|
En línea
|
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein
|
|
|
routerico
Desconectado
Mensajes: 34
|
Cuando se utiliza un método timeout anotado con @timeout, la especificación nos dice que el método debe retornar void y sólo recibir como parámetro un objeto javax.ejb.TimerService o en su defecto, no debe recibir parámetros. Para registrar un timeout podemos hacerlo de la siguiente manera (de acuerdo a la tercera forma): createTimer(long duracionInicial, long duracionIntervalo, serializable objetoAsociado);
Así que puedes crearlo así: @Resource TimerService timer; // opcional, aquí inyectamos al objeto TimerService public void doSomething(Objeto obj) { timer.createTimer(15*60*1000, 15*60*1000, obj); }
Lo que se ha hecho es crear un TimerService cuya primera véz de ejecución será un timeout de 15 minutos y el intérvalo entre ejecuciones será de 15 minutos también. Así mismo, hemos asociado al TimerService un objeto que debe implementar Serializable. Bien, ahora el método que se ejecutará luego del registro del TimerService será: @timeout doOtherThing (Timer timer ) { Objeto obj = (Objeto) timer.getInfo(); // hacer otras cosas... }
El método getInfo() retorna el objeto asociado al TimerService. Si al momento de crear el TimerService en lugar de un objeto se declara null, el valor devuelto será null también. Ahora, si estoy en lo correcto, podemos cancelar un TimerService. Por ejemplo si en lugar de un objeto le hemos asociado un nombre, podríamos detenerlo así: public void stop (String timerName ) { for(Object obj : timerService. getTimers()) { if (t.getInfo().equals(timerName)) { t.cancel(); } } }
Pero si no hemos asociado un nombre, podemos detenerlos iterando el conjunto de TimerService: public void stop() { List<Timer> timers = timerService.getTimers(); if (timers != null) { t.cancel(); } } }
Salu2. PD: Usa las etiquetas Geshi para colocar código. Muchas gracias por tu ayuda, pero tengo algunas dudas de implementación. Si quisiera controlar el timeout de un método como el anterior, ¿sería algo así? import java.io.Serializable; import javax.annotation.Resource; import javax.ejb.Stateless; import javax.ejb.Timeout; import javax.ejb.Timer; import javax.ejb.TimerService; @Stateless public class TimeoutRespuesta { @Resource TimerService timer; public void doSomething (Object obj ) { } @Timeout void doOtherThing (Timer timer ) { } public int sum(int i){ return i; } }
import timeout.TimeoutRespuesta; public class Suma { public static void main (String[] args ) { int res = (new TimeoutRespuesta()).sum(sumar(1,2)); } public static int sumar(int a, int b){ return a+b; } }
Gracias.
|
|
|
En línea
|
|
|
|
Usuario Invitado
Desconectado
Mensajes: 625
|
No te entiendo. ¿Dices que quieres controlar el timeout dinámicamente? Si es así, puedes controlarlo cancelándolo y volviendo a crearlo. Por ejemplo: public void startOrModifyTimer (long initialExpiration, long interval, String name ){ for (Timer timer : timer. getTimers()) { if (timer.getInfo().equals(name)) { timer.cancel(); } } TimerConfig config = new TimerConfig(); config.setInfo(name); config.setPersistent(false); timerService.createIntervalTimer(initialExpiration, interval, config); }
|
|
|
En línea
|
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein
|
|
|
routerico
Desconectado
Mensajes: 34
|
No te entiendo. ¿Dices que quieres controlar el timeout dinámicamente? Si es así, puedes controlarlo cancelándolo y volviendo a crearlo. Por ejemplo: public void startOrModifyTimer (long initialExpiration, long interval, String name ){ for (Timer timer : timer. getTimers()) { if (timer.getInfo().equals(name)) { timer.cancel(); } } TimerConfig config = new TimerConfig(); config.setInfo(name); config.setPersistent(false); timerService.createIntervalTimer(initialExpiration, interval, config); }
Hola, muchas gracias. Ésto es lo que quiero hacer, pongo un ejemplo: Tengo un proyecto web en Java, ahora bien, tengo un método que hace una consulta a la base de datos, pues imagínate que se queda pillado el método y no devuelve nunca el resultado de la consulta, en vez de quedarse bloqueado, lo que quiero es mediante el timeout de un EJB, controlar y decir si se invoca el método y pasan 20 segundos y no se recibe respuesta que se salga del método y no se quede pillado ahí. Saludos.
|
|
|
En línea
|
|
|
|
Usuario Invitado
Desconectado
Mensajes: 625
|
Yo lo haría así:
1.- Hacer la consulta en un Thread. 2.- Haces la consulta. 3.- Registras el Timer a 20 segundos. 4.- En el método @Timeout cancelas el Thread y por consiguiente la consulta.
|
|
|
En línea
|
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein
|
|
|
routerico
Desconectado
Mensajes: 34
|
Yo lo haría así:
1.- Hacer la consulta en un Thread. 2.- Haces la consulta. 3.- Registras el Timer a 20 segundos. 4.- En el método @Timeout cancelas el Thread y por consiguiente la consulta.
Vale gracias, hice los pasos: 1.- Hacer la consulta en un Thread. public class ConsultaBBDD implements Runnable{ public void run() { String datos = consultarBBDD (); } public String consultarBBDD (){ return "datosBBDD"; } }
2.- Haces la consulta.public class ClasePrincipal { t.start(); return ""; } }
3.- Registras el Timer a 20 segundos. 4.- En el método @Timeout cancelas el Thread y por consiguiente la consulta.import java.io.Serializable; import javax.annotation.Resource; import javax.ejb.Stateless; import javax.ejb.Timeout; import javax.ejb.Timer; import javax.ejb.TimerService; @Stateless public class TimeoutRespuesta { @Resource TimerService timer; public void doSomething (Object obj ) { } @Timeout void doOtherThing (Timer timer ) { t.stop(); } }
¿Sería algo parecido a lo que puse?. Lo que me falta sería la invocación del EJB dentro de la ClasePrincipal, pero es que no sé exactamente cómo sería, ¿qué método suyo tendrían que invocar?. Muchas gracias. Saludos.
|
|
|
En línea
|
|
|
|
Usuario Invitado
Desconectado
Mensajes: 625
|
Exactamente. Me captaste la idea Para invocar al método @Timeout no es necesario invocar nada. En el momento en que registras un TimerService, el método @Timeout se invocará automáticamente cuando expire el tiempo de espera. Otra cosa. Un TimerService tiene que ser usado en un servidor de aplicaciones ya que, éste será inyectado por medio de @Resource por el contenedor del servidor. Además creo que para usar un EJB Timer es necesario especificar si es un Bean local o remoto por medio de sus interfaces. Te dejo un ejemplo: Interface local:import javax.ejb.Local; import javax.ejb.Timer; @Local public interface BeanLocal { public void start (String name ); public void execute (Timer timer ); public void stop (String name ); }
El EJBimport javax.annotation.Resource; import javax.ejb.Singleton; import javax.ejb.Timeout; import javax.ejb.Timer; import javax.ejb.TimerService; @Singleton public class TimerBean implements BeanLocal { @Resource TimerService timer; @Override public void start (String name ) { boolean exists = false; for(Timer temp : timer. getTimers()) { if(temp.getInfo().equals(name)) exists = true; } if(!exists) timer.createTimer(15000, 3000, name); } @Override @Timeout public void execute (Timer timer ) { System. out. println("Método @Timeout invocado: "+timer. getInfo()); } @Override public void stop (String name ) { for(Timer temp : timer. getTimers()) { if(temp.getInfo().equals(name)) temp.cancel(); } } }
Luego, donde uses el EJB Timer lo inyectas también: @Inject private BeanLocal myEJB;
Para registrar un TimerService, simplemente haría: myEJB.start("identificadorTimer");
Y para cancelar un TimerService: myEJB.stop("identificadorTimer");
Nota: Yo le puse @Singleton porque quería que solo exista una instancia del EJB Timer. Funciona igual con EJB Stateless y Stateful. UPDATE De acuerdo a la especificación de EJB 3.1, si va a ser un EJB local solamente, se puede obviar la interface @Local. Solo basta ponerle al EJB @LocalBean para indicar que es un EJB local. @LocalBean @Singleton public class TimerBean {
|
|
« Última modificación: 1 Febrero 2015, 21:32 pm por Gus Garsaky »
|
En línea
|
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein
|
|
|
routerico
Desconectado
Mensajes: 34
|
Exactamente. Me captaste la idea Para invocar al método @Timeout no es necesario invocar nada. En el momento en que registras un TimerService, el método @Timeout se invocará automáticamente cuando expire el tiempo de espera. Otra cosa. Un TimerService tiene que ser usado en un servidor de aplicaciones ya que, éste será inyectado por medio de @Resource por el contenedor del servidor. Además creo que para usar un EJB Timer es necesario especificar si es un Bean local o remoto por medio de sus interfaces. Te dejo un ejemplo: Interface local:import javax.ejb.Local; import javax.ejb.Timer; @Local public interface BeanLocal { public void start (String name ); public void execute (Timer timer ); public void stop (String name ); }
El EJBimport javax.annotation.Resource; import javax.ejb.Singleton; import javax.ejb.Timeout; import javax.ejb.Timer; import javax.ejb.TimerService; @Singleton public class TimerBean implements BeanLocal { @Resource TimerService timer; @Override public void start (String name ) { boolean exists = false; for(Timer temp : timer. getTimers()) { if(temp.getInfo().equals(name)) exists = true; } if(!exists) timer.createTimer(15000, 3000, name); } @Override @Timeout public void execute (Timer timer ) { System. out. println("Método @Timeout invocado: "+timer. getInfo()); } @Override public void stop (String name ) { for(Timer temp : timer. getTimers()) { if(temp.getInfo().equals(name)) temp.cancel(); } } }
Luego, donde uses el EJB Timer lo inyectas también: @Inject private BeanLocal myEJB;
Para registrar un TimerService, simplemente haría: myEJB.start("identificadorTimer");
Y para cancelar un TimerService: myEJB.stop("identificadorTimer");
Nota: Yo le puse @Singleton porque quería que solo exista una instancia del EJB Timer. Funciona igual con EJB Stateless y Stateful. De acuerdo, muchísimas gracias. Mañana lo voy a probar a ver qué sucede. Un saludo!!
|
|
|
En línea
|
|
|
|
routerico
Desconectado
Mensajes: 34
|
@Inject private BeanLocal myEJB;
Para registrar un TimerService, simplemente haría: myEJB.start("identificadorTimer");
Y para cancelar un TimerService: myEJB.stop("identificadorTimer");
Nota: Yo le puse @Singleton porque quería que solo exista una instancia del EJB Timer. Funciona igual con EJB Stateless y Stateful. UPDATE De acuerdo a la especificación de EJB 3.1, si va a ser un EJB local solamente, se puede obviar la interface @Local. Solo basta ponerle al EJB @LocalBean para indicar que es un EJB local. @LocalBean @Singleton public class TimerBean {
Hola compañero, me está dando NullPointerException cuando en el proyecto web. - Me creo el atributo del EJB @Inject private BeanLocal myEJB;
Pero cuando invoco al método start, ahí me da NullPointerException myEJB.start("identificadorTimer");
¿Cómo tendría que instanciar el EJB dentro de una clase del Proyecto Web para que no me de null? Un saludo y muchas gracias de nuevo.
|
|
|
En línea
|
|
|
|
|
Mensajes similares |
|
Asunto |
Iniciado por |
Respuestas |
Vistas |
Último mensaje |
|
|
Controlar circuito mediante LPT
Electrónica
|
nser
|
3
|
6,889
|
6 Octubre 2006, 01:50 am
por skapunky
|
|
|
Controlar TimeOut en conexión
Java
|
imaginawireless
|
4
|
6,088
|
23 Febrero 2012, 04:41 am
por RyogiShiki
|
|
|
Controlar un led mediante un Transistor como interruptor
Electrónica
|
ButterySAM777
|
0
|
2,699
|
30 Diciembre 2013, 15:05 pm
por ButterySAM777
|
|
|
'Hackers' pueden controlar su celular mediante una herramienta ya integrada
Noticias
|
Mister12
|
0
|
2,094
|
2 Agosto 2014, 07:33 am
por Mister12
|
|
|
Read timeout
Java
|
Xedrox
|
2
|
2,413
|
5 Septiembre 2014, 20:45 pm
por Chuidiang
|
|