Pular para o conteúdo principal

Java e Redes: Implementando um ServerSocket

Introdução

Uma dúvida de um aluno da Academia Java me incentivou a dar continuidade ao assunto de programação Java para redes, iniciado em Java e Redes: Consultando um serviço com Socket que mostrava como consultar um serviço do NIST por meio da API de redes da Plataforma Java, este post apresenta a implementação de um java.net.ServerSocket que responde aos clientes com um determinado arquivo.

Exemplo

Para a criação de um ServerSocket que será responsável por escrever os bytes de um determinado arquivo no canal de comunicação com o cliente, definiremos a porta no construtor, conforme segue:
int port = 8080;
ServerSocket server = new ServerSocket(8080);

Apenas a criação do objeto server da classe ServerSocket não permite ainda a comunicação, para tanto é necessário que o ServerSocket passe a aceitar conexões, e isto é feito pelo método accept, este método é bloqueante, ou seja, faz com que o programa fique parado até que o servidor receba uma requisição, e quando isto ocorrer cria um Socket para comunicação; Socket este que tem sua referência armazenada na variável socket, de acordo com a linha apresentada abaixo:
Socket socket = server.accept();
E então, após uma requisição ser recebida, podemos recuperar o fluxo para escrita no socket, por meio do método getOutputStream e montar o fluxo de escrita otimizado com uso de um objeto do tipo BufferedOutputStream , conforme trecho de código abaixo:
OutputStream socketOutput = socket.getOutputStream();
BufferedOutputStream output = new BufferedOutputStream(socketOutput);
Para ler o arquivo determinado criaremos um fluxo de leitura otimizado, conforme segue:
String pathname = "feed.png";
FileInputStream fileInput = new FileInputStream(pathname);
BufferedInputStream input = new BufferedInputStream(fileInput);
Depois de montado o fluxo de entrada (input), que lê um determinado arquivo, e o fluxo de saída (output), basta realizar a leitura da entrada e escrever na saída, enquanto existirem bytes a serem lidos, conforme trecho abaixo:
int data = 0;
while( (data = input.read()) !=-1){
 output.write(data);
}
O método accept está em um laço infinito while(true){} e a forma de interromper a execução do servidor é interromper o processo no console ou na sua IDE preferida, ou ainda determinar a quantidade de vezes que o servidor vai executar o laço.

ServerFileSocket.java

Segue o código completo da classe que implementa o servidor que responde aos clientes com o arquivo determinado.
package br.eti.castro.blog.samples.networking;
// imports ocultos
public class ServerFileSocket {
 //o tratamento de excessoes foi simplificado (throws UnknownHostException, IOException)
 //para simplificar o entendimento do exemplo
 public static void main(String[] args) throws IOException {
  //declarando de objetos para a leitura do arquivo
  String pathname = "feed.png";//pode conter o caminho
  BufferedInputStream input = null;
  FileInputStream fileInput = null;
  
  //declarando de objetos para a escrita no fluxo do socket
  OutputStream socketOutput = null;
  BufferedOutputStream output = null;
  //Criando um servidor que atendera na porta 8080
  int port = 8080;
  ServerSocket server = new ServerSocket(port);
  while(true){
   //O metodo accept faz com que o servidor aguarde uma requisicao
   //Quando isto ocorrer cria um Socket para comunicacao
   Socket socket = server.accept();
   //montando o fluxo de leitura do arquivo otimizado
   fileInput = new FileInputStream(pathname);
   input = new BufferedInputStream(fileInput);
   
   //montando o fluxo de dados para escrita no socket
   socketOutput = socket.getOutputStream();
   output = new BufferedOutputStream(socketOutput);
   int data = 0;
   //lendo o arquivo
   while( (data = input.read()) !=-1){
    //escrevendo o dado lido no socket
    output.write(data);
   }
   //fechando o fluxo de entrada
   input.close();
   //fechando o fluxo de saida
   output.flush();
   output.close();
   //fechando o socket
   socket.close();
   
  }
 }
}
ClientFileSocket.java

Esta classe não está explicada detalhadamente no artigo, porém acredito que o artigo anterior: Java e Redes: Consultando um serviço com Socket, juntamente com este podem esclarecer sobre o funcionamento da classe.

