Pular para o conteúdo principal

O novo try no Java 7, por uma linguagem mais simples

O Java 7, lançado em Julho, além de reviver mudanças no Java também trouxe novas características afim de tornar a linguagem um pouco mais simples e menos verbosa, ou seja, as instruções podem ser escritas com um volume menor de código.

Nós da Globalcode adoramos novidades, ainda mais relacionada ao Java, usamos e incentivamos nossos alunos e clientes a experimentarem o Java 7. Nesse post vou comentar sobre as características do novo bloco try do Java.

Para começar, primeiro, vamos analisar uma demonstração simples de um programa Java que lê o conteúdo de arquivo texto linha a linha:
import java.io.*;

public class DemoTryAntigo {

  public static void main(String[] args) {
    FileReader in = null;
    BufferedReader buff = null;
        
    try {
      in = new FileReader("/home/yaw/teste.txt"); //caminho do arquivo
      buff = new BufferedReader(in, 1024);
            
      StringBuilder builder = new StringBuilder();
      String s = null;
      while ((s = buff.readLine()) != null) {
        builder.append(s).append("\n");
      }      
      System.out.println("Conteudo do arquivo:\n\n"+builder);
    } catch(IOException iox) {
      System.out.println("Falha ao ler arquivo: "+iox.getMessage());
    } finally {
      if (buff != null) {
        try {
          buff.close();
        } catch(IOException bufx) {
          System.out.println("Falha ao fechar buffer: "+bufx.getMessage());
        }
      }
      if (in != null) {
        try {
          in.close();
        } catch(IOException inx) {
          System.out.println("Falha ao fechar arquivo: "+inx.getMessage());
        }
      }
    }
  }

}

Nenhuma complexidade nesse exemplo, o ponto negativo é o excesso de código (linhas 22 a 35) no bloco finally necessário para liberar os recursos de I/O utilizados na leitura do arquivo. O programa escrito dessa forma compila e funciona no Java 7, mas a partir do Java 7 podemos tirar proveito do try-with-resources: um recurso é um objeto que deve ser encerrado quando o bloco try for concluído. Veja como ficaria o mesmo programa utilizando essa funcionalidade do Java 7:
import java.io.*;

public class DemoNovoTry {

  public static void main(String[] args) {
    try (FileReader in = new FileReader("/home/yaw/teste.txt");
         BufferedReader buff = new BufferedReader(in, 1024)) {

      StringBuilder builder = new StringBuilder();
      String s = null;
      while ((s = buff.readLine()) != null) {
        builder.append(s).append("\n");
      }
      System.out.println("Conteudo do arquivo:\n\n"+builder);
    } catch(IOException iox) {
      System.out.println("Falha ao ler arquivo: "+iox.getMessage());
    }
  }

}

O código ficou bem mais limpo! Uma regra importante sobre o try-with-resources é que o objeto (recurso) definido no try deve implementar ou interface java.lang.AutoCloseable (nova) ou java.io.Closeable, qualquer outro conteúdo o compilador não irá aceitar. No exemplo demonstrado ambas FileReader e BufferedReader implementam AutoCloseable, essa foi uma das mudanças na API de I/O ocorridas no Java7.

Na verdade, nos bastidores, o Java faz o trabalho “sujo” de implementar o bloco de finalização para garantir que os recursos sejam encerrados, omitindo o código aos programadores. O Java transforma cada recurso declarado no try em uma variável final e, define um bloco try-finally aninhado a declaração desse recurso. Outra característica importante é que em casos com mais de um recurso definido no try (o nosso exemplo cai nessa situação) o Java encerra os recursos na ordem inversa em que são definidos, garantindo que todos os recursos serão devidamente encerrados. A seguir o código que o compilador do Java 7 utiliza para gerar o byte code:
import java.io.*;

public class DemoNovoTry {

  public static void main(String[] args) {
    try {
      final FileReader in = new FileReader("/home/yaw/teste.txt");
      /*synthetic*/ Throwable primaryException0$ = null;
      try {
        final BufferedReader buff = new BufferedReader(in, 1024);
        /*synthetic*/ Throwable primaryException1$ = null;
        try {
          StringBuilder builder = new StringBuilder();
          String s = null;
          while ((s = buff.readLine()) != null) {
            builder.append(s).append("\n");
          }
          System.out.println("Conteudo do arquivo:\n\n" + builder);
        } catch (/*synthetic*/ final Throwable t$) {
          primaryException1$ = t$;
          throw t$;
        } finally {
          if (buff != null)
            if (primaryException1$ != null) 
              try {
                buff.close();
              } catch (Throwable x2) {
                primaryException1$.addSuppressed(x2);
              }
            else
              buff.close();
        }
      } catch (/*synthetic*/ final Throwable t$) {
        primaryException0$ = t$;
        throw t$;
      } finally {
        if (in != null) 
          if (primaryException0$ != null)
            try {
              in.close();
            } catch (Throwable x2) {
              primaryException0$.addSuppressed(x2);
            }
          else
            in.close();
      }
    } catch (IOException iox) {
      System.out.println("Falha ao ler arquivo: " + iox.getMessage());
    }
  }

}

