Diferencia entre revisiones de «Programación de fíos con Java»
Línea 3: | Línea 3: | ||
= Creación de fíos en Java = | = Creación de fíos en Java = | ||
+ | Para executar un programa de Java, iníciase a máquina virtual de Java (JVM) cun único fío que executa o método main() dunha clase. Pero desde este fío podense crear máis fíos, a partir de calquera clase que implemente a interfaz Runnable. Nesta clase hai que programar o método run(), que se executa cando se lanza o fío. | ||
+ | |||
+ | Un fío xestionase con un obxecto da clase Thread. Despois de crealo podese lanzar co seu método start() . Con esto crease o fío e se executa o método start() . | ||
+ | |||
+ | O seguiente programa de exemplo lanza dous fios dun proceso. A funcionalidad dos fíos implementase nunha clase Fío, que implementa a interfaz Runnable. O lanzar o fío execútase o método run() , que escribe un identificador que se pasa no constructor. | ||
+ | |||
+ | :::<syntaxhighlight lang="java" line enclose="div" highlight="" > | ||
+ | package lanzahilos; | ||
+ | class Fio implements Runnable { | ||
+ | private final String nome; | ||
+ | Fio(String nome) { | ||
+ | this.nome = nome; | ||
+ | } | ||
+ | @Override | ||
+ | public void run() { | ||
+ | System.out.printf(“Ola, son o fío: %s.\n”, this.nome); | ||
+ | System.out.printf(“Fío %s terminado.\n”, this.nome); | ||
+ | } | ||
+ | } | ||
+ | public class LanzaFios { | ||
+ | public static void main(String[] args) { | ||
+ | Thread h1 = new Thread(new Fio(“H1”)); | ||
+ | Thread h2 = new Thread(new Fio(“H2”)); | ||
+ | h1.start(); | ||
+ | h2.start(); | ||
+ | System.out.println(“Fio principal terminado.”); | ||
+ | } | ||
+ | } | ||
+ | :::</syntaxhighlight> | ||
= A clase Thread = | = A clase Thread = |
Revisión del 04:59 13 sep 2021
Sumario
Introducción
Comezaremos aprendendo a crear e lanzar novos fios (threads) dun proceso con Java, e facer que diferentes fíos dun mesmo proceso colaboren e se coordinen entre sí para facer unha tarefa. No resto deste capítulo veranse os mecanismos e as clases que proporciona a linguaxe Java para a programación de fíos e a súa sincronización.
Creación de fíos en Java
Para executar un programa de Java, iníciase a máquina virtual de Java (JVM) cun único fío que executa o método main() dunha clase. Pero desde este fío podense crear máis fíos, a partir de calquera clase que implemente a interfaz Runnable. Nesta clase hai que programar o método run(), que se executa cando se lanza o fío.
Un fío xestionase con un obxecto da clase Thread. Despois de crealo podese lanzar co seu método start() . Con esto crease o fío e se executa o método start() .
O seguiente programa de exemplo lanza dous fios dun proceso. A funcionalidad dos fíos implementase nunha clase Fío, que implementa a interfaz Runnable. O lanzar o fío execútase o método run() , que escribe un identificador que se pasa no constructor.
1 package lanzahilos; 2 class Fio implements Runnable { 3 private final String nome; 4 Fio(String nome) { 5 this.nome = nome; 6 } 7 @Override 8 public void run() { 9 System.out.printf(“Ola, son o fío: %s.\n”, this.nome); 10 System.out.printf(“Fío %s terminado.\n”, this.nome); 11 } 12 } 13 public class LanzaFios { 14 public static void main(String[] args) { 15 Thread h1 = new Thread(new Fio(“H1”)); 16 Thread h2 = new Thread(new Fio(“H2”)); 17 h1.start(); 18 h2.start(); 19 System.out.println(“Fio principal terminado.”); 20 } 21 } 22 :::
A clase Thread
Sincronización de fíos
Interrupción de fios
Prioridades
Depuración (debugging) de aplicaciones multifío
Mecanismos de alto nivel para concurrencia
Resumo
- Os fios en Java represéntanse mediante obxectos da clase Thread.
- Podense crear fíos a partir de obxectos de calquera clase que implemente a interfaz Runnable. A propia clase Thread implementa esta interfaz.
- O proceso que realiza un fío impleméntase no método run . Iníciase a execución dun fío co método start, que a súa vez executa o método run.
- O método join suspende a execución do fío actual á espera de que termine a execución do fío para o que se invoca.
- A palabra clave synchronized fai que un fío so poida executar un bloque de código tras obter o bloqueo dun obxecto de bloqueo, que se libera unha vez executado o bloque.
- Se o obxecto de bloqueo está bloqueado, suspéndese a execución do fío á espera de conseguir o bloqueo do obxecto de bloqueo. Deste xeito conseguese que o bloque de código se execute en exclusión mutua con outros fíos.
- Pódese aplicar:
- para métodos non estáticos (nese caso, o obxecto de bloqueo é o obxecto para o que se executa o método, this )
- para métodos estáticos (nese caso, o obxecto de bloqueo é a clase a que pertence ó método)
- para bloques de código cualesquiera (nese caso, especifícase explícitamente o obxecto de bloqueo).
- O interbloqueo ou deadlock prodúce cuando dous ou máis fíos manteñen bloqueados un conxunto de obxectos de bloqueo, e a súa execución está suspendida á espera de conseguir bloquear cada un un novo obxecto de bloqueo, que non poden bloquear porque xa están bloqueados por outros fíos.
- Podese evitar o interbloqueo establecendo unha orde para os obxectos de bloqueo, e todos os hilos se bloquean segundo este orde.
- Si, una vez conseguido el bloqueo sobre un objeto de bloqueo obj en un bloque de código synchronized(obj) , un hilo comprueba que no se dan las condiciones necesarias para continuar con su ejecución, puede ejecutar wait() sobre el objeto de bloqueo con obj.wait() . Con ello, su ejecución queda suspendida, y en espera no activa, mientras que otro objeto no ejecute obj.notifyAll() u obj.notify() . notifyAll() reanuda todos los procesos a la espera en el objeto de bloqueo, mientras que notify() reanuda solo uno de ellos.
- Podense incluir nunha clase os mecanismos de sincronización apropiados para permitir o uso dos seus obxectos por distintos fíos de xeito concurrente. Desa maneira, non é preciso implementar neses fíos ningún mecanismo de sincronización para controlar o acceso concurrente ós objetos da clase. Se dice entonces que la clase es thread-safe o segura para su uso concurrente por parte de varios hilos. O paquete java.util.concurrent inclúe una variedad de clases thread-safe.
- En xeral, deben programarse os fios de xeito que se permita a súa interrupción por parte doutros fíos. Para iso deben xestionar de maneira apropiada a excepción InterruptedException, comprobar cada certo tiempo se foron interrumpidos con isInterrupted() ou interrupted() , e realizar as accións oportunas no seu caso.
- Os entornos de desarrollo adoitan permitir a depuración de aplicacións multifío e proporcionar ferramentas para a detección de interbloqueos.
- o paquete java.util.concurrent proporciona mecanismos de alto nivel para concurrencia.