package br.eti.castro.blog.samples.networking;
//imports ocultos
public class ClientFileSocket {
 //o tratamento de excessoes foi simplificado (throws UnknownHostException, IOException)
 //para simplificar o entendimento do exemplo
 public static void main(String[] args) throws UnknownHostException, IOException {
  //declarando de objetos para a leitura do fluxo do socket
  BufferedInputStream input = null;
  InputStream socketInput = null;
  //declarando de objetos para a escrita do arquivo
  String pathname = "novo.png";
  FileOutputStream fileOutput = null;
  BufferedOutputStream output = null;
  //criando o socket para o localhost na porta 8080
  String host = "localhost";
  int port = 8080;
  Socket socket = new Socket(host, port);
  //recuperando o fluxo de dados
  socketInput = socket.getInputStream();
  //montando o fluxo de leitura do socket otimizado 
  input = new BufferedInputStream(socketInput);
  //montando o fluxo de escrita do arquivo otimizado 
  fileOutput = new FileOutputStream(pathname);
  output = new BufferedOutputStream(fileOutput);
  //lendo o socket
  int i = 0;
  while( (i = input.read()) !=-1){
   //escrevendo o arquivo
   output.write(i);
  }
  //fechando o fluxo de entrada
  input.close();
  //fechando o fluxo de saida
  output.flush();
  output.close();
  
  //fechando o socket
  socket.close();
 }
}
O exemplo está preparado para rodar ambas as classes na mesma máquina, pois a classe ClientFileSocket está configurada para procurar o ServerSocket na mesma máquina (host = "localhost"), porém é interessante rodar cada uma das classes em uma máquina diferente, bastando para tanto alterar o valor da variável host.

Outro ponto importante, é com relação a porta utilizada, caso a porta 8080 já esteja sendo utilizada, basta alterar a variável port, que deve ter sempre o mesmo valor na classe ServerFileSocket e na classe ClientFileSocket.
Além disso é possível alterar o arquivo enviado, alterando o valor da variável pathname na classe ServerFileSocket e também o nome em que a classe ClientFileSocket salva o arquivo alterando variável de mesmo nome.

Teste se possível em máquinas diferentes, alterando as variáveis indicadas, tente ainda alterar o arquivo enviado.

Até mais!
Marcelo de Castro
http://twitter.com/mcastroinfo
http://www.globalcode.com.br/instrutores/MarceloCastro
Outros posts: http://blog.globalcode.com.br/search/label/MarceloCastro

Comentários

Postagens mais visitadas deste blog

10 reasons why we love JSF

1. One-slide technology: it's so simple that I can explain basic JSF with one slide. 2. Easy to extend: components, listeners, render kit, Events, Controller, etc. 3. Real-world adoption: JBoss, Exadel, Oracle, IBM, ... 4. Architecture model: you can choose between more than 100 different architecture. 5. Open-mind community: using JSF you are going to meet very interesting people. 6. We are using JSF the last 5 years and we found very good market for JSF in Brazil 7. Progress: look to JSf 1.1 to JSF 1.2, JSF 1.2 to JSF 2.0. People are working really hard! 8. Many professionals now available 9. It's a standard. It's JCP. Before complain, report and help! 10. Ed Burns, spec leader, is an old Globalcode community friend! EXTRA: My wife is specialist in JSF. She's my F1 for JSF :) Nice job JSF community! -Vinicius Senger

Alguma certificação Java nos seus planos em 2010 ?

A discussão sobre certificações é super polêmica, alguns acham muito importante, outros acham totalmente insignificante. Eu particularmente acho que é uma meta de aprendizado, que nos força a aprender bem detalhadamente uma tecnologia. É claro que é "burlável" quero dizer, todo mundo conhece alguém que é certificado e "não sabe nada". Mas, evidentemente quando você se prepara, estuda, vai aprender muitas coisas novas e solidificar ainda mais seu conhecimento. Eu sempre fui muito "orientada" por deadlines... estas metas nos obrigam a encaixar algumas horas de estudo nesta vida tão corrida. E o inicio do ano é muito bom para estas iniciativas, porque estamos um pouco mais descansados, com aquele espírito de inicio de ano. Imagine chegar em março e já ter cumprido uma das metas de final de ano, e estar com a certificação "tirada" ? Não acho que precisa ser um caçador de certificações, e tentar tirar todas as certificações possíveis, mas acho ...

