Tutorial de multithreading em Java com programa e exemplos
Qualquer aplicação pode ter vários processos (instâncias). Cada um desses processos pode ser atribuído como um único thread ou vários threads. Veremos neste tutorial como realizar várias tarefas ao mesmo tempo e também aprenderemos mais sobre threads e sincronização entre threads.
Neste tutorial Multithreading em Java, vamos aprender:
- O que é thread único
- O que é multithreading em Java?
- Ciclo de vida do thread em Java
- Sincronização de thread Java
- Exemplo de multithread Java
O que é thread único?
Um único thread em Java é basicamente uma unidade leve e menor de processamento. Java usa threads usando uma “Classe de Thread”.
Existem dois tipos de encadeamento – encadeamento de usuário e encadeamento de daemon (as threads daemon são usadas quando queremos limpar o aplicativo e são usadas em segundo plano).
Quando um aplicativo é iniciado, o thread do usuário é criado. Poste isso, podemos criar muitos threads de usuário e threads de daemon.
Exemplo de thread único:
package demotest; public class GuruThread { public static void main(String[] args) { System.out.println("Single Thread"); } }
Vantagens do thread único:
- Reduz a sobrecarga no aplicativo à medida que um único thread é executado no sistema
- Além disso, reduz o custo de manutenção do aplicativo.
O que é multithreading em Java?
Multithreading em Java é um processo de execução de dois ou mais threads simultaneamente para utilização máxima da CPU. Os aplicativos multithread executam dois ou mais threads executados simultaneamente. Por isso, também é conhecido como simultaneidade em Java. Cada thread é executado paralelamente um ao outro. Múltiplos threads não alocam área de memória separada, portanto, economizam memória. Além disso, a alternância de contexto entre threads leva menos tempo.
Exemplo de vários tópicos:
package demotest; public class GuruThread1 implements Runnable { public static void main(String[] args) { Thread guruThread1 = new Thread("Guru1"); Thread guruThread2 = new Thread("Guru2"); guruThread1.start(); guruThread2.start(); System.out.println("Thread names are following:"); System.out.println(guruThread1.getName()); System.out.println(guruThread2.getName()); } @Override public void run() { } }
Vantagens do multithread:
- Os usuários não estão bloqueados porque os encadeamentos são independentes e podemos realizar várias operações às vezes
- Como os threads são independentes, os outros threads não serão afetados se um thread atender a uma exceção.
Ciclo de vida do thread em Java
O ciclo de vida de um thread:
Ciclo de vida da thread em Java
Existem vários estágios do ciclo de vida do segmento, conforme mostrado no diagrama acima:
- Novo
- Executável
- Correndo
- Aguardando
- Morto
- Novo: Nesta fase, o thread é criado usando a classe “Thread class”. Permanece neste estado até o programa iniciar o segmento. Também é conhecido como thread nascido.
- Executável: Nesta página, a instância do thread é invocada com um método start. O controle de thread é dado ao escalonador para finalizar a execução. Depende do agendador, se deve executar o encadeamento.
- Em execução: Quando o encadeamento começa a ser executado, o estado é alterado para o estado “executando”. O agendador seleciona um encadeamento do pool de encadeamentos e começa a ser executado no aplicativo.
- Aguardando: Este é o estado em que um thread tem que esperar. Como existem vários threads em execução no aplicativo, há a necessidade de sincronização entre os threads. Assim, um thread tem que esperar, até que o outro thread seja executado. Portanto, esse estado é referido como estado de espera.
- Morto: Este é o estado quando o encadeamento é encerrado. O encadeamento está em estado de execução e, assim que concluir o processamento, estará em "estado morto".
Alguns dos métodos comumente usados para threads são:
Método | Descrição |
---|---|
iniciar() | Este método inicia a execução do encadeamento e a JVM chama o método run() no encadeamento. |
Suspensão(int milissegundos) | Este método faz o thread dormir, portanto, a execução do thread irá pausar por milissegundos fornecidos e depois disso, novamente o thread começa a ser executado. Isso ajuda na sincronização dos threads. |
getNome() | Retorna o nome do encadeamento. |
setPriority(int newpriority) | Altera a prioridade do encadeamento. |
rendimento () | Faz com que o thread atual seja interrompido e outros threads sejam executados. |
Exemplo: Neste exemplo de programa multithreading em Java, vamos criar um thread e explorar métodos integrados disponíveis para threads.
package demotest; public class thread_example1 implements Runnable { @Override public void run() { } public static void main(String[] args) { Thread guruthread1 = new Thread(); guruthread1.start(); try { guruthread1.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } guruthread1.setPriority(1); int gurupriority = guruthread1.getPriority(); System.out.println(gurupriority); System.out.println("Thread Running"); } }
Explicação do código:
- Linha de código 2: Estamos criando uma classe “thread_Example1” que está implementando a interface Runnable (ela deve ser implementada por qualquer classe cujas instâncias devem ser executadas pela thread.)
- Linha de código 4: Ele substitui o método run da interface executável, pois é obrigatório substituir esse método
- Linha de código 6: Aqui definimos o método principal no qual iniciaremos a execução da thread.
- Linha de código 7: Aqui estamos criando um novo nome de thread como “guruthread1” instanciando uma nova classe de thread.
- Linha de código 8: usaremos o método “start” da thread usando a instância “guruthread1”. Aqui o encadeamento começará a ser executado.
- Linha de código 10: Aqui estamos usando o método “sleep” do thread usando a instância “guruthread1”. Portanto, o encadeamento ficará inativo por 1.000 milissegundos.
- Código 9-14: Aqui, colocamos o método sleep no bloco try catch, pois há uma exceção verificada que ocorre, ou seja, exceção interrompida.
- Linha de código 15: Aqui estamos definindo a prioridade do encadeamento para 1 de qualquer prioridade que tenha sido
- Linha de código 16: Aqui estamos obtendo a prioridade do thread usando getPriority()
- Linha de código 17: Aqui estamos imprimindo o valor obtido de getPriority
- Linha de código 18: Aqui estamos escrevendo um texto que o thread está executando.
Ao executar o código acima, você obtém a seguinte saída:
Saída:
5 é a prioridade do Thread, e Thread Running é o texto que é a saída do nosso código.
Sincronização de thread Java
No multithreading, há o comportamento assíncrono dos programas. Se um thread estiver gravando alguns dados e outro estiver lendo dados ao mesmo tempo, poderá criar inconsistência no aplicativo.
Quando há necessidade de acessar os recursos compartilhados por dois ou mais threads, a abordagem de sincronização é utilizada.
Java forneceu métodos sincronizados para implementar o comportamento sincronizado.
Nesta abordagem, uma vez que a thread alcança dentro do bloco sincronizado, nenhuma outra thread pode chamar esse método no mesmo objeto. Todos os encadeamentos precisam esperar até que esse encadeamento termine o bloco sincronizado e saia disso.
Dessa forma, a sincronização ajuda em um aplicativo multithread. Um thread tem que esperar até que outro thread termine sua execução somente então os outros threads são permitidos para execução.
Pode ser escrito na seguinte forma:
Synchronized(object) { //Block of statements to be synchronized }
Exemplo de multithread Java
Neste exemplo Java multithread, pegaremos dois threads e buscaremos os nomes do thread.
Exemplo1:
GuruThread1.java package demotest; public class GuruThread1 implements Runnable{ /** * @param args */ public static void main(String[] args) { Thread guruThread1 = new Thread("Guru1"); Thread guruThread2 = new Thread("Guru2"); guruThread1.start(); guruThread2.start(); System.out.println("Thread names are following:"); System.out.println(guruThread1.getName()); System.out.println(guruThread2.getName()); } @Override public void run() { } }
Explicação do código:
- Linha de código 3: Pegamos uma classe “GuruThread1” que implementa Runnable (deve ser implementada por qualquer classe cujas instâncias devem ser executadas pelo thread.)
- Linha de código 8: Este é o método principal da classe
- Linha de código 9: Aqui estamos instanciando a classe Thread e criando uma instância chamada “guruThread1” e criando um thread.
- Linha de código 10: Aqui estamos instanciando a classe Thread e criando uma instância chamada “guruThread2” e criando um thread.
- Linha de código 11: Estamos iniciando o tópico, ou seja, guruThread1.
- Linha de código 12: Estamos iniciando o tópico, ou seja, guruThread2.
- Linha de código 13: Exibindo o texto como "Os nomes dos threads estão a seguir:"
- Linha de código 14: Obtendo o nome do encadeamento 1 usando o método getName() da classe do encadeamento.
- Linha de código 15: Obtendo o nome do encadeamento 2 usando o método getName() da classe do encadeamento.
Ao executar o código acima, você obtém a seguinte saída:
Saída:
Os nomes dos threads estão sendo emitidos aqui como
- Guru1
- Guru2
Exemplo 2:
Neste exemplo de multithreading em Java, aprenderemos sobre como substituir os métodos run() e start() de uma interface executável e criar dois threads dessa classe e executá-los adequadamente.
Além disso, estamos tendo duas aulas,
- Um que implementará a interface executável e
- Outro que terá o método main e será executado de acordo.
package demotest; public class GuruThread2 { public static void main(String[] args) { // TODO Auto-generated method stub GuruThread3 threadguru1 = new GuruThread3("guru1"); threadguru1.start(); GuruThread3 threadguru2 = new GuruThread3("guru2"); threadguru2.start(); } } class GuruThread3 implements Runnable { Thread guruthread; private String guruname; GuruThread3(String name) { guruname = name; } @Override public void run() { System.out.println("Thread running" + guruname); for (int i = 0; i < 4; i++) { System.out.println(i); System.out.println(guruname); try { Thread.sleep(1000); } catch (InterruptedException e) { System.out.println("Thread has been interrupted"); } } } public void start() { System.out.println("Thread started"); if (guruthread == null) { guruthread = new Thread(this, guruname); guruthread.start(); } } }
Explicação do código:
- Linha de código 2: Aqui estamos pegando uma classe “GuruThread2” que terá o método main nela.
- Linha de código 4: Aqui estamos pegando um método main da classe.
- Linha de código 6-7: Aqui estamos criando uma instância da classe GuruThread3 (que é criada nas linhas abaixo do código) como “threadguru1” e estamos iniciando o thread.
- Linha de código 8-9: Aqui estamos criando outra instância da classe GuruThread3 (que é criada nas linhas abaixo do código) como “threadguru2” e estamos iniciando o thread.
- Linha de código 11: Aqui estamos criando uma classe “GuruThread3” que está implementando a interface executável (ela deve ser implementada por qualquer classe cujas instâncias devem ser executadas pela thread.)
- Linha de código 13-14: estamos pegando duas variáveis de classe das quais uma é do tipo classe thread e outra da classe string.
- Linha de código 15-18: estamos substituindo o construtor GuruThread3, que recebe um argumento como tipo de string (que é o nome da thread) que é atribuído à variável de classe guruname e, portanto, o nome da thread é armazenado.
- Linha de código 20: Aqui estamos substituindo o método run() da interface executável.
- Linha de código 21: Estamos gerando o nome do encadeamento usando a instrução println.
- Linha de código 22-31: Aqui estamos usando um loop for com contador inicializado em 0, e não deve ser menor que 4 (podemos pegar qualquer número, portanto, aqui o loop será executado 4 vezes) e incrementando o contador. Estamos imprimindo o nome do encadeamento e também fazendo o encadeamento dormir por 1.000 milissegundos dentro de um bloco try-catch, pois o método sleep levantou uma exceção verificada.
- Linha de código 33: Aqui estamos substituindo o método start da interface executável.
- Linha de código 35: Estamos gerando o texto "Tópico iniciado".
- Linha de código 36-40: Aqui estamos tomando uma condição if para verificar se a variável de classe guruthread tem valor nela ou não. Se for nulo, estamos criando uma instância usando a classe de thread que recebe o nome como parâmetro (valor para o qual foi atribuído no construtor). Depois disso, o encadeamento é iniciado usando o método start().
Ao executar o código acima, você obtém a seguinte saída:
Saída :
Existem dois tópicos, portanto, recebemos duas vezes a mensagem “Tópico iniciado”.
Obtemos os nomes do encadeamento conforme os produzimos.
Ele entra no loop for onde estamos imprimindo o contador e o nome do thread e o contador começa com 0.
O loop é executado três vezes e entre o encadeamento é suspenso por 1000 milissegundos.
Portanto, primeiro, temos guru1, depois guru2, depois novamente guru2 porque o thread dorme aqui por 1.000 milissegundos e, em seguida, guru1 e novamente guru1, o thread dorme por 1.000 milissegundos, então temos guru2 e depois guru1.
Resumo
Neste tutorial, vimos aplicativos multithread em Java e como usar single e multi thread em Java.
- Explicar o multithreading em Java:no multithreading, os usuários não são bloqueados, pois os threads são independentes e podem realizar várias operações ao mesmo tempo
- Vários estágios do ciclo de vida do segmento são,
- Novo
- Executável
- Correndo
- Aguardando
- Morto
- Também aprendemos sobre sincronização entre threads, o que ajuda o aplicativo a funcionar sem problemas.
- A programação multithread em Java facilita muito mais tarefas de aplicativos.
Java
- Programa Java Hello World
- Funções C++ com exemplos de programas
- Tutorial de coleções de C# com exemplos
- Método Java String indexOf() com substring e exemplos
- Método Java String charAt() com exemplo
- Método Java String compareTo():como usar com exemplos
- Sobrecarga de construtor em Java:o que é e exemplos de programas
- Programa Java para verificar o número primo
- Algoritmo de ordenação por inserção em Java com exemplo de programa
- Classificação de seleção no programa Java com exemplo