Analisando cuidadosamente o código acima, é possível identificar as mudanças realizadas na estrutura de exceções do Java 7: o método addSuppressed na classe Throwable, que permite que uma exceção secundária seja vinculada a outra exceção primária. Veja o trecho com o finally do BufferedReader (linhas 22 a 32), ao fechar o buffer o programa verifica se existe alguma exceção gerada durante a leitura do buffer, dessa forma se ocorrer alguma exceção durante o close do buffer essa será vinculada a exceção primária lançada do try.

Outro método novo em Throwable é o getSuppressed, que retorna um array com exceções vinculadas a exceção principal, um novo construtor da classe também foi criado no Java 7.

A API do JDBC, versão 4, foi ajustada no Java 7 para respeitar o AutoCloseable. A seguir um programa Java abre uma conexão com uma base de dados MySQL e executa uma instrução SQL.
(Importante: ao testar garanta que o driver esteja configurado; o caminho, usuário e senha do banco de dados estejam corretos; e a existência da tabela clientes)

import java.sql.*;

public class DemoTryJDBC {

  public static void main(String[] args) throws SQLException, Exception {
     Class.forName("com.mysql.jdbc.Driver");
     String query = "select nome, cpf from clientes";
     String urlDB = "jdbc:mysql://localhost:3306/testdb";

     try (Connection con = DriverManager.getConnection(urlDB,"user","user");
       Statement stmt = con.createStatement();
       ResultSet rs = stmt.executeQuery(query)) {

       while (rs.next()) {
         String nome = rs.getString("nome");
         String cpf = rs.getString("cpf");
         
         System.out.printf("Nome:%s\t Cpf:%s %n", nome, cpf);
       }
     }
  }
    
}

Outra novidade do Java 7, ainda no escopo do bloco try, é a possibilidade de definir o mesmo tratamento de erro para diversos tipos de exceções utilizando o multicatch. O próximo o utilizo a API de reflexão do Java para criar uma String, veja que o catch determina o mesmo tratamento para as exceções: ClassNotFoundException, InstantiationException ou IllegalAccessException.
public class DemoMulticatch {
    
  public static void main(String[] args) {
    Object o = null;
    try {
      Class clazz = Class.forName("java.lang.String");
      o = clazz.newInstance();
      System.out.println(o.getClass());
    } catch (ClassNotFoundException | InstantiationException | IllegalAccessException ex) {
      System.err.println("Erro na criação do objeto: "+ex.getMessage());
    }
  }

}

Sobre o multicatch, uma regra importante é que o compilador não aceita a declaração utilizando exceções polimórficas, com relacionamento via herança. Por exemplo, não é permitido utilizar o tipo Exception dentro dessa instrução catch. Também não é permitido definir checked exceptions que não são lançadas dentro do try, no exemplo acima uma IOException não seria aceita dentro do catch. Para uncheked exceptions (Runtime) o compilador é flexível, seria permitido declarar uma NullPointerException naquele catch.

Durante o TDC2011 edição São Paulo e Florianópolis, falamos bastante sobre o Java 7! Na próxima edição, em Goiânia, não poderia ser diferente, no sábado (29/10) vai rolar a palestra Tirando proveito dos novos recursos do Java 7 na trilha Java.

O Java 7 disponibiliza outros recursos que deixam a linguagem mais sucinta, para saber mais sobre essas outras novidades ou características relacionadas acesse os links abaixo:

[]s
Eder Magalhães
www.yaw.com.br
twitter.com/youandwe
twitter.com/edermag


Comentários

Yara Senger disse…
Parabéns pelo artigo Eder! Que energia para fazer coisas, postar noticias, palestras! Show de bola!

Postagens mais visitadas deste blog

TDC BUSINESS, chega a São Paulo com novas trilhas de Inteligência Artificial e Inovação

Maior conferência de profissionais de tecnologia do Brasil abordará temas em alta no momento como, por exemplo, Inteligência Artificial, Segurança, Ciência de Dados e Inovação O TDC BUSINESS, a 17° edição do The Developer's Conference na cidade de São Paulo, que acontece entre os dias 19 e 21 de Setembro, reunirá profissionais e especialistas da área para troca de experiência, compartilhamento de conteúdos e networking. Com o tema central: “Tecnologia para negócios transformadores”, o evento será totalmente híbrido, ocorrendo presencialmente no espaço Pro Magno, e com transmissão simultânea e atividades de network pela internet. A expectativa é reunir mais de 14.000 pessoas, somando a participação presencial e online.   Segundo Yara Mascarenhas, Fundadora e Host do Evento, “nosso objetivo com o TDC é inspirar a colaboração entre os profissionais e empresas para construir uma nova realidade para o mercado de TI.  Vamos juntar tecnologia e negócios com as trilhas técnicas...

12 ESQUENTAS TDC ESPERANDO POR VOCÊ

O TDC nasceu do sonho de profissionais que valorizam o compartilhamento de conhecimento, experiências e oportunidades entre as comunidades de desenvolvedores.  Por isso, o crescimento do TDC nunca poderia ficar longe do crescimento das comunidades! Pensando nessa união, criamos um projeto quente: O ESQUENTA TDC ! Visando divulgar os meetups de comunidades que querem transformar e inspirar carreiras, foi co-criado este projeto lindo e cheio de apoio, onde o TDC abraça meetups de diversas comunidades espalhadas pelo Brasil e elas contribuem com a divulgação da edição regente.  O TDC Innovation vai rolar de 14 a 16 de JUNHO, mas até lá vão acontecer muitos encontros incríveis!  Fique por dentro das comunidades que vão participar, quais temas irão abordar e claro as datas e locais para você participar. 💥Esquentas TDC Innovation 2023: 24/05 - Esquenta TDC: Carreira e Desenvolvimento Presencial em Floripa das 19:00 as 21:30 com inscrição AQUI Comunidade: Google Developer...

Site da Globalcode: Design e integração com JSF, Facelets, RichFaces e JBossSeam

O site da Globalcode foi criado por uma agência e depois desenvolvido internamente, utilizando algumas das nossas tecnologias preferidas: - JavaServer Faces - Facelets - RichFaces - JPA - JBoss Seam (que não poderia faltar na nossa arquitetura). O site é abastecido com informações do sistema de gestão educacional, o GES ( Global Education System ). Na prática isto quer dizer que as turmas, o conteúdo dos cursos, o Orçamento Eletrônico , matrícula eletrônica em minicursos gratuitos, páginas Unidades Globalcode , tudo vem do GES. Ou seja, estes dados são alterados automaticamente quando alteramos os dados no GES e não precisa passar nas mãos de ninguém da área de TI. E o pessoal de vendas e negócios adora ficar livre do pessoal da TI. A utilização de Facelets gerou muitos benefícios e facilidades, e agora até o André Frota , que era mais designer que programador fica aflito para eliminar uma ou outra página 100% HTML, que não usa Facelets. Além dos muitos novos requisitos ...

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...

JSON fácil em Java com GSon !

Ola pessoal ! O formato JSON ( J ava S cript O bject N otation) vem se consagrando cada vez mais na comunicação de dados, principalmente nos dispositivos móveis devido a esse formato ser mais leve que o XML e também mais legível. Uma prova disso são as inúmeras bibliotecas que existem para manipular esse formato, e no caso do Android, o suporte ao JSON é nativo. Mas apesar de ter esse suporte nativo, algumas operações devem ser feitas manualmente e o código acaba ficando um pouco verboso e repetitivo, já que para cada objeto que se deseja transmitir é necessário fazer um método que lê as propriedades do JSON e faz as devidas atribuições no seu objeto Java. Vamos supor o seguinte objeto sendo transmitido em JSON: {   user: {     id: 123456,     name: "Neto Marin",     username: "netomarin",     email: "netomarin@globalcode.com.br"   } } Se você fosse tratar um Webservice que envia esse JSON para o seu aplicativo Android...

JavaOne Brasil, dicas para submissão de palestras

Não quero parecer pretensiosa dando dicas para submissão de palestras para o JavaOne Brasil, mas sim repassar os tantos conselhos e sugestões recebidas pelos vetaranos do JavaOne: Bruno Souza e Leonardo Galvão que revisaram dezenas de submissões para o JavaOne e ajudaram a aprovar tantas palestras, e também misturar um pouco da minha experiência na seleção de palestras nos eventos realizados pela Globalcode e SouJava . 10 anos de JavaOne: http://www.globalcode.com.br/noticias/Globalcode10AnosNoJavaOne Os palestrantes ganham a entrada! A submissão pode ser feita em português! O passo mais importante para ser aprovado como palestrante no JavaOne é sem dúvida nenhuma submeter pelo menos uma palestra. Então, independente de qualquer coisa, participe, arrisque, divulgue.  Mas, se quiser aumentar as suas chances...   1) Leve a sério: peça para amigos fazerem uma leitura crítica do texto, e claro uma boa revisão ortográfica. 2) Submissão de várias palestras ou variações do ...