O que é Lógica de programação?

Este é o segundo de uma série de posts voltados aos leitores do blog que estão dando início à carreira de desenvolvimento de software. O assunto de hoje é a lógica de programação. Para ler antes: Entendendo como funciona a programação de computadores: linguagens de programação, lógica, banco de dados A lógica de programação é um pré-requisito para quem quer se tornar um desenvolvedor de software, independente da linguagem de programação que se pretende utilizar. Mas o que é de fato a Lógica de Programação e como saber se eu tenho esse pré-requisito? A lógica de programação nada mais é do que a organização coerente das instruções do programa para que seu objetivo seja alcançado. Para criar essa organização, instruções simples do programa, como mudar o valor de uma variável ou desenhar uma imagem na tela do computador, são interconectadas a estruturas lógicas que guiam o fluxo da execução do programa. Isso é muito próximo ao que usamos em nosso cotidiano para realizar atividad...

JSF 2 - Composite Components, você não precisa mais ser um ninja

Estamos em uma nova era da computação, os dados não estão mais localizados em um banco dentro de sua empresa, vivemos a explosão de redes sociais, informações são geradas a todo instante, e se torna essencial que sua aplicação conheça os serviços disponíveis na web e consumam suas APIs geralmente disponíveis por serviços REST. Legal, mas como ficam meus aplicativos Java EE neste novo cenário? Para quem vem acompanhando a evolução da plataforma, é notório que todo esforço vem sendo utilizado para aumentar a produtividade e a integração com novos serviços. Basicamente duas especificações surgem com muita força para atender este cenário, a JSR - 314 (JSF-2) e JSR - 311 (JAX-RS), neste post exploraremos a JSR-314 (JSF2) e sua nova forma de criar Composite Components. Uma das grandes queixas dos desenvolvedores JSF era a complexidade em criar composite components, era necessário um vasto conhecimento sobre o ciclo de vida de uma aplicação JSF. Agora, você não precisa ser mais um “ninja” em ...

JavaMail: Enviando mensagem HTML com anexos

Introdução Depois do post "JavaMail: Enviando e-mail com Java" , que apresentava como enviar um e-mail com Java, resolvi complementar a assunto apresentando como enviar uma mensagem formatada, em HTML , e também como realizar o envio de anexos. Bibliotecas Além da biblioteca JavaMail, veja mais no post anterior , é necessário incluir o JavaBeans Activation Framework (JAF), apenas se a versão utilizada for anterior ao JSE 6.0 , que já tem o JAF incluso. O JAF está disponível em http://www.oracle.com/technetwork/java/javase/downloads/index-135046.html , e neste download encontramos, alguns exemplos na pasta demo , documentação, incluindo javadocs, na pasta docs e a biblioteca activation.jar , que deve ser acrescentada no classpath da aplicação para versões anteriores ao JSE 6.0. Exemplo Primeiramente devemos realizar a configuração da javax.mail.Session e da javax.mail.internet.MimeMessage , estes passos podem ser vistos no post anterior . Agora vamos montar um...

Sistema interativo de TV Digital com Ginga-J

No início de 2009, os estudantes de Sistema de Informação do Centro Universitário de Votuporanga ( UNIFEV ), Caio César Pereira de Souza e Rodrigo Gonçalves Constantino me apresentaram uma proposta para que eu fosse co-orientador junto ao professor orientador Djalma Domingos da Silva , em seu Trabalho de conclusão de curso (TCC) com tema TV Digital. A base que motivou o assunto, foi a palestra apresentada por Maurício Leal na I Conferência Java Noroeste sobre o tema TV Digital, realizada em 2006 em Votuporanga-SP. Ficamos muito entusiasmados com a possibilidade de interatividade na TV Digital, e a grande quantidade de possibilidades de desenvolvimento de aplicativos nesta área. Acompanhamos de perto as notícias na imprensa e todo o esforço e iniciativas realizadas pelo Fórum do Sistema Brasileiro de TV Digital Terrestre (SBTVD) , que organizou e produziu especificações ABNT, normatizando o sistema de TV Digital Terrestre. O foco do TCC foi realizar o desenvolvimento de uma pequena ...