Lo último que necesitamos saber con los hilos es el cómo interactúan entre ellos, cómo es que se comunican cuando tienen o liberan el seguro de un objeto en particular. La clase Object cuenta con 3 métodos que ayudan a los hilos a comunicarse entre ellos, los métodos son: wait(), notify() y notifyAll(). Algo importante a tomar en cuenta con los hilos es:
wait(), notify() y notifyAll() deben ser llamados desde dentro de un contexto sincronizado. Un hilo no puede invocar wait() o notify() en un objeto a menos de que posea el seguro de dicho objeto.
Más info tras el salto...
De la misma manera en que todo objeto posee un seguro, cada objeto puede tener una lista de hilos que están esperando una notificación del objeto. Un hilo se posiciona en esta lista de espera por medio de una llamada al método wait() del objeto al que está esperando. Desde ese momento, el hilo no ejecuta ninguna instrucción hasta que el método notify() es llamado por el objeto. Si existen varios hilos en la cola de espera del objeto, solo uno podrá ser escogido (sin un orden garantizado) por el programador de hilos para acceder a dicho objeto y obtener su seguro. Un ejemplo en código sería el siguiente:
public class ThreadA {
public ThreadA() {
}
public static void main(String[] args)
{
ThreadB b = new ThreadB();
b.start();
synchronized(b){
try {
System.out.println("Esperando a que B se complete...");
b.wait();
} catch (InterruptedException ex) { }
System.out.println("Total:" +b.total);
}
}
}
class ThreadB extends Thread{
long total;
public void run(){
synchronized(this)
{
for(long i=0; i<1000000000;i++)
total+=i;
notify();
}
}
}
Si lo ejecutamos obtenemos algo así...
Esperando a que B se complete...
Total:499999999500000000
En caso de que omitiéramos la llamada a
wait(), el resultado sería el siguiente:
Total:0
Un punto a tomar en cuenta es que para poder llamar al método wait(), se tuvo que sincronizar el bloque de código y obtener el seguro de b, de lo contrario se obtiene una excepción de tipo java.lang.IllegalMonitorStateException.
Como se ha mencionado anteriormente, un solo objeto puede tener una lista de espera de varios hilos, en dichos casos es recomendable utilizar notifyAll() en vez de notify(), ya que notifyAll() notificará que está a punto de liberar el seguro del objeto a todos los hilos de la lista de espera.
Repasando, los métodos definidos por la clase Object son
wait(), notify() y
notifyAll(). Los métodos definidos por la clase Thread son
start(), yield(), sleep() y join(), sleep() y yield() son
estáticos mientras que start() y join()
no lo son. El método definido en la interfaz Runnable es
run().
Hemos terminado con los Hilos en Java. Si tienes alguna duda deja tu comentario.
Más sobre programación en Java
aquí.
muy bueno, un abrazo desde Argentina!
ResponderBorraresta muy bien el programa pero le sugieron que suban mas para analixarlo gracias.
ResponderBorraroye muchas gracias por este aporte el ejemplo es muy claro, pero creo que lo podrias ampliar un poco mas gracias
ResponderBorrarMe vino muy bien todo esto. Un saludo desde España
ResponderBorrarMuy útil. Me los he ventilado en media hora haciendo pruebas de por medio. Voy a hacer como neo y decir "Ya se hilos" aunque se pasa de pretencioso por mi parte :D
ResponderBorrarGracias! desde México.
ResponderBorrarasi es como se deben de explicar las cosas, sencillas y con codigo funcional, para que uno pueda hacer sus pruebas, gracias!
Disculpa y para pasarle parametros a un hilo ?
ResponderBorrarse usaria la misma logica en la que acceden a una variable (como en el ejemplo del banco) y sincronizarlos o darles tiempos en los que van a estar accediendo a las variables ?
o existe algun otro modo para pasarle directamente un parametro como lo haces con el setName() e irle seteando nombres o como ?
Simplemente Excelente!!! Thank so much....
ResponderBorrarEste comentario ha sido eliminado por el autor.
ResponderBorrarHe programado a mi forma el ejemplo de la cuenta bancaria y varios hilos con el nombre de la persona que va a retirar dinero. Todos podían sacar gracias a synchornized, pero el orden de ejecución era aleatorio. Entonces le introduje el uso de wait y response, a saber: Los hilos llevan asociado un mismo objeto, implementación de Runnable, que es con el que sincronizo dentro del método run. Ahora mismo, dentro de mi main, el código viene a ser algo así:
ResponderBorrarHilo cuenta = new Hilo(); // hilo implementa a la interfaz runnable
Thread juanito = new Thread(cuenta, "Juanito");
Thread pepito = new Thread(cuenta, "Pepito");
Thread lourdes = new Thread(cuenta, "Lourdes");
Thread jesus = new Thread(cuenta, "Jesús");
Thread edelmiro = new Thread(cuenta, "Edelmiro");
juanito.start();
synchronized(cuenta) {
try
cuenta.wait();
}
catch (InterruptedException e) { }
pepito.start();
}
synchronized(cuenta) {
try
{
cuenta.wait();
}
catch (InterruptedException e) { }
lourdes.start();
}
synchronized(cuenta) {
try {
cuenta.wait();
}
catch (InterruptedException e) { }
jesus.start();
}
synchronized(cuenta) {
try {
cuenta.wait();
}
catch (InterruptedException e) { }
edelmiro.start();
}
Esto me da como resultado que tengo controlado el orden de ejecución de dichos hilos, que aunque no es muy práctico programarlo por hilos, me valeparala comprensión de los conceptos. Pero así no fue como empecé a trabajar. En un principio, esperé a los propios hilos en lugar del objeto cuenta:
synchronized(juanito) {
try {
juanito.wait();
}
catch (InterruptedException e) { }
pepito.start();
}
synchronized(pepito) {
try
{
pepito.wait();
}
catch (InterruptedException e) { }
lourdes.start();
}
Es decir, Pepito espera a Juanito, Lourdes espera a Pepito, y así sucesivamente. Me pareció intuitivo, pero me resulta curioso que funcione cuando no existe una correspondencia entre el hilo y el objeto del que obtengo el seguro, que siempre es el mismo. Y más curioso aún, que no necesite hacer un Notify en este caso, funciona igual sin él. Imagino que al enviar la orden wait a un hilo, cuando dicho hilo muere, la espera se ha terminado. Aclárame esto, por favor. Y enhorabuena por el monográfico.
Estupendo aporte, pero... sería posible que postearas una sexta parte ? algo como un ejercicio práctico que abarcara todo lo de hilos y los conceptos vistos en los 5 post anteriores, de verdad muchas gracias por tus aportes, personas como tú hacen grande la Internet :D
ResponderBorrarHola amigo, estupendos tutoriales de hilos, ¿sería posible que hicieras un ejemplo en donde trabajaras con un hilo padre e hilos hijo? no sé muy bien como hacerlo, espero tu respuesta, muy bueno tu blog!
ResponderBorrarJaja, estoy en las mismas, necesito ayuda con algún ejemplo sobre el manejo de un hilo padre y sus hilos hijos, deberías hacer una continuación de este post, a propósito, gracias por tus aportes.
ResponderBorrargracias gracias gracias mil GRACIAS!, Genial mi compa, aprendi hilos sin tantos compliques, realmente me han servido mucho tus tutoriales, he leido los comentarios de arriba y creo que también necesito aclaraciones acerca de cómo carajos manejar los hilos hijos respecto a su hilo padre.
ResponderBorrarBuenos tutoriales, pero podrías subir algo relacionado con el multithreading ? (creo que se llama así) igual he aprendido mucho y me he sacado muchas dudas de encima con tus posts, gracias hermano!
ResponderBorrarHola Monillo007, en verdad tus tutoriales son como caídos del cielo <3 aunque tengo dudas acerca de cómo agrupar y trabajar los hilos, deberías hacer un 6to tutorial, te estaría muy agradecida ;) gracias por todo
ResponderBorrarHola Monillo, estoy feliz con tus tutoriales, increibles! me ha despejado muchas dudas sobre hilos, ¿será que puedes hacer uno que hable directamente sobre un hilo padre y varios de sus hilos hijos? ya varios visitantes han pedido lo mismo, voy a estar pendiente de tu blog, gracias por todo :3
ResponderBorrarExcelente tutorial :y: Muchas Gracias!
ResponderBorrarGenial amigo, me ha ayudado mucho. Saludos desde Venezuela.
ResponderBorrar