O documento resume algumas funcionalidades pouco conhecidas do Java, como inicialização de mapas usando classes internas anônimas, métodos que retornam subclasses, e o uso de exceções não checadas.
2. Josenaldo de Oliveira Matos FilhoAnalista de Sistema Everis8 anos de experiência com Java e desenvolvimentoUaiJUGLeaderApaixonado por programação e desenvolvimento de softwarehttp://jnaldo.comQuem sou eu?
3. São “features” pouco usadas do JavaNenhuma feature é escondida, apenas obscura, desconhecidaAlgumas delas são bem esquisitasMuitas consideradas até mesmo impossíveisUse com cautelaEm alguns casos, são facilidadesPorém, em outros, deixam as coisas mais complicadas.Podem piorar, por exemplo, a legibilidadeO que são HiddenFeatures
4. Uma das formas mais estranhas de se criar um Map inicializado é através da Double BraceInitialization.Ex: Double BraceInitializationMapmap = newHashMap() {{ put("a key", "a value"); put("anotherkey", "anothervalue"); }}; Veja mais: http://www.c2.com/cgi/wiki?DoubleBraceInitialization
5. WTF?Classes internas anônimas que herda de HashMapPortanto, herda métodos como put()Bloco de inicializaçãoChamado logo após a chamada a super()Em um construtor sem o “super”, passa a impressão de eu é chamado antes do construtorPode ser usado para agrupar lógica usada por vários construtoresÉ legal, mas evite!Double BraceInitialization
6. Uma adição do Java 1.5 ainda pouco conhecida (ou percebida)Um método sobrescrito pode retornar uma subclasse da classe retornada pelo método da superclasse (ou interface)Necessário para o funcionamento de genericsCovariantReturnclass Super {Collection<String> values() { ... }}classSubClassextends Super { @OverrideList<String> values() { ... }}
7. Quer lançar uma NullPointerException?Lançando NullPointerExcenptionpublicvoidmandaNullNaCara() {thrownull;}Como assim? O compilador pirou? Tá cobrando propina pra deixar isso passar?
8. A instrução throw, ao ser processada, avalia a expressão à direita. Se a expressão gera um Throwable, este é lançadoEx: throwgetException()Se a expressão completa normalmente, produzindo um valor não nulo, retorna esse valorEx: thrownewSograEmCasaException()Se a expressão resulta em null, o null é convertido para NullpointerException!Lançando NullPointerExcenptionVeja mais: http://www.adarshr.com/papers/npe http://java.sun.com/docs/books/jls/
9. Quer tirar uma exceção lançada de campo?Faz o return no finally!O código abaixo não lança a exceçãoBrincando com finallypublic static void fazAlgumaCoisa() { try { //Fazalgumacoisa… throw new RuntimeException(); } finally { return; } }
10. Quem for pego fazendo isso vai sofrer vudu reverso!Um erro é para ser tratado ou lançado, nunca escondido!AtençãoVeja Mais: http://jamesjava.blogspot.com/2006/03/dont-return-in-finally-clause.html
11. Como lançar uma exceção CHECADA de um método que não declara exceção!Agora mais bizarro ainda!importjava.rmi.RemoteException;classThrower {publicstaticvoidspit(final Throwable exception) {classEvilThrower<T extendsThrowable> { @SuppressWarnings("unchecked")privatevoidsneakyThrow(Throwable exception) throws T {throw (T) exception; } }newEvilThrower<RuntimeException>().sneakyThrow(exception); }}
12. Agora mais bizarro ainda!publicclassThrowerSample {publicstaticvoidmain( String[] args ) {Thrower.spit(newRemoteException("gounchecked!")); }}Mim não gostar de lançar exceção assim!
13. Java não tem goto, mas...LabeledBlocksQuero ver isso funcionar!// codigosaifora:{ for (int i = 0; i < N; ++i) { for (int j = i; j < N; ++j) { for (int k = j; k < N; ++k) { //mais código pogbreaksaifora; } } }}
14. LabeledBlockspublicstatic String getErrorMsg(String _data){ String _errMsg = “”; VALIDATION: { if (_data.length() > 10) { _errMsg = “ERR: Data lengthexceed 10 chars”; break VALIDATION; } // Resto docódigo de validação } // Tratamento da mensagem de erro if (_errMsg.length() != 0) { _errMsg += “\nPleasefixtheerror.”; } return _errMsg;} Problem?
15. Muita gente não sabe, mas enums podem:Ter MétodosTer AtributosTer ConstrutoresImplementar interfacesEnumVeja mais: http://java.sun.com/j2se/1.5.0/docs/guide/language/enums.html
17. Enumpublicenum Salas implements Sala{ PRIMEIRA{public Sala norte() {return SEGUNDA; } }, SEGUNDA{public Sala sul() {return PRIMEIRA; } }public Sala norte() { returnnull; }public Sala sul() { returnnull; }public Sala lest() { returnnull; }public Sala oeste() { returnnull; }}public interface Sala{public Sala norte();public Sala sul();public Sala leste();public Sala oeste();}
18. Desde a versão 1.5, o Java aceita parâmetros variáveisVar argspublicvoidfoo(String... bars) { for (String bar: bars) System.out.println(bar);}publicvoidtest() {foo("first","second","third")}
19. Printf e String.formatPrintf do entrou na versão 1.5String.format funciona analogamente, mas retorna ao invés de imprimirdouble x = 27.5;double y = 33.75;System.out.printf("x = %f y = %g", x, y);double x = 27.5;double y = 33.75;String s = String.format("x = %f y = %g", x, y);System.out.println(“String.format = “ + s);
20. Divisão por 0publicclass teste {publicstaticvoidmain(String[] args) { try{ int x = 15;int z = 0; double y = x / z; System.out.println(“A divisão é “ + y); }catch(ArithmeticException a){ System.out.println(“Erro!! Divisão por zero!!!”); } }}Ok. Isto lança uma ArithmeticException
21. Divisão por 0publicclass teste {publicstaticvoidmain(String[] args) {try{double x = 15;double z = 0;double y = x / z; System.out.println(“A divisão é “ + x); }catch(ArithmeticException a){ System.out.println(“Erro!! Divisão por zero!!!”); } }}WTF???????
22. Quer ver um dump da hierarquia de componentes numa aplicação Swing?Ctrl + Shift + F1Resultado no consoleBoa ferramenta para depuração! Swing
23. Qual o resultado disso?E isso?Brincando com URLsnew URL("http://www.guj.com.br").equals(new URL("http://208.109.100.149") )public intumMetodo(){System.out.println(“Acessando o Google:”); http://www.google.com return 1;}
24. Como acessar os métodos e campos privados de fora dessa classe?Arrebentando o encapsulamentopublicclassFoo {privateint bar;publicFoo() {setBar(17); }privatevoidsetBar(int bar) {this.bar=bar; }publicintgetBar() {return bar; }public String toString() {return "Foo[bar="+bar+"]"; }}
26. Arrebentando o encapsulamento System.out.println(foo); Field field=Foo.class.getDeclaredField("bar");field.setAccessible(true);field.set(foo, 23); System.out.println(foo); }}Isso vai dar merda!
27. Acesso esquisitopublicclassFoo {staticintfubar = 42;FoogetFoo() {returnnull; }publicstaticvoidmain(String args[]) {Foofoo = newFoo(); System.out.println(foo.getFoo().fubar); }}// saida: 42Agora você tá de sacanagem!
28. É possível criar um programa sem Main!Classe sem main!publicclassWithoutMain {static { System.out.println("Lookman, no main!!"); System.exit(0); }}$ javaWithoutMainLookman, no main!!
29. Distribuída com o JDK a partir da versão1.6_07Profiler leveIntegra uma série de ferramentas de linha de comandoVisualVMVeja mais: http://visualvm.java.net/
30. Todo arquivo .class começa com o valor hexadecimal 0xCAFEBABE, paraidentificá-lo como um arquivo de Bytecodeválido.Querexibiruma Splash screen?Bastarodar o programa com o parâmetro de máriona virtual splash:caminhoCuriosidadesjava -splash:imagem.jpeg -jar Reverso.jar
31. Classes anônimasVocê pode definir uma classe anônima e imediatamente chamar um método, mesmo que ela não implemente nenhuma interfacenewObject() {voidfoo(String s) { System.out.println(s); }}.foo("Hello");
32. Permite a criação de uma thread eu será chamada somente quando a JVM for encerradaGlobal JVM FinalizerLiberar recursosMatar programasFunciona com System.exit(), ou com CTRL-C / kill -15Obviamente, nãofunciona com kill -9, em *nixShutdown Hooks