Download Tutorial para el Examen de certificacion: SCJP 1.2
Document related concepts
no text concepts found
Transcript
Tutorial para el Examen de certificacion: SCJP 1.2 Traducción del tutorial de Marcus Green. RuGI Isaac Ruiz Guerra rugi@javahispano.org javaHispano Tu lenguaje, tu comunidad. Versión Julio del 2003. Tutorial para el examen de certificación: SCJP 1.2 El presente documento es una traducción/adaptación del tutorial de Marcus Green correspondiente a la versión de diciembre del 2002. Dicho tutorial cubre los objetivos correspondientes al examen SCJP 1.2. La mayor parte de este documento cubre tambien los objetivos para la versión 1.4 únicamente se deben pasar por alto las secciones 8 y 11, y agregar el tema de Aserciones. Esta traducción se realizó con la colaboración de: Nombre: Isaac Ruiz Guerra Joslyn Flores Romero Francisco Fernández Miser Alberto Molpeceres Sección: 1, 2, 3, 4, 6, 8, 9, 10 7 5 11 Puedes hacer cualquier comentario sobre la traducción a rugi@javahispano.org La dirección del tutorial de Marcus Green: http://www.jchq.net/certkey/index.htm Un artículo de Emili Miedes de Elías sobre Aserciones: http://www.javahispano.org/articles.article.action?id=57 Este documento forma parte del proyecto “Examen de Certificación”: http://www.javahispano.org/text.viewer.action?file=proyectos Copyright (c) 2003,Isaac Ruiz Guerra. Este documento puede ser distribuido solo bajo los términos y condiciones de la licencia de Documentación de javaHispano v1.0 o posterior (la última versión se encuentra en http://www.javahispano.org/licencias/). 2 javaHispano. Tu comunidad, tu lenguaje. Sección 1. Declaración y control de Acceso ......................7 Objetivo 1........................................................................................................................ 7 Arreglos................................................................................................................... 7 Declaración sin asignación. ........................................................................................ 7 Creación y Declaración Simultanea............................................................................ 8 Arreglos Java VS Arreglos C/C++ . .................................................................... 8 Los arreglos conocen su tamaño. ........................................................................... 9 Arreglos Java VS Arreglos VB............................................................................ 9 Combinar declaración con inicialización.................................................................. 10 Objetivo 2...................................................................................................................... 15 Comparando las clases de C++/VB con las de Java. ................................................ 15 El rol de las clases en Java........................................................................................ 15 La clase más sencilla................................................................................................. 16 Creando una clase sencilla- HelloWorld................................................................... 16 La magia del nombre main ....................................................................................... 16 Creando una instancia de una Clase.......................................................................... 17 Creando Métodos. ..................................................................................................... 18 Variables Automáticas .............................................................................................. 19 Modificadores y Encapsulamiento............................................................................ 20 Usando modificadores en combinación .................................................................... 26 Objetivo 3...................................................................................................................... 35 Nota de este objetivo................................................................................................. 35 ¿Qué es un constructor? ............................................................................................ 35 ¿Cuándo proporciona Java el constructor predefinido?............................................ 35 El prototipo del constructor predefinido ................................................................... 37 Objetivo 4...................................................................................................................... 43 Nota de este objetivo................................................................................................. 43 Métodos en la misma clase ....................................................................................... 43 Métodos en una subclase. ......................................................................................... 44 Sección 2. Control de Flujo y Manejo de Excepciones. ...........47 Objetivo 1...................................................................................................................... 47 La sentencia if/else.................................................................................................... 47 La Sentencia switch................................................................................................... 48 Argumentos validos para las sentencias if y switch. ................................................. 49 El operador ternario ?................................................................................................ 49 Objetivo 2...................................................................................................................... 55 La sentencia for......................................................................................................... 55 Los ciclos while y do, nada inesperado................................................................. 56 La sentencia goto, ¿ciencia o religión? ..................................................................... 57 Break y Continue ...................................................................................................... 57 Saltar a una Etiqueta. ................................................................................................ 58 Objetivo 3...................................................................................................................... 64 Comparando con Visual Basic y C/C++................................................................... 64 La Cláusula finally .................................................................................................... 65 Sobrescribiendo métodos que lanzan excepciones ................................................... 66 3 Tutorial para el examen de certificación: SCJP 1.2 Sección 3. Recolector de basura –Garbage Collector- ............72 Objetivo 1...................................................................................................................... 72 ¿Por qué querrías utilizar el recolector de basura? ................................................... 72 Java y el recolector de basura. .................................................................................. 72 Sección 4. Fundamentos del lenguaje. ...........................76 Objetivo 1...................................................................................................................... 76 Nota sobre este objetivo............................................................................................ 76 La sentencia package ................................................................................................ 76 La sentencia import ............................................................................................... 77 Clases y la declaración de clases internas................................................................. 77 Clases internas declaradas dentro de métodos .......................................................... 79 Visibilidad de campos para clases definidas dentro de un método........................... 80 Creando una interface ............................................................................................... 81 Objetivo 2...................................................................................................................... 85 Objetivo 3...................................................................................................................... 87 Objetivo 4...................................................................................................................... 90 Variables ................................................................................................................... 90 Arreglos..................................................................................................................... 91 Objetivo 5...................................................................................................................... 95 Tamaño de los tipos primitivos enteros .................................................................... 95 Declarando literales enteros...................................................................................... 95 Tamaño de los tipos primitivos de coma flotante ..................................................... 96 Indicando tipos de datos con un literal sufijo ........................................................... 97 Los tipos boolean y char........................................................................................... 97 Declarando literales String ................................................................................... 99 Sección 5. Operadores y Asignaciones ..........................102 Objetivo 1.................................................................................................................... 102 El operador instanceof ....................................................................................... 102 El operador + .......................................................................................................... 103 Asignando variables primitivas de tipos diferentes ................................................ 104 Asignando referencias de objetos de diferentes tipos. ............................................ 105 Operadores de desplazamiento de bits.................................................................... 106 Operadores de desplazamiento con signo << y >> ................................................. 107 Desplazamiento sin signo a la derecha >>>............................................................ 108 Applet BitShift ........................................................................................................ 108 Objetivo 2.................................................................................................................... 113 Diferencia entre el método equals y el operador = = .............................................. 113 Usando el método equals con Strings ..................................................................... 113 Usando el método equals con la Clase Boolean ..................................................... 114 Usando el método equals con objetos..................................................................... 115 Objetivo 3.................................................................................................................... 119 El efecto corto circuito con los operadores lógicos. ............................................... 119 Operadores a nivel de bits....................................................................................... 120 Pensando en binario ................................................................................................ 120 Objetivo 4.................................................................................................................... 124 Nota sobre este objetivo.......................................................................................... 124 Referencias de Objetos como parámetros de métodos. .......................................... 124 4 javaHispano. Tu comunidad, tu lenguaje. Primitivas como parámetros de métodos ................................................................ 125 Sección 6. Sobrecarga, sobreescritura, tipos en tiempo de ejecución y orientación a objetos .............................128 Objetivo 1) .................................................................................................................. 128 Relaciones “es un” y “tiene un”.............................................................................. 128 Encapsulamiento ..................................................................................................... 128 Tipos en Tiempo de ejecución ................................................................................ 130 Objetivo 2) .................................................................................................................. 135 Comentario sobre el objetivo .................................................................................. 135 Sobrecarga de métodos ........................................................................................... 135 Sobrescritura de métodos....................................................................................... 136 Invocando constructores de clases base.................................................................. 137 Invocando constructores con this() ..................................................................... 138 El constructor y la jerarquía de clases..................................................................... 139 Objetivo 3) .................................................................................................................. 146 Nota sobre este objetivo.......................................................................................... 146 Instanciando una clase ............................................................................................ 146 Clases internas ........................................................................................................ 146 Clases Anidadas de nivel Superior (Top Level) ..................................................... 147 Clases Miembro ...................................................................................................... 147 Clases creadas dentro de métodos........................................................................... 148 Clases anónimas...................................................................................................... 148 Sección 7. Threads ............................................152 Objetivo 1) .................................................................................................................. 152 ¿Qué es un Thread ?................................................................................................ 152 Las dos formas de crear un Hilo (Thread) .............................................................. 152 Instanciando e Iniciando un hilo. ............................................................................ 153 Objetivo 2) .................................................................................................................. 157 Comentario de este objetivo.................................................................................... 157 Razones por las que un thread puede ser bloqueado............................................... 157 Objetivo 3) .................................................................................................................. 161 ¿Por qué necesitarías el protocolo wait/notify? ...................................................... 161 synchronized ........................................................................................................... 161 wait/notify ............................................................................................................... 162 Sección 8. El paquete java.awt y su disposición ...............167 Objetivo 1.................................................................................................................... 167 Nota sobre este objetivo.......................................................................................... 167 Comparando la disposición de componentes de Java y Visual Basic..................... 167 La filosofía de los administradores de disposición ................................................. 167 Administradores de disposición que debes conocer para el examen. ..................... 169 Responsabilidades de los Administradores de Disposición VS los Contenedores . 169 Curiosidades del administrador BorderLayout ....................................................... 170 El administrador de disposición GridLayout .......................................................... 171 GridBagLayout ....................................................................................................... 172 Usando gridx y gridy para sugerir la posición de un componente.......................... 175 ipadx y ipady para controlar el relleno interior de los componentes ...................... 176 Componentes dentro de un panel utilizando GridBagLayout................................. 178 5 Tutorial para el examen de certificación: SCJP 1.2 Fijando componentes dentro de la cuadrícula......................................................... 180 Elementos GridBag no cubiertos por este ejercicio............................................... 182 Objetivo 2.................................................................................................................... 189 El modelo oyente -Listener- de eventos.................................................................. 189 El modelo de eventos del JDK 1.1.......................................................................... 189 Sección 9. El paquete java.lang.Math ..........................196 Objetivo 1.................................................................................................................... 196 Nota sobre este objetivo.......................................................................................... 196 abs ........................................................................................................................... 196 ceil........................................................................................................................... 196 floor......................................................................................................................... 197 max y min ............................................................................................................... 197 random .................................................................................................................... 198 round ....................................................................................................................... 198 sin cos tan................................................................................................................ 199 sqrt........................................................................................................................... 199 Resumen.................................................................................................................. 199 Objetivo 2.................................................................................................................... 203 Nota sobre este objetivo.......................................................................................... 203 Sección 10. El paquete java.util ..............................206 Objetivo 1.................................................................................................................... 206 Nota sobre este objetivo.......................................................................................... 206 Las colecciones antiguas......................................................................................... 206 Las nuevas colecciones ........................................................................................... 206 Set ........................................................................................................................... 207 List .......................................................................................................................... 207 Map ......................................................................................................................... 207 Usando Vectores ..................................................................................................... 208 Usando Hashtables.................................................................................................. 209 BitSet....................................................................................................................... 209 Sección 11. El paquete java.io ................................214 Objetivo 1.................................................................................................................... 214 Objetivo 2.................................................................................................................... 222 Objetivo 3.................................................................................................................... 228 Objetivo 4.................................................................................................................... 229 FileInputStream and OutputStream ........................................................................ 229 BufferedInputStream y BufferedOutputStream...................................................... 230 DataInputStream y DataOutputStream ................................................................... 230 La clase File......................................................................................................... 231 RandomAccesFile ................................................................................................... 231 Objetivo 5.................................................................................................................... 235 FileInputStream and FileOutputStream .................................................................. 235 RandomAccessFile ................................................................................................. 236 6 javaHispano. Tu comunidad, tu lenguaje. Sección 1. Declaración y control de Acceso Objetivo 1. Escribir código que declare, construya e inicie arreglos de cualquier tipo base, usando cualquiera de las formas permitidas, para declaración y para inicialización. Arreglos. Los arreglos en Java son similares en sintaxis a los arreglos en otros lenguajes como C/C++ y Visual Basic. Sin embargo, Java elimina la característica de C/C++ mediante la cual puedes pasar los corchetes ([]) accediendo a los elementos y obteniendo sus valores utilizado punteros. Esta capacidad en C/C++, aunque poderosa, propicia la escritura de software defectuoso. Debido a que Java no soporta esta manipulación directa de punteros, se remueve esta fuente de errores. Un arreglo es un tipo de objeto que contiene valores llamados elementos. Esto te da un conveniente contenedor de almacenamiento para un grupo de valores que pueden modificarse durante el programa, y permite que accedas y cambies los valores según lo necesites. A diferencia de las variables que son accedidas por un nombre, los elementos de un arreglo son accedidos por números comenzando por cero. De esta manera puedes “avanzar” a través del arreglo, accediendo a cada elemento en turno. Todos los elementos de un arreglo deben ser del mimo tipo. El tipo de elementos de un arreglo, se decide cuando se declara el arreglo. Si lo que necesitas es una manera para almacenar un grupo de elementos de tipos diferentes, puedes usar las clases de colecciones, estas son una nueva característica en Java 2 y son discutidas en la sección 10. Declaración sin asignación. La declaración de un arreglo no asigna almacenamiento alguno, ésta solo anuncia la intención de crear un arreglo. Una diferencia significativa con la manera en que C/C++ declara un arreglo es que el tamaño no se especifica con el identificador. Por lo tanto lo siguiente causara un error durante la compilación int num[5]; El tamaño de un arreglo se da cuando este se crea realmente con el operador new: int num[]; num = new int[5]; 7 Tutorial para el examen de certificación: SCJP 1.2 Creación y Declaración Simultanea. Lo anterior puede ser compactado en una sola línea así: int num[] = new int[5]; Además los corchetes pueden ser también colocados después del tipo de dato o después del nombre del arreglo. Por lo tanto son legales las siguientes declaraciones: int[] num; int num[]; Puedes leer esto de la siguiente manera: • • Un arreglo de enteros llamado num Un arreglo llamado num de tipo entero. Arreglos Java VS Arreglos C/C++ . Una gran diferencia entre java y C/C++ es que en java se conocen de que tamaño son los arreglos, y el lenguaje proporciona protección para no sobrepasar los limites del arreglo. Este comentario es particularmente útil si vienes de un ambiente de programación como Visual Basic donde no es muy común iniciar el conteo de un arreglo desde 0. También ayuda a evitar uno de los mas insidiosos errores de C/C++. Por lo tanto, lo siguiente causará el siguiente error en tiempo de ejecución: ArrayIndexOutOfBoundException int[] num = new int [5]; for (int i = 0;i<6;i++){ num[i] = i*2; } Para recorrer un arreglo, la manera mas común es a través del campo length de los arreglos, así: int [] num = new int[5]; for(int i = 0;i<num.length;i++){ num[i]=i*2; } 8 javaHispano. Tu comunidad, tu lenguaje. Los arreglos conocen su tamaño. Solo en caso de que te hayas saltado la comparación con C/C++ -hecha anteriormentelos arreglos en Java siempre conocen cual es su tamaño, y este se representa en el campo length. Así, puedes llenar dinámicamente un arreglo con el código siguiente: int myarray[] = new int [10]; for(int j=0; j<myarray.length; j++){ myarray[j]=j; } Nota que los arreglos tienen el campo length y no el método length(). Cuando comiences a utilizar Strings usaras el método length, de la siguiente manera: s.length(); En un arreglo length es un campo (o propiedad) no un método. Arreglos Java VS Arreglos VB Los arreglos en Java siempre comienzan desde cero. Los arreglos en VB pueden comenzar desde 1 si se usa la declaración Option base. En Java no hay equivalente para el comando redim preserve a través del cual puedes cambiar el tamaño de un arreglo sin borrar su contenido. Pero por supuesto puedes crear un nuevo arreglo con un nuevo tamaño y copiar los elementos actuales a ese arreglo y obtener el mismo resultado. La declaración de un arreglo puede tener múltiples conjuntos de corchetes (“[]”). Java formalmente no soporta los arreglos multidimensionales; sin embargo si soporta arreglos de arreglos; también conocidos como arreglos anidados. Una diferencia importante entre los arreglos multidimensionales, como en C/C++ y los arreglos anidados, es que cada arreglo no tiene que ser del mismo tamaño. Si vemos un arreglo como una matriz, la matriz no tiene que ser una matriz rectángulo . De acuerdo con la Especificación del Lenguaje Java: (http://java.sun.com/docs/books/jls/html/10.doc.html#27805) “El número de corchetes indica la profundidad de un arreglo anidado” En otros lenguajes esto se conoce como la dimensión de un arreglo. Así podrías almacenar las coordenadas de un mapa con un arreglo de 2 dimensiones: int [][]; la 1ª dimensión podría ser la coordenada X y la segunda la coordenada Y. NT.: En otras palabras; cada arreglo puede tener un tamaño distinto. 9 Tutorial para el examen de certificación: SCJP 1.2 Combinar declaración con inicialización. En vez de ciclarse a través de un arreglo para llevar a cabo la inicialización, un arreglo puede ser creado e inicializado a la vez en una sola declaración. Esto es particularmente recomendable para arreglos pequeños. La siguiente línea creara un arreglo de enteros y lo llenara con los números 0 hasta el 4. int k[] = new int[] {0,1,2,3,4}; Observa que en ninguna parte necesitas especificar el número de elementos en el arreglo. Puedes analizar esto, preguntándote si el siguiente código es correcto: Int k = new int[5]{0,1,2,3,4}; //¡¡¡ error, no compilara !!! Puedes crear y llenar arreglos simultáneamente con cualquier tipo de dato, por lo tanto puedes crear un arreglo de cadenas, de la siguiente manera: String s[] = new String [] {“Zero”,”One”,”Two”,”Three”,”Four”}; Los elementos de un arreglo pueden ser direccionados como lo harías en C/C++: String s[] = new String [] {“Zero”,”One”,”Two”,”Three”,”Four”}; System.out.println(s[0]); Esto mostrará la cadena Zero. Ejercicio 1) Crea una clase con un método que simultáneamente cree e inicialize un arreglo de cadenas. Inicializa el arreglo con cuatro nombres, después muestra el primer nombre contenido en el arreglo. Ejercicio 2) Crea una clase que cree un arreglo de cadenas de 5 elementos llamado Fruit en el nivel de clase, pero no lo inicializes con cualquier valor. Crea un método llamado amethod. En amethod inicializa los primeros cuatro elementos con nombres de frutas. Crea otro método llamado modify y cambia el contenido del primer elemento del arreglo Fruit para que contenga la cadena “bicycle”. Dentro del método modify crea un ciclo for que imprima todos los elementos del arreglo. Solución sugerida para el ejercicio 1. public class Bevere{ public static void main(String argv[]){ Bevere b = new Bevere(); b.Claines(); 10 javaHispano. Tu comunidad, tu lenguaje. } public void Claines(){ String[] names= new String[]{"Peter","John","Balhar","Raj"}; System.out.println(names[0]); } } Nota: La sintaxis para la creación e inicialización simultanea no es evidente y vale la pena practicarla. Se preguntó por el primer nombre del arreglo para asegurar que no utilizarás names[1]. Solución Sugerida para el ejercicio 2. public class Barbourne{ String Fruit[]= new String[5]; public static void main(String argv[]){ Barbourne b = new Barbourne(); b.amethod(); b.modify(); } public void amethod(){ Fruit[0]="Apple"; Fruit[1]="Orange"; Fruit[2]="Bannana"; Fruit[3]="Mango"; } public void modify(){ Fruit[0]="Bicycle"; for(int i=0; i< Fruit.length; i++){ System.out.println(Fruit[i]); } } } Nota: Cuando el ciclo ejecute la salida de elemento final este será null. 11 Tutorial para el examen de certificación: SCJP 1.2 Preguntas. Pregunta 1) ¿Cómo puedes redimensionar un arreglo con una declaración sencilla manteniendo el contenido original? Pregunta 2) Quieres averiguar el valor del último elemento de un arreglo. Escribiendo el siguiente Código.¿Qué pasara cuando lo compiles y ejecutes? public class MyAr{ public static void main(String argv[]){ int[] i = new int[5]; System.out.println(i[5]); } } Pregunta 3) Quieres hacer un ciclo a través de un arreglo y detenerte cuando llegues al ultimo elemento. Siendo un buen programador en Java, y olvidando lo que alguna vez conociste sobre C/C++, sabes que los arreglos contienen información acerca de su tamaño. ¿Cuál de las siguientes sentencias puedes usar? 1)myarray.length(); 2)myarray.length; 3)myarray.size 4)myarray.size(); Pregunta 4) Tu jefe esta complacido por que escribiste el programa Hello World y te ha dado un aumento. Ahora te encomienda que crees un juego como el TicTacToe(o puntos y cruces como lo conocí en mi adolescencia). Decides que para eso necesitas un arreglo multidimensional. ¿Cuál de las siguientes líneas realizan el trabajo? 1) 2) 3) 4) int i=new int[3][3]; int[] i=new int[3][3]; int[][] i=new int[3][3]; int i[3][3]=new int[][]; Pregunta 5) 12 javaHispano. Tu comunidad, tu lenguaje. Quieres encontrar una manera más elegante para llenar un arreglo que a través de un ciclo con la sentencia for. ¿Cuál de las siguientes líneas usarías? 1) myArray{ [1]="One"; [2]="Two"; [3]="Three"; end with 2)String s[5]=new String[] {"Zero","One","Two","Three","Four"}; 3)String s[]=new String[] {"Zero","One","Two","Three","Four"}; 4)String s[]=new String[]={"Zero","One","Two","Three","Four"}; Respuestas Respuesta 1) No puedes redimensionar un arreglo. Necesitas crear un nuevo arreglo temporal con un tamaño diferente y llenarlo con el contenido del arreglo original. Java Proporciona contenedores redimencionables con clases como Vector o uno de los miembros de la clase collection . Respuesta 2) Intentar avanzar mas allá del final del arreglo genera un error en tiempo de ejecución. Debido a que los arreglos son indexados desde 0, el elemento final será i[4] y no i[5]. Respuesta 3) 2) myarray.length; Respuesta 4) 3) int[][] = new int[3][3]; Respuesta 5) 3) String s[] = new String[]{“Zero,”One”,”Two”,”Three”,”Four”}; 13 Tutorial para el examen de certificación: SCJP 1.2 Otras Fuentes de este tema: Este tema es cubierto en el Tutorial de Sun en: http://java.sun.com/docs/books/tutorial/java/data/arraysAndStrings.html Richard Baldwin trata este tema en: http://www.Geocities.com/Athens/Acropolis/3797/Java028.htm Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec1.html#obj1 Bruce Eckel Thinking In Java Capítulo 8 14 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2. Declarar clases, clases internas, métodos, variables de instancia estáticas, variables y variables automáticas(métodos locales), haciendo un uso apropiado de todos los modificadores permitidos (como son: public, final, static, abstract, y demás). Conocer importancia de cada uno de estos modificadores ya sea solos o en combinación Comentario sobre este objetivo Encuentro un poco molesto que en el objetivo se use la frase “y demás”. Supongo que esto significa que debes estar consciente de los modificadores: native transient synchronized volatile Comparando las clases de C++/VB con las de Java. Debido a que Java fue diseñado para facilitar que programadores C++ lo aprendieran, este tiene muchas similitudes en la manera en que ambos lenguajes trabajan con las clases. Ambos lenguajes tienen herencia, polimorfismo, y ocultamiento de datos usando modificadores de visibilidad. Algunas de sus diferencias fueron hechas para hacer de Java un lenguaje más fácil de aprender y usar. El lenguaje C++ implementa la herencia múltiple y así una clase puede tener más de un padre (o clase base) Java únicamente permite la herencia simple, es decir, las clases pueden tener únicamente un padre. Para sobreponer esta limitación, Java tiene una característica llamada: Interfaces, a la cual los diseñadores del lenguaje le decidieron dar algunas de las ventajas de la herencia múltiple, pero sin las desventajas. Todas las clases en Java son descendientes de una gran clase antecesora llamada Object. El rol de las clases en Java Las clases son el corazón de Java, todo el código en Java ocurre dentro de una clase. No hay un concepto de código llano y de posición libre. Hasta la más sencilla aplicación HelloWorld involucra la creación de una clase. Para indicar que una clase desciende de otra clase se usa la palabra clave extends. Si no se usa extends, la clase descenderá de la clase Object lo cual proporciona a la clase creada, algunas funciones básicas incluyendo la habilidad para mostrar su nombre y algunas de las capacidades requeridas en los Threads. 15 Tutorial para el examen de certificación: SCJP 1.2 La clase más sencilla. Los requerimientos mínimos para definir una clase son: la palabra clave class, el nombre de la clase y las llaves de apertura y cerradura. Entonces: class classname {} es una definición sintacticamente correcta para una clase, pero no es una clase particularmente útil (sorprendentemente, me he encontrado definiendo clases como esta, cuando creo ejemplos para ilustrar la herencia) Normalmente una clase también incluirá un especificador de acceso antes de la palabra clave class y por supuesto, el cuerpo de la clase entre las llaves. Así, lo siguiente es una plantilla mas sensata para definir una clase: public class classmane{ // el cuerpo de la clase va acá } Creando una clase sencilla- HelloWorld Este es un sencillo programa HelloWorld que mostrar: la cadena “hello world” en la pantalla. public class HelloWorld{ public static void main(String argv[]){ System.out.println("hello world"); } }//Fin de la Definición de la clase La palabra clave public es un modificador de visibilidad que indica que la clase deberá ser visible para cualquier otra clase. Solo una clase por archivo puede ser declarada publica. Si declaras mas de una clase en un archivo como publica, ocurrirá un error en tiempo de ejecución. Observa que Java es sensible a mayúsculas en todos los aspectos. El archivo que contenga esta clase deberá llamarse HelloWorld.java La palabra clave class indica que una clase comienza a ser definida y HelloWorld es el nombre de la clase. Observa que la llave de cierre que finaliza la definición de la clase no involucra un punto y coma de cerradura. El comentario: //Fin de la Definición de la clase Usa el estilo de comentario de una sola línea que esta disponible en C/C++. Java también comprende los comentarios multilíneas de la forma /* */. La magia del nombre main 16 javaHispano. Tu comunidad, tu lenguaje. Dando a un método la siguiente forma, esté obtiene cierta significancia (o magia) cuando indica a Java que es aquí en donde el programa debe empezar a ejecutarse. (similar al main del lenguaje C). public static void main(String argv[]){ Esta línea indica que esta definiéndose un método llamado main, y que toma como parámetros (o argumentos) a un arreglo de cadenas. Este método es publico, es decir es visible desde cualquier parte de la clase. La palabra clave static indica que este método puede ejecutarse sin crear una instancia de la clase. Si eso no significa nada para ti, no te preocupes, en su momento se cubrirán con detalle los métodos estáticos (static) en otra parte de este tutorial. La palabra clave void indica el tipo de dato retornado por el método cuando se llama. Usar void indica que ningún valor será retornado. El parámetro del método main: String argv[] Indica que el método toma un arreglo de tipo String. Los corchetes indican –como ya vimos- un arreglo. Observa que el tipo de dato String comienza con una “S” mayúscula. Esto es importante ya que Java es totalmente sensible a las mayúsculas. Sin estas exactitudes, la Maquina virtual de Java (JVM) no reconocerá el método como el lugar en donde se debe comenzar la ejecución del programa. Creando una instancia de una Clase La aplicación HelloWorld, como describí anteriormente, es útil para ilustrar la más sencilla de las aplicaciones que puedes crear. Pero le falta mostrar uno de los elementos más cruciales al usar las clases, la palabra clave new. La cual indica la creación de una nueva instancia de la clase. En la aplicación HelloWorld esto no era necesario ya que el único método que se llamó era System.out.println que es un método estático y no requiere la creación de una clase que utilice la palabra new. Los métodos estáticos pueden acceder solo a variables estáticas, de las que sólo puede existir una instancia por la clase. La aplicación de HelloWorld puede modificarse ligeramente para ilustrar la creación de una nueva instancia de una clase. public class HelloWorld2{ public static void main(String argv[]){ HelloWorld2 hw = new HelloWorld2(); hw.amethod(); } public void amethod(){ System.out.println("Hello world"); } } 17 Tutorial para el examen de certificación: SCJP 1.2 Este código crea una nueva instancia de si mismo en la línea: HelloWorld2 hw = new HelloWorld2(); Esta sintaxis para crear una nueva instancia de una clase es básica para el uso de clases. Observa cómo el nombre de la clase aparece dos veces. La primera vez indica el tipo de dato de la referencia a la clase. Esta necesidad no es la misma que el tipo actual de la clase puesto que ésta se indica después de usar la palabra clave new. El nombre de esta instancia de la clase es hw. Éste simplemente es un nombre escogido para una variable. Este es un nombramiento convencional (es decir, utilizando la convención para escritura de código Java)ya que una instancia de una clase empieza con una letra minúscula, considerando que la definición de una clase empieza con una letra mayúscula. El paréntesis vacío para el nombre de la clase HelloWorld() indica que la clase está creándose sin algún parámetro en su constructor. Si estuvieras creando una instancia de una clase que se inicializa con un valor o un arreglo como la clase Label o Button el paréntesis contendría uno o mas valores de inicialización. Creando Métodos. Como ilustro en el último ejemplo HelloWorld2, un método en Java es similar a una función en C/C++ y a una función o sub función en Visual Basic. El método llamado amethod en este ejemplo se declara como: public que indica que puede accederse desde cualquier parte. Retorna el tipo: void Indicando que ningún valor será retornado. Y los paréntesis vacíos indican que no toma parámetros. El mismo método se podría haber definido de estas maneras alternativas: private void amethod(String s) private void amethod(int i, String s) protected void amethod(int i) Estos ejemplos ilustran algunas otras firmas típicas de declaración de métodos. El uso de las palabras clave private y protected se cubrirá en otro apartado. La diferencia entre los métodos de Java y métodos en lenguajes no orientados a objetos como C es que los métodos pertenecen a una clase. Esto significa que se llaman usando la anotación del punto que indica la instancia de la clase a la que el método pertenece (Los métodos estáticos son una excepción a esto pero no te preocupes sobre ellos por el momento) 18 javaHispano. Tu comunidad, tu lenguaje. Así en HelloWorld2 el método amethod se llamo así: HelloWorld hw = new HelloWorld(); hw.amethod(); Si se hubieran creado otras instancias de la clase HelloWorld el método podría llamarse desde cada instancia de la clase. Cada instancia de la clase tendría acceso a sus propias variables. Así lo siguiente involucraría la llamada al código del método amethod de las diferentes instancias de la clase. HelloWorld hw = new HelloWorld(); HelloWorld hw2 = new HelloWorld(); hw.amethod(); hw2.amethod(); Las dos instancias de la clase: hw y hw2 podrían tener acceso a variables diferentes. Variables Automáticas Las variables automáticas son las variables de los métodos. Estas entran en el alcance del programa cuando empieza a ejecutarse el código del método y dejan de existir una vez que el método deja de ejecutarse. Son sólo visibles dentro del método y son útiles principalmente para la manipulación de datos temporales. Si quieres que un valor permanezca entre las llamadas de un método entonces la variable necesita ser creada al nivel de la clase. Una variable automática será la "sombra" de una variable de nivel de clase. Así el código siguiente mostrará 99 y no 10: public class Shad{ public int iShad=10; public static void main(String argv[]){ Shad s = new Shad(); s.amethod(); }//fin de main public void amethod(){ int iShad=99; System.out.println(iShad); }//fin de amethod } 19 Tutorial para el examen de certificación: SCJP 1.2 Modificadores y Encapsulamiento. Los modificadores de visibilidad son parte del mecanismo de encapsulamiento para Java. El encapsulamiento permite la separación de la interfase y la implementación de métodos. A menudo estos métodos son para recuperar y actualizar los valores de una variable local privada. Son conocidos como métodos accessor (acceder o obtener el valor de la variable) y mutator (modificar el contenido de una variable). La convención para nombrar a estos métodos es setFoo para modificar y getFoo para obtener el contenido de una variable. Así si estuvieras almacenado una variable llamada age deberías hacerla privada y actualizarla con: setAge y recuperar su valor con: getAge Usando código para modificar variables; el valor de la variable también puede verificarse, por ejemplo: si el valor esta dentro de un rango en particular o revisar si es un numero positivo. Private Las variables privadas son sólo visibles dentro de la misma clase donde se crean. Esto significa que NO son visibles dentro de las subclases. Esto permite que la variable sea aislada de ser modificada por cualquier método, exceptuando aquellos en la clase actual. Como se dijo anteriormente, esto es útil para separar la interfase de la implementación. class Base{ private int iEnc=10; public void setEnc(int iEncVal){ if(iEncVal < 1000){ iEnc=iEncVal; }else System.out.println("Enc value must be less than 1000"); //Or Perhaps thow an exception }//End if } public class Enc{ public static void main(String argv[]){ Base b = new Base(); b.setEnc(1001); }//End of main } 20 javaHispano. Tu comunidad, tu lenguaje. Public El modificador public puede aplicarse a una variable (campo) o una clase. Es probablemente el primer modificador que conociste aprendiendo Java. Si revisas el código para HelloWorld.Java, que se programó en un principio, la clase se declaró como: public class HelloWorld Esto es porque para iniciar el método mágico la JVM, sólo revisa en una clase declarada como publica: public static void main(String argv[]) Una clase pública tiene alcance global, y una instancia puede crearse desde cualquier parte dentro o fuera de un programa. Únicamente una clase en un archivo puede definirse con la palabra clave public. Si defines más de una clase en un archivo con la palabra clave public el compilador generará un error class Base { public int iNoEnc=77; } public class NoEnc{ public static void main(String argv[]){ Base b = new Base(); b.iNoEnc=2; System.out.println(b.iNoEnc); }//End of main } Observa que generalmente ésta no es la manera sugerida ya que no permite ninguna separación entre la interfase y implementación del código. Si decidieras cambiar el tipo de dato de iNoEnc, tendrías que modificar la implementación de cada parte del código externo que lo modifica. Protected El modificador protected es una pequeña rareza. Una variable protegida es visible dentro de una clase, en una subclase, y en el mismo paquete, pero no en otra parte. La cualidad de ser visible dentro del mismo paquete puede darle más visibilidad de la que podrías sospechar. Cualquier clase que esté en el mismo directorio se considera que está en el paquete predefinido, y así las clases protegidas serán visibles. Esto significa que una variable protegida tiene más visibilidad que una variable definida sin modificador de acceso. 21 Tutorial para el examen de certificación: SCJP 1.2 Se dice que una variable definida sin modificador de acceso tiene visibilidad predefinida. La visibilidad predefinida significa que una variable puede verse dentro de la clase, y desde otra parte dentro del mismo paquete, pero no desde una subclase que no esté en el mismo paquete. Static Static no es directamente un modificador de visibilidad, aunque en la práctica tiene este efecto. El modificador static puede aplicarse a una clase, a un método y una variable. Marcando una variable como estática se indica que sólo una copia de esa variable existirá por clase. Esto está en contraste con las variables normales donde por ejemplo una copia de una variable integer pertenece a cada instancia de una clase. Así en el ejemplo siguiente existirán tres instancias de la variable integer iMyVal y cada instancia podra contener un valor diferente. class MyClass{ public int iMyVal=0; } public class NonStat{ public static void main(String argv[]){ MyClass m1 = new MyClass(); m1.iMyVal=1; MyClass m2 = new MyClass(); m2.iMyVal=2; MyClass m3 = new MyClass(); m3.iMyVal=99; //Esto mostrará 1. Ya que cada instancia de // la clase tiene su propia copia de MyVal System.out.println(m1.iMyVal); }//End of main } El Ejemplo siguiente muestra que pasa cuando tienes múltiples instancias de una clase que contiene una variable estática, en este ejemplo la variable estática es iMyEval. class MyClass{ public static int iMyVal=0; }//End of MyClass public class Stat{ public static void main(String argv[]){ MyClass m1 = new MyClass(); m1.iMyVal=0; MyClass m2 = new MyClass(); m2.iMyVal=1; MyClass m3 = new MyClass(); m2.iMyVal=99; //Debido a que iMyVal es static, //hay sólo una copia de él no importa 22 javaHispano. Tu comunidad, tu lenguaje. //cuantas instancias de la clase se creen. //Este código mostrará el valor de 99 System.out.println(m1.iMyVal); }//End of main } Ten presente que no puedes acceder a variables no estáticas desde dentro de un método estático. Así lo siguiente causará un error en tiempo de compilación. public class St{ int i; public static void main(String argv[]){ i = i + 2;//Causará un error al compilar } } Un método estático no puede ser sobreescrito para ser no estático en una clase hija. Un método estático no puede ser sobreescrito para ser no estático en una clase hija. No hay ninguna regla similar con referencia a la sobrecarga. El código siguiente causará un error cuando intentes sobreescribir el método amethod de la clase Base para ser no estático. class Base{ public static void amethod(){ } } public class Grimley extends Base{ public void amethod(){}//Causa un error en tiempo de compilación } El compilador Jikes de IBM produce el siguiente error: Found 1 semantic error compiling "Grimley.java": 6. public void amethod(){} <-------> *** Error: The instance method "void amethod();" cannot override the static method "void amethod();" declared in type "Base" Native El modificador native sólo se usa para los métodos e indica que el cuerpo del código esta escrito en un lenguaje que no es Java como C y C++. Se escriben a menudo métodos 23 Tutorial para el examen de certificación: SCJP 1.2 nativos para propósitos específicos de la plataforma como acceder a algún elemento de hardware del cual la Maquina Virtual de Java no este consciente. Otra razón es utilizarlo donde se requiere mayor desempeño. Un método nativo termina con punto y coma en lugar de un bloque del código. Así lo siguiente llamaría una rutina externa, escrita quizás en C++, public native fastcalc(); Abstract Es fácil pasar por alto el modificador abstract y perderse algunas de sus implicaciones. Es la clase de modificador que a los examinadores les gusta para hacer preguntas complicadas. El modificador abstracto puede aplicarse a las clases y a los métodos. Cuando se aplica a un método indica que éste no tendrá cuerpo y el código sólo puede ejecutarse cuando se implemente en una clase hija. Sin embargo hay algunas restricciones sobre cuando y donde puedes tener métodos abstractos y reglas sobre las clases que los contienen. Cuando se aplica a una clase, indica que la clase tiene un por lo menos método abstracto. Si una clase tiene algún método abstracto esta debe ser asimismo declarada abstracta . Sin embargo, no te distraigas pensado que una clase abstracta no puede tener métodos no abstractos. Cualquier clase que descienda de una clase abstracta debe implementar los métodos abstractos de la clase base o debe declararlos abstractos ella misma. ¿Tienden estas reglas a generar la pregunta por qué querrías crear métodos abstractos? Los métodos abstractos son principalmente beneficiosos a los diseñadores de clases. Ofrecen al diseño de una clase una manera de crear un prototipo para métodos que han de ser implementados, pero la implementación real queda a las personas que usan las clases después. Aquí hay un ejemplo de una clase abstracta con un método abstracto. De nuevo nota que la propia clase se declara abstracta, de otra manera habría ocurrido un error en tiempo de compilación. La siguiente clase es abstracta y se compilará sin problemas: public abstract class abstr{ public static void main(String argv[]){ System.out.println("hello in the abstract"); } public abstract int amethod(); 24 javaHispano. Tu comunidad, tu lenguaje. } Final El modificador final indica que un método no puede ser heredado. Otra manera de entender esto es que una clase final no puede ser una clase padre. Cualquier método en una clase final es automáticamente final. Esto puede ser útil si no quieres que otros programadores "hechen a perder tu código". Otro beneficio es la eficacia puesto que el compilador tiene menos trabajo al trabajar con un método final. Esto se cubre mejor en el Volumen 1 de Core Java. El código siguiente ilustra el uso del modificador final en una clase. Este código mostrará la cadena "amethod": final class Base{ public void amethod(){ System.out.println("amethod"); } } public class Fin{ public static void main(String argv[]){ Base b = new Base(); b.amethod(); } } Shynchronized La palabra clave synchonized se usa para prevenir que más de un thread acceda a un bloque de código a la vez. Ve la sección 7 sobre threads para entender más cómo funciona esto. Transient La palabra clave transient es uno de los modificadores frecuentemente menos usados. Indica que una variable no debe escribirse fuera cuando una clase es serializada. Volatile Probablemente no conseguirás realizar alguna pregunta sobre la palabra clave volatile. Lo peor que conseguirás es reconocer que realmente es un palabra clave de Java. Según Barry Boone: "le dice al compilador que una variable puede cambiar asincrónicamente debido a los threads" 25 Tutorial para el examen de certificación: SCJP 1.2 Acepta que es parte del lenguaje y entonces mejor preocúpate por otras cosas. Usando modificadores en combinación Los modificadores de visibilidad no pueden usarse en combinación, así una variable no puede ser privada y pública, pública y protegida o protegido y privada. Puedes tener combinaciones de los modificadores de visibilidad y los modificadores mencionados, por supuesto, en mi lista "y demás". • • • • native transient synchronized volatile Así puedes tener un método public static native. Donde pueden usarse los modificadores Modificador Método Variable clase public si si si private si si si(anidable) protected si si si(anidable) abstract si no si final si si si transient no si no native si no no volatile no si no Ejercicio 1) Crea un archivo llamado Whitley.java. En este archivo define una clase llamada Base con un método abstracto llamado lamprey que retorne un valor de tipo int. En el mismo archivo crea una clase llamada Whitley que descienda de la clase Base. Proporciona a la clase Whithley un método llamado lamprey con el código necesario para mostrar en pantalla la cadena “lamprey”. Crea un método nativo para la clase llamado mynative. Ahora compila y ejecuta el código. Ejercicio 2) 26 javaHispano. Tu comunidad, tu lenguaje. Crea una clase publica llamada Malvern. Crea una clase interna y privada llamada Great que tenga un método llamado show de la forma public void. Haz que este método muestre en pantalla la cadena “Show”. Proporciona a la clase Malvern un método publico llamado go que cree una instancia de Great y después de ser instanciada llame al método show. En el método main de la clase Malvern crea una instancia de sí misma. Haz que esta instancia llame al método go. Compila y ejecuta el código. Solución sugerida para el ejercicio 1. abstract class Base{ abstract int lamprey(); } public class Whitley extends Base{ public static void main(String argv[]){ } public int lamprey(){ System.out.println("lamprey"); return 99; } native public void mynative(); } Solución sugerida para el ejercicio 2. public class Malvern{ public static void main(String argv[]){ Malvern m = new Malvern(); m.go(); } public void go(){ Great g = new Great(); g.show(); } private class Great{ public void show(){ System.out.println("Show"); } } } 27 Tutorial para el examen de certificación: SCJP 1.2 Preguntas. Pregunta 1) ¿Qué pasará cuando intentes compilar y ejecutar este código? abstract class Base{ abstract public void myfunc(); public void another(){ System.out.println("Another method"); } } public class Abs extends Base{ public static void main(String argv[]){ Abs a = new Abs(); a.amethod(); } public void myfunc(){ System.out.println("My func"); } public void amethod(){ myfunc(); } } 1) El código se compilará y se ejecutará, mostrando la cadena “My func” 2) El código indicara que la clase Base no tiene métodos abstractos. 3) El código se compilara pero indicará un error en tiempo de ejecución (run time error): la clase base no tiene métodos abstractos 4) El compilador indicara que el método myfunc en la clase Base no tiene cuerpo. Pregunta 2) ¿Qué pasara cuando intentes compilar y ejecutar este código? public class MyMain{ public static void main(String argv){ System.out.println("Hello cruel world"); } } 28 javaHispano. Tu comunidad, tu lenguaje. 1) El compilador indicara que main es una palabra reservada y no puede ser usada para nombrar una clase. 2) El código se compilará y cuando se ejecute mostrara en pantalla la cadena “Hello cruel world”. 3) El código se compilara pero indicará un error en tiempo de ejecución: el constructor no esta definido. 4) El código se compilara pero indicara que la función main no esta definida correctamente. Pregunta 3) ¿De los siguientes cuales son modificadores de Java? 1) 2) 3) 4) public private friendly transient Pregunta 4) ¿Qué pasará cuando intentes compilar y ejecutar este código? class Base{ abstract public void myfunc(); public void another(){ System.out.println("Another method"); } } public class Abs extends Base{ public static void main(String argv[]){ Abs a = new Abs(); a.amethod(); } public void myfunc(){ System.out.println("My func"); } public void amethod(){ myfunc(); } } 1) El código se compilará y se ejecutará, mostrando la cadena “My func” 2) El compilador indicará que la clase Base no esta declarada como abstracta. 3) El código se compilará pero indicará un error en tiempo de ejecución: la clase base no tiene métodos abstractos 4) El compilador indicará que el método myfunc en la clase Base no tiene cuerpo. 29 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 5) ¿Para que defines un método como nativo? 1) Para conseguir acceder a Hardware desconocido por Java 2) Para definir un nuevo tipo de dato como un entero sin signo. 3) Para escribir código optimizado para el mejor desempeño en un lenguaje como C/C++. 4) Para superar la limitación del alcance privado de un método Pregunta 6) ¿Qué pasará cuando intentes compilar y ejecutar este código? class Base{ public final void amethod(){ System.out.println("amethod"); } } public class Fin extends Base{ public static void main(String argv[]){ Base b = new Base(); b.amethod(); } } 1) Un error en tiempo de compilación indicando que una clase con algún método final debe ser declarado final por si misma. 2) Un error en tiempo de compilación indicará que no puedes heredar de una clase con métodos final 3) Un error en tiempo de ejecución indicando que Base no esta definida como final 4) Éxito en la compilación y al ejecutarse mostrará en pantalla “amethod”. Pregunta 7) ¿Qué pasará cuando intentes compilar y ejecutar este código? public class Mod{ public static void main(String argv[]){ } public static native void amethod(); } 1) Error en la compilación: método nativo no puede ser static. 30 javaHispano. Tu comunidad, tu lenguaje. 2) Error en la compilación: método nativo debe retornar algún valor. 3) Compilación correcta pero habrá un error en tiempo de ejecución a menos que hallas hecho código para hacer útil el método nativo. 4) Compilación y ejecución sin error. Pregunta 8) ¿Qué pasará cuando intentes compilar y ejecutar este código? private class Base{} public class Vis{ transient int iVal; public static void main(String elephant[]){ } } 1) Un error en tiempo compilación: la clase Base no puede ser privada. 2) Un error en tiempo de compilación indicando que un entero no puede ser transient 3) Un error en tiempo de compilación indicando que transient no es un tipo de dato. 4) Un error en tiempo de compilación indicando que el método main esta deformado Pregunta 9) ¿Qué pasara cuando intentes compilar y ejecutar estos dos archivos en el mismo directorio? //Archivo P1.java package MyPackage; class P1{ void afancymethod(){ System.out.println("What a fancy method"); } } //Archivo P2.java public class P2 extends P1{ afancymethod(); } 1) Ambos se compilarán y P2 mostrará en pantalla “Wath a fancy method” cuando se ejecute. 2) No se compilará ninguno. 3) Ambos se compilarán pero P2 tendrá un error en tiempo de ejecución. 4) P1 se compilará limpiamente pero P2 tendrá un error al compilarse. 31 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 10) ¿Cuáles de las declaraciones siguientes son legales? 1) 2) 3) 4) public protected amethod(int i) public void amethod(int i) public void amethod(void) void public amethod(int i) Respuestas. Respuesta 1) 1) El código se compilara y se ejecutara, mostrando la cadena “My func” Una clase abstracta puede tener métodos no abstractos, pero cualquier clase que descienda de ella, debe implementar todos los métodos abstractos. Respuesta 2) 1) El código se compilara pero indicara que la función main no esta definida correctamente. La función main recibe un String en lugar de un arreglo de String que es lo correcto. Respuesta 3) 1) public 2) private 4) transient Aunque algunos textos usan la palabra “friendly” cuando tratan el tema de visibilidad, esta no es una palabra reservada de Java. Toma en cuenta que será casi seguro que el examen contenga problemas que te pidan que identifiques palabras claves de Java en una lista. Respuesta 4) 2) El compilador indicara que la clase Base no esta declarada como abstracta. El mensaje de error utilizando JDK 1.1. es: Abs.java:1: class Base must be declared abstract. It does not define void myfunc () from class Base. class Base{ 32 javaHispano. Tu comunidad, tu lenguaje. ^ 1 error Respuesta 5) 1) Para conseguir acceder a hardware desconocido por Java 3) Para escribir código optimizado para el mejor desempeño en un lenguaje como C/C++ Aunque la creación de código “PURE JAVA”es altamente conveniente, particularmente para permitir la independencia de plataforma, no debe de tomarse como una religión, y hay ocasiones en que se requiere el código nativo. Respuesta 6) 4)Éxito en la compilación y al ejecutarse mostrará en pantalla “amethod”. Este código llama a una versión de amethod de la clase Base. Si intentas implementar una versión sustituida del método en la clase Fin conseguirás un error en tiempo de compilación. Respuesta 7) 4)Compilación y ejecución sin error. Esta línea no es una llamada a un método nativo –solo se esta declarando- y por lo tanto no ocurre un error en la ejecución. Respuesta 8) 1)Un error en tiempo compilación: la clase Base no puede ser privada. Una clase de un nivel alto, al igual que una clase base, no puede ser declarada como privada Respuesta 9) 4)P1 se compilará limpiamente pero P2 tendrá un error al compilarse. Aunque P2 esta en el mismo directorio que P1, debido a que P1 fue declarado como parte de un paquete, P1 no esta visible para P2. Respuesta 10) 2)public void amethod(int i) Si pensaste que la opción 3 es legal utilizando como parámetro el argumento void, debes quitarte un poco de C/C++ de la cabeza. 33 Tutorial para el examen de certificación: SCJP 1.2 La opción 4 no es legal porque el tipo de datos debe aparecer después de cualquier especificador de acceso. Otras Fuentes para este tema. Este tema es cubierto en el tutorial de Sun como Modificadores de clases: http://java.sun.com/docs/books/tutorial/reflect/class/getModifiers.html Controlando el acceso a los miembros de una clase: http://java.sun.com/docs/books/tutorial/java/javaOO/accesscontrol.html Richard Baldwin Cubre este tema en http://www.Geocities.com/Athens/Acropolis/3797/Java040.htm Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec1.html#obj2 Bruce Eckel Thinking in Java Chapter 5 (aunque tiene la cabeza llena de C++ e insiste en incluir el modificador “friendly”) 34 javaHispano. Tu comunidad, tu lenguaje. Objetivo 3. Para una clase dada, determinar si se debe crear un constructor predefinido, y si ese es el caso, declarar el prototipo de ese constructor. Nota de este objetivo Éste es un objetivo pequeño pero oportuno, que se concentra en un aspecto fácilmente pasado por alto en Java ¿Qué es un constructor? Necesitas comprender el concepto de constructor para entender este objetivo. Brevemente, es el código que se ejecuta automáticamente cuando se crea la clase. Los constructores a menudo se usan para iniciar valores en la clase. Los constructores tienen el mismo nombre de a clase y no retornan ningún valor. En el examen puedes encontar preguntas sobre métodos que tengan el mismo nombre de la clase pero que retornan algún tipo, como int o String. Ten cuidado y asegúrate que cualquier método que asumas que es un constructor no debe retornar algún tipo de dato. Aquí hay un ejemplo de una clase con un constructor que muestra "Greetings from Crowle" cuando se crea una instancia de la clase. public class Crowle{ public static void main(String argv[]){ Crowle c = new Crowle(); } Crowle(){ System.out.println("Greetings from Crowle"); } } ¿Cuándo proporciona Java el constructor predefinido? Si no defines específicamente algún constructor, el compilador inserta “fuera de escena” un constructor invisible con cero parámetros, frecuentemente esto solo es de importancia teórica, pero lo verdaderamente importante es que solo recibes el constructor predefinido con cero parámetros si no creas algún constructor propio. 35 Tutorial para el examen de certificación: SCJP 1.2 Si creas tus propios constructores, Java ya no te proporciona el constructor predefinido con cero parámetros. Así que tienes que especificarlo. En cuanto creas algún constructor propio, pierdes el constructor predefinido con cero parámetros. Si intentas crear una instancia de una clase, sin pasarle algún parámetro (invocando la clase con el constructor con cero parámetros), obtendrás un error. Así, cuando crees algún constructor para una clase necesitarás crear el constructor con cero parámetros. Esta es una de las razones por las que generadores de código como Borland/Inprise o JBuilder, crean un constructor con cero parámetros cuando generan el esqueleto de una clase. El ejemplo siguiente muestra código que no compilará. Cuando el compilador verifica la creación de la instancia c de la clase Base, inserta una llamada al constructor con cero parámetros. Debido a que la clase Base tiene un constructor que recibe un entero, el constructor con cero parámetros ya no esta disponible. Esto se puede arreglar creando en la clase Base un constructor con cero parámetros que “no haga nada”. //Advertencia: no se compilará class Base{ Base(int i){ System.out.println("single int constructor"); } } public class Cons { public static void main(String argv[]){ Base c = new Base(); } } //Esto si se compilará class Base{ Base(int i){ System.out.println("single int constructor"); } Base(){} } public class Cons { public static void main(String argv[]){ Base c = new Base(); } } 36 javaHispano. Tu comunidad, tu lenguaje. El prototipo del constructor predefinido El objetivo te pide que estes consciente del prototipo del constructor predefinido. Naturalmente no debe tener ningún parámetro. La predefinición más evidente es no tener ningún especificador de alcance, pero puedes definir un constructor como público o protegido. Los constructores no pueden ser nativos, abstractos, estáticos, sincronizados ni finales. Ese pedazo de información se derivó directamente de un mensaje de error del compilador. Parece que la calidad de los mensajes de error están mejorando con las nuevas versiones de Java. He oído que el nuevo compilador de Java de IBM tiene un buen informador de errores. Podrían aconsejarte que tuvieras disponible más de una versión del compilador de Java para verificar tu código y buscar los errores. Ejercicios Ejercicio 1) Crea una clase llamada Salwarpe con un método llamado hello que muestre la cadena “hello”. En el método main de la clase crea una instancia de sí misma llamada s1 y llama al método hello desde la instancia. Compila y ejecuta el programa para que puedas ver el resultado. Ejercicio 2) Usando todavía el archivo Salwarpe.java comenta las líneas que crean la instancia y llaman al método hello. Crea un constructor publico para la clase que tome un parámetro entero y muestre el valor del entero. Crea una instancia de la clase llamada s2 proporciónale el valor de 99 al constructor. Compila y ejecuta el programa para que puedas ver mostrado el resultado. Ejercicio 3) Descomenta las líneas que crean la instancia s1 y modifica el programa para que compile, ejecute y muestre ambos la cadena Hello y el numero 99. Solución sugerida para el ejercicio 1. public class Salwarpe { public static void main(String argv[]){ Salwarpe s1 = new Salwarpe(); s1.hello(); } public void hello(){ System.out.println("Hello"); } 37 Tutorial para el examen de certificación: SCJP 1.2 } Solución sugerida para el ejercicio 2. public class Salwarpe { public static void main(String argv[]){ //Salwarpe s1 = new Salwarpe(); //s1.hello(); Salwarpe s2 = new Salwarpe(99); } public void hello(){ System.out.println("Hello"); } public Salwarpe(int i){ System.out.println(99); } } Solución sugerida para el ejercicio 3. public class Salwarpe { public static void main(String argv[]){ Salwarpe s1 = new Salwarpe(); s1.hello(); Salwarpe s2 = new Salwarpe(99); } public void hello(){ System.out.println("Hello"); } public Salwarpe(int i){ System.out.println(99); } public Salwarpe(){} } Observa como fue necesario crear el constructor con cero parámetros para este último ejercicio. Una vez que has creado algún constructor para una clase, Java ya no te proporciona “tras escena” el constructor con cero parámetros que estaba disponible en el ejercicio 1. 38 javaHispano. Tu comunidad, tu lenguaje. Preguntas Pregunta 1) Dada la siguiente definición de clase: class Base{ Base(int i){} } class DefCon extends Base{ DefCon(int i){ //XX } } ¿Cuál se las líneas siguientes será individualmente valida si se sustituye por la línea marcada con //XX? 1) super(); 2) this(); 3) this(99); 4)super(99); Pregunta 2) Dada la siguiente definición de clase: public class Crowle{ public static void main(String argv[]){ Crowle c = new Crowle(); } Crowle(){ System.out.println("Greetings from Crowle"); } } ¿Cuál es el tipo de dato retornado por el constructor? 1) null 2) integer 3) String 4) no es retornado algún tipo de dato Pregunta 3) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? 39 Tutorial para el examen de certificación: SCJP 1.2 public class Crowle{ public static void main(String argv[]){ Crowle c = new Crowle(); } void Crowle(){ System.out.println("Greetings from Crowle"); } } 1)Se compilara y mostrará la cadena “Greetings from Crowle”. 2)Error en tiempo de compilación: un constructor no puede retornar un tipo de dato. 3)Se compilara y mostrará la cadena “void”. 4)Se compilara y no mostrará nada al ejecutarse. Pregunta 4) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? class Base{ Base(int i){ System.out.println("Base"); } } class Severn extends Base{ public static void main(String argv[]){ Severn s = new Severn(); } void Severn(){ System.out.println("Severn"); } } 1)Se compilará al ejecutarse y mostrará la cadena “Seven”. 2)Error en tiempo de compilación. 3)Se compilará y no mostrará nada al ejecutase 4)Se compilará y mostrará la cadena “Base” Pregunta 5) ¿Cuáles de las siguientes declaraciones son verdaderas? 1)El constructor predefinido retorna un tipo void. 2)El constructor predefinido toma como parámetro un valor tipo void. 3)El constructor predefinido no toma parámetros. 40 javaHispano. Tu comunidad, tu lenguaje. 4)El constructor predefinido no se crea si la clase contiene algún constructor propio Respuestas Respuesta 1) 4)super (99) Debido a que la clase Base tiene definido un constructor, el compilador no insertará el constructor predefinido con cero parámetros. Por consiguiente, llamar a super() causará un error. Una llamada a this() es un intento de llamar a un constructor con cero parámetros inexistente en la clase actual. La llamada a this(99) causará una referencia circular y causará un error en tiempo de compilación. Respuesta 2) 4) no es retornado algún tipo de dato Debe ser bastante obvio de que ningún tipo de dato se retorna, así como por definición los constructores no tienen tipos de datos Respuesta 3) 4)Se compilara y no mostrará nada al ejecutarse. Debido a que el método Crowle retorna un tipo de dato no es un constructor, por consiguiente la clase se compilará y al ejecutarse el método Crowle no se ejecutará – pues no ha sido llamado-. Respuesta 4) 2)Error en tiempo de compilación. Ocurre un error cuando la clase Severn intenta llamar al constructor con cero parámetros en la clase Base. Respuesta 5) 3)El constructor predefinido no toma parámetros. 4)El constructor predefinido no se crea si la clase contiene algún constructor propio 41 Tutorial para el examen de certificación: SCJP 1.2 Otras Fuentes de este tema: Esta tema es cubierto en el Tutorial de Sun : http://java.sun.com/docs/books/tutorial/java/javaOO/constructors.html Richard Baldwin cubre este tema en http://www.Geocities.com/Athens/Acropolis/3797/Java042.htm#default constructor Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec1.html#obj3 Bruce Eckel Thinking In Java Capítulo 4 42 javaHispano. Tu comunidad, tu lenguaje. Objetivo 4 Establecer los tipos de datos validos de retorno para cualquier método, dada las declaraciones de todos lo métodos relacionados con este o con la clase padre. Nota de este objetivo La frase del objetivo parece ser bastante obscura. Parece estar pidiéndote que entiendas la diferencia entre sobrecarga y sobrescritura. Para aprovechar mejor este objetivo necesitas una comprensión básica sobre la sobrecarga y sobrescritura de métodos. Esto se ve en la: Sección 6: Orientación a Objetos, sobrecarga y sobreescritura en tiempo de ejecución Métodos en la misma clase Si dos o más métodos en la misma clase tienen el mismo nombre, se dice que el método esta sobrecargado. Puedes tener dos métodos en una clase con el mismo nombre pero deben tener parámetros de diferente tipo y en diferente orden. Es el orden de los parámetros y los tipos de los mismos los que distinguen entre cualquiera de dos versiones de un método sobrecargado a uno en especial. El tipo de dato de retorno no contribuye a la distinción entre métodos. El código siguiente generará un error en tiempo de compilación, el compilador observa en amethod un intento por definir el mismo método dos veces. Causa un error que dirá algo como lo siguiente: method redefined with different return type: void amethod(int) was int amethod(int) class Same{ public static void main(String argv[]){ Over o = new Over(); int iBase=0; o.amethod(iBase); } //Las siguientes definiciones generan un error al compilar public void amethod(int iOver){ System.out.println("Over.amethod"); } public int amethod(int iOver){ System.out.println("Over int return method"); return 0; } 43 Tutorial para el examen de certificación: SCJP 1.2 } El tipo de dato de retorno no contribuye a realizar la distinción entre un método y otro. Métodos en una subclase. Puedes sobrecargar un método en una subclase. Todo lo que se requiere es que la nueva versión debe tener parámetros de diferente tipo y en diferente orden. Los nombres de los parámetros o el tipo de dato retornado por el método no se toman en cuenta para la sobrecarga de métodos. Si vas a sobrescribir un método, por ejemplo para reemplazar su funcionalidad completamente en una subclase, la versión sobrescrita del método debe tener la misma definición que la versión de la clase base de la que esta descendiendo. Esto incluye el tipo de dato de retorno. Si creas un método en una subclase con el mismo nombre y definición pero retorna un tipo de dato distinto, generarás el mismo mensaje de error que en el ejemplo anterior. Es decir: method redefined with different return type: void amethod(int) was int amethod(int) El compilador lo ve como un intento fallido por sobrecargar el método en lugar de verlo como una sobrescritura del método. Preguntas Pregunta 1) Dada la siguiente definición de clase: public class Upton{ public static void main(String argv[]){ 44 javaHispano. Tu comunidad, tu lenguaje. } public void amethod(int i){} //Here } ¿Cuáles de las líneas siguientes son válidas para ser colocadas después del comentario //Here? 1) public int amethod(int z){} 2) public int amethod(int i,int j){return 99;} 3) protected void amethod(long l){ } 4) private void anothermethod(){} Pregunta 2) Dada la siguiente definición de clase: class Base{ public void amethod(){ System.out.println("Base"); } } public class Hay extends Base{ public static void main(String argv[]){ Hay h = new Hay(); h.amethod(); } } ¿Cuál de los siguientes método puede ser de la clase Hay para que compile y provoque que el programa muestre la cadena “Hay” ? 1) public int amethod(){ System.out.println("Hay");} 2) public void amethod(long l){ System.out.println("Hay");} 3) public void amethod(){ System.out.println("Hay");} 4) public void amethod(void){ System.out.println("Hay");} Respuestas Respuesta 1) 45 Tutorial para el examen de certificación: SCJP 1.2 2) public int amethod(int i, int j) {return 99;} 3) protected void amethod (long l){} 4) private void anothermethod(){} La opción 1 no compilara por dos razones. La primera es que obviamente exige que un entero sea retornado- y como podemos ver hace falta un return -. La otra es que es evidente un intento por redefinir un método dentro de la misma clase. El cambio de nombre del parámetro de i a z no tiene efecto y un método no puede ser sobrescrito dentro de la misma clase. Respuesta 2) 3) public void amethod(){ System.out.println("Hay");} La opción 3 es una sobrescritura del método de la clase Base, para alguna invocación de esta versión se utilizarán cero parámetros La opción 1 generará un error que indicará que se está intentando redefinir un método con un diferente tipo de dato de retorno. Aunque la opción 2 compilará, la llamada a amethod invocará el método de la clase Base y la salida sea la cadena “Base”. La opción 4 fue diseñada para sorprender a aquellos con una cabeza llena de C/C++, en Java no hay tal cosa: la utilización de void para indicar que no hay parámetros. Otras Fuentes de este tema: Jyothi Krishnan http://www.geocities.com/SiliconValley/Network/3693/obj_sec1.html#obj4 En esta liga Jyothi sugiere que vayas al objetivo 19: http://www.geocities.com/SiliconValley/Network/3693/obj_sec6.html#obj19 46 javaHispano. Tu comunidad, tu lenguaje. Sección 2. Control de Flujo y Manejo de Excepciones. Objetivo 1. Escribir código que use sentencias if y switch e identificar los tipos validos de argumentos para estas sentencias. La sentencia if/else. Ls sentencia if/else funciona en Java tal como podrías esperar que funcione en otros lenguajes. las sentencias switch/case tienen unas peculiaridades. La sintaxis para la sentencia if/else es: if(condicion tipo_boolean){ //si el boolean es verdadero se hace esto }else { //sino, se hace esto } Java no tiene la palabra clave "then" como en Visual Basic. Las llaves son un indicador general en Java de una declaración compuesta que permite ejecutar múltiples líneas de código como resultado de alguna evaluación. Esto se conoce como bloque de código. La parte del "else" siempre es opcional. Una idiosincrasia de Java es que la sentencia if debe tomar un valor del tipo boolean. No puedes usar la convención de C/C++ donde cualquier valor distinto de cero es verdadero y 0 es falso. Así , en Java lo siguiente simplemente no compilará: int k =-1; if(k){//no se compilará! System.out.println("do something"); } Esto debido a que debes efectuar explícitamente una pregunta sobre la variable k que devuelva un valor tipo boolean, como en el ejemplo siguiente: if(k == -1){ System.out.println("do something"); //compilación OK! } Así como en C/C++ puedes no utilizar las llaves: 47 Tutorial para el examen de certificación: SCJP 1.2 boolean k=true; if(k) System.out.println("do something"); Esto a veces es considerado un mal estilo de programación, porque si modificas el código después para incluir declaraciones adicionales estas quedarán fuera del bloque condicional (qué solo abarca una línea por no utilizar llaves) Así: if(k) System.out.println("do something"); System.out.println("also do this"); La segunda línea siempre se ejecutará. La Sentencia switch La opinión de Peter van der Lindens sobre la sentencia switch se resume cuando dice: "Muerte a la sentencia switch" Este es un asunto al que le debes prestar atención extra. El argumento de una sentencia switch debe ser un byte, char, short o un int. Puedes encontrar un ejercicio en el examen que use un float o un long como argumento en una sentencia switch. Una pregunta bastante común parece estar sobre el uso de la sentencia break en el proceso descendente a través de una sentencia switch. Aquí hay un ejemplo de este tipo de pregunta. int k=10; switch(k){ case 10: System.out.println("ten"); case 20: System.out.println("twenty"); } El sentido común nos indicaría que después de ejecutar las instrucciones que siguen la primera sentencia case, y habiéndose encontrado con otra sentencia case, el compilador terminaría su descenso a través de la sentencia switch. Sin embargo, por razones mejor conocidas solo por los diseñadores del lenguaje, una sentencia case solo se detiene cuando se encuentran con una sentencia break. Como resultado, en el ejemplo anterior las cadenas: "ten" y "twenty" son mostradas. Otra pequeña peculiaridad que puede surgir en las preguntas es donde colocar la sentencia default. 48 javaHispano. Tu comunidad, tu lenguaje. La cláusula default no necesita ir al final de una sentencia case. El lugar convencional para la sentencia default está al final de las opciones case. Así normalmente el código se escribirá como sigue: int k=10; switch(k){ case 10: System.out.println("ten"); break; case 20: System.out.println("twenty"); break; default: System.out.println("This is the default output"); } Esta aproximación refleja la manera en que piensa la mayoría de las personas. Una vez que se han probado todas las posibilidades, se realiza la sentencia default. Sin embargo, el código de una sentencia switch con la sentencia default hasta arriba es sintácticamente correcto, pero no aconsejable. int k=10; switch(k){ default: //Coloca default hasta abajo y no acá System.out.println("This is the default output"); break; case 10: System.out.println("ten"); break; case 20: System.out.println("twenty"); break; } Argumentos validos para las sentencias if y switch. Como mencioné anteriormente una sentencia if puede tomar solo un valor tipo boolean y una sentencia switch puede tomar solamente un byte, char, short o un int. El operador ternario ? 49 Tutorial para el examen de certificación: SCJP 1.2 Algunos programadores dicen que el operador ternario es útil. Yo no considero para que. No se menciona específicamente en los objetivos así que por favor háganme saber si aparece en su examen. Pregunta 1) ¿Qué pasará cuando intentes compilar y ejecutar el siguiente código? public class MyIf{ boolean b; public static void main(String argv[]){ MyIf mi = new MyIf(); } MyIf(){ if(b){ System.out.println("The value of b was true"); } else{ System.out.println("The value of b was false"); } } } 1) Error en tiempo de compilación: la variable b no fue inicializada 2) Error en tiempo de compilación: el parámetro para la sentencia if debe evaluar a un boolean 3) Error en tiempo de compilación: no se puede crear y asignar un valor simultáneamente a un boolean 4) Compilación y ejecución mostrando el segundo mensaje. Pregunta 2) ¿Qué pasará cuando intentes compilar y ejecutar este código? public class MyIf{ public static void main(String argv[]){ MyIf mi = new MyIf(); } MyIf(){ boolean b = false; 50 javaHispano. Tu comunidad, tu lenguaje. if(b=false){ System.out.println("The value of b is"+b); } } } 1) Error en tiempo de ejecución: un boolean no puede agregarse con el operador + 2) Error en tiempo de compilación: el parámetro para la sentencia if debe evaluar a un boolean 3) Error en tiempo de compilación: no se puede crear y asignar un valor simultáneamente a un boolean 4) Compilación y ejecución sin salida. Pregunta 3) ¿Qué pasará cuando intentes compilar y ejecutar este código? public class MySwitch{ public static void main(String argv[]){ MySwitch ms= new MySwitch(); ms.amethod(); } public void amethod(){ } char k=10; switch(k){ default: System.out.println("This is the default output"); break; case 10: System.out.println("ten"); break; case 20: System.out.println("twenty"); break; } } 1) Ninguna de estas opciones 2) Error en tiempo de compilación: el valor para el switch debe ser de tipo entero 3) Se Compilará y ejecutará mostrando "This is the default output" 4) Se Compilará y ejecutará mostrando "ten" 51 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 4) ¿Qué pasará cuando intentes compilar y ejecutar el siguiente código? public class MySwitch{ public static void main(String argv[]){ MySwitch ms= new MySwitch(); ms.amethod(); } public void amethod(){ } int k=10; switch(k){ default: //Put the default at the bottom, not here System.out.println("This is the default output"); break; case 10: System.out.println("ten"); case 20: System.out.println("twenty"); break; } } 1) Ninguna de estas opciones 2) Error en tiempo de compilación: el valor para el switch debe ser de tipo entero 3) Se Compilará y ejecutará mostrando "This is the default output" 4) Se Compilará y ejecutará mostrando "ten" Pregunta 5) ¿ De las siguientes líneas, cual podría usarse como parámetro para una declaración switch? 1) byte b=1; 2) int i=1; 3) boolean b=false; 4) char c='c'; Respuestas Respuesta 1) 4) Compilación y ejecución mostrando el segundo mensaje. 52 javaHispano. Tu comunidad, tu lenguaje. Debido a que la variable boolean b se creó al nivel de clase, no necesita ser inicializada explícitamente y en cambio toma el valor predefinido de un boolean que es falso. Una sentencia if debe evaluar un valor boolean y así la variable b cumple este criterio. Respuesta 2) 4) Compilación y ejecución sin salida. Debido a que la variable b es un boolean no habrá ningún error causado por la sentencia if. Si b fuera de cualquier otro tipo de dato hubiera ocurrido un error cuando intentaras realizar una asignación en lugar de una comparación. La expresión: if(b=false) es una asignación y normalmente representa un error del programador. A menudo el programador tenía en mente decir: if (b==false) Si el tipo de dato de b hubiera sido cualquiera menos boolean habría ocurrido un error al compilarse. El requisito para la sentencia if es que devuelva un boolean y debido a que: if (b=false) devuelve un boolean (false), esto es aceptable (pero inservible). Respuesta 3) 4) Se Compilará y ejecutará mostrando "ten" Respuesta 4) 1) Ninguna de estas opciones Debido a la falta de una sentencia break después de la línea: System.out.println("ten"); la salida real será: "ten" seguido de "twenty" Respuesta 5) 53 Tutorial para el examen de certificación: SCJP 1.2 1) byte b=1; 2) int i=1; 4) char c='c'; Una sentencia switch puede tomar únicamente a un byte, char short o a un int como parámetro. Otras Fuentes para este tema: El tutorial de Sun http://java.sun.com/docs/books/tutorial/java/nutsandbolts/while.html Richard Baldwin Cubre este tema en: http://www.Geocities.com/Athens/Acropolis/3797/Java026.htm#the if-else statement Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec2.html#obj5 54 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2. Escribir código que use todas las formas de ciclos incluyendo el etiquetado y desetiquetado, el uso de break y continue y conocer los valores tomados por las variables contadoras de los ciclos durante y después de la ejecución de cada ciclo. La sentencia for El método más común para enciclar es usar la sentencia for. Como en C++ y a diferencia de C, la variable que controla el ciclo puede crearse e inicializarse dentro de la sentencia for. Así public class MyLoop{ public static void main(String argv[]){ MyLoop ml = new MyLoop(); ml.amethod(); } public void amethod(){ for(int K=0;K<5;K++){ System.out.println("Outer "+K); for(int L=0;L<5;L++) {System.out.println("Inner "+L);} } } } Este código iterara 5 veces en el ciclo interno por cada iteración del ciclo externo. Así la salida será: Outer Inner Inner Inner Inner inner Outer Inner Inner 0 0 1 2 3 4 1 0 2 Etc, etc. El sentencia for es el equivalente del ciclo for/next en Visual Basic. Debes tomar en cuanta que la sintaxis es: for(inicialización; expresión condicional; incremento) 55 Tutorial para el examen de certificación: SCJP 1.2 La expresión condicional debe ser una expresión tipo boolean de una manera similar a la sentencia if. En el código ejemplo sobre la sentencia for, esta fue seguida por un bloque del código señalado por llaves. De la misma manera que un sentencia if no exige un bloque de código, puedes tener una sentencia for que simplemente maneje una línea, de esta manera: for(int i=0;i<5;i++) System.out.println(i); Nota que en ninguna de las versiones se termina la línea del for con punto y coma. Si lo haces, el ciclo for solo iterara en esa línea hasta que la condición se cumpla y entonces el programa continuará en "línea recta". No tienes que crear variable de inicialización(como en este caso) dentro del ciclo for, pero si lo haces significará que la variable saldrá del alcance en cuanto se termine el ciclo. Esto puede ser considerado una ventaja por lo que se refiere a mantener el alcance de las variables tan pequeño como sea posible. Los ciclos while y do, nada inesperado. Los ciclos while y do funcionan mucho más de lo que esperarías de su equivalente en otros lenguajes. Así un while se ejecutará cero o más veces según la condición y el do se ejecutará una o más veces. La sintaxis ara el ciclo while es: while(condición){ cuerpodelciclo; } La condición es una expresión tipo boolean parecida al de una sentencia if. De nuevo no puedes usar la convención de C/C++ donde 0 es verdadero y cualquier otro valor es falso. De manera que podrías crear un ciclo while de la siguiente manera: while(i<4){ i++; System.out.println("Loop value is :"i); } Nota que si la variable i valiera 4 o más cuando se alcanzara la sentencia while no se produciría ningún resultado. Por contraste un ciclo do siempre se ejecutará, al menos, una vez. 56 javaHispano. Tu comunidad, tu lenguaje. Así con el código siguiente siempre conseguirás por lo menos una ejecución del cuerpo del ciclo no importando el valor de la variable i al entrar en el ciclo. do{ System.out.println("value of : "+i); } while (i <4); Muchos programadores intentan usar el ciclo for en lugar del ciclo while cuando pueden concentrar en una sola línea la creación, inicialización, prueba e incremento del contador. La sentencia goto, ¿ciencia o religión? Los diseñadores de Java decidieron que ellos estaban de acuerdo con el gurú de la programación Edsger Dijkstra quien escribió un artículo famoso titulado "Goto considered harmful". Debido a que el uso indistinto de sentencias goto pueden dar como resultado un código, cual espagueti, difícil de mantener, su uso se ha reducido y es considerado como un mal estilo de programación. Hay situaciones donde sería útil y para ayudar en esas situaciones Java ofrece las versiones etiquetadas y desetiquetadas de las palabras clave break y continue. Break y Continue Estas sentencias te permiten interrumpir condicionalmente ciclos. Sin embargo, no te permiten saltar a otra parte de programa. Es probable que el examen incluya este tema en la forma de un conjunto de ciclos anidados. Tienes que indicar que números se mostrarán, al ejecutar el código, antes de que termine el ciclo debido al mecanismo de la sentencia break. Este es un ejemplo de la clase de preguntas irritables que probablemente te encuentres en el examen. public class Br{ public static void main(String argv[]){ Br b = new Br(); b.amethod(); } public void amethod(){ for(int i=0;i <3;i ++){ System.out.println("i"+i+"\n"); outer://<==Point of this example if(i>2){ break outer;//<==Point of this example 57 Tutorial para el examen de certificación: SCJP 1.2 }//End of if for(int j=0; j <4 && i<3; j++){ System.out.println("j"+j); }//End of for }//End of for }//end of Br method } Tienes que elegir que combinación de letras se mostrará con la ejecución del código. A propósito el código “\n” (en caso de que aún no lo sepas) significa que se mostrará una línea en blanco. Saltar a una Etiqueta. Es deseable a menudo, saltar de un ciclo interno a un ciclo externo dada una condición. Puedes hacer esto etiquetando las sentencias break y continue. Una etiqueta es simplemente una palabra no reservada seguida de dos puntos. Poniendo el nombre de la etiqueta después del break o continue tu código saltará hasta la etiqueta. Esto es conveniente para hacer condicional una parte de un ciclo. Por supuesto, puedes hacer lo mismo con una sentencia if, pero una sentencia break es más conveniente. No puedes saltar a otro ciclo o a otro método pero una vez terminado el ciclo actual es muchas veces muy útil. La sentencia break abandona completamente el proceso del ciclo activo, la sentencia continue abandona únicamente el proceso de la iteración activa del ciclo activo Tomando en cuenta el siguiente código: public class LabLoop{ public static void main(String argv[]){ LabLoop ml = new LabLoop(); ml.amethod(); } public void amethod(){ outer: for(int i=0;i<2;i++){ for(int j=0;j<3;j++){ if(j>1) //prueba con break en lugar de continue continue outer; System.out.println("i "+ i + " j "+j); 58 javaHispano. Tu comunidad, tu lenguaje. } } }//fin del for externo outer System.out.println("Continuing"); }//fin de amethod Se mostrará lo siguiente: i 0 j 0 i 0 j 1 i 1 j 0 i 1 j 1 Continuing Si sustituyeras la sentencia continue por break, el contador i se detendría en cero ya que el proceso del ciclo externo se abandonaría en lugar de simplemente continuar al siguiente incremento. Pregunta 1) ¿Que pasará cuando intentes compilar y ejecutar el siguiente código en un método? for(int i=0;i<5;){ System.out.println(i); i++; continue; } 1) 2) 3) 4) Error en tiempo de compilación, sentencia for incorrecta. Error en tiempo de compilación: continue dentro de un for Error en tiempo de ejecución: sentencia continue no alcanzada. Se compilará y se ejecutará mostrando los valores del 0 al 4. Pregunta 2) ¿Qué pasará cuando intentes compilar y ejecutar el siguiente código? public class LabLoop{ public static void main(String argv[]){ 59 Tutorial para el examen de certificación: SCJP 1.2 LabLoop ml = new LabLoop(); ml.amethod(); mainmethod: System.out.println("Continuing"); } } public void amethod(){ outer: for(int i=0;i<2;i++){ for(int j=0;j<3;j++){ if(j>1) break mainmethod; System.out.println("i "+ i + " j "+j); } }//End of outer for } 1) i 0 j 0 i 0 j 1 Continuando 2) i 0 j 0 i 0 j 1 i 1 j 0 i 1 j 1 Continuando 3) Error en tiempo de Compilación. 4) i 0 j 0 i 0 j 1 i 1 j 0 i 1 j 1 i 2 j 1 Continuando Pregunta 3) ¿Qué pasará cuando intentes compilar y ejecutar el siguiente código? public void amethod(){ outer: for(int i=0;i<2;i++){ 60 javaHispano. Tu comunidad, tu lenguaje. for(int j=0;j<2;j++){ System.out.println("i="+i + " j= "+j); if(i >0) break outer; } } System.out.println("Continuing with i set to ="+i); } 1)Error en tiempo de compilación. 2) i=0 j= 0 i=0 j= 1 i=1 j= 0 3) i=0 i=0 i=1 i=2 j= j= j= j= 0 1 0 0 4) i=0 j= 0 i=0 j= 1 Pregunta 4) ¿Qué pasará cuando intentes compilar y ejecutar el siguiente código? int i=0; while(i>0){ System.out.println("Value of i: "+i); } do{ System.out.println(i); } while (i <2); 1) Value of i: 0 Seguido de 0 1 2 2) 0 1 2 61 Tutorial para el examen de certificación: SCJP 1.2 3) Value of i: 0 Seguido de una secuencia de 0’s 4) Mostrará una secuencia de 0’s Respuestas Respuesta 1) 4)Se compilará y se ejecutará mostrando los valores del 0 al 4. Esta es una versión un tanto extraña de la sentencia for, pero es totalmente valida. Respuesta 2) 3) Error en tiempo de Compilación. No puede saltar arbitrariamente a otro método, esto traerían consigo todo los males ya manifestados sobre la sentencia goto. Respuesta 3) 1)Error en tiempo de compilación. Realmente esta no es una pregunta acerca del funcionamiento de break y continue. Este código no se compilará porque la variable utilizada no existe fuera el ciclo for. Por lo tanto, la sentencia System.out.println, causará un error en tiempo de compilación. Respuesta 4) 4)Mostrará una secuencia de 0’s El ciclo while no ejecutará nada en absoluto si la condición no es verdadera en su primera iteración y debido a que no hay ningún incremento de algún valor en el ciclo do se ejecutará infinitamente. Otras Fuentes para este tema El tutorial de Sun http://java.sun.com/docs/books/tutorial/java/nutsandbolts/while.html Jyothi Krishnan http://www.geocities.com/SiliconValley/Network/3693/obj sec2.html#obj6 62 javaHispano. Tu comunidad, tu lenguaje. Richard Baldwin http://www.Geocities.com/Athens/Acropolis/3797/Java026.htm#flow of control Bruce Eckel http://codeguru.earthweb.com/java/tij/tij0045.shtml#Heading131 63 Tutorial para el examen de certificación: SCJP 1.2 Objetivo 3. Escribir código que haga un uso apropiado de las excepciones y las cláusulas del manejo de excepciones (try, catch y finally), declarar métodos y sobreescribir métodos que lanzen excepciones. Las sentencias try y catch son parte del manejo de excepciones construido en Java. Ni C/C++ " ni Visual Basic tienen un equivalente directo a las excepciones construidas en Java. C++ soporta excepciones pero es de manera opcional, y Visual Basic soporta la captura de errores con On Error/Goto. Las excepciones en Java son construidas como parte del lenguaje. Por ejemplo si estas realizando instrucciones de Entrada y Salida (I/O) debes manipular excepciones. Puedes colocar por supuesto un manejador nulo que no haga nada. Lo siguiente es una pequeña parte del código que he acostumbrado a usar con Borland/Inprise JBuilder para detener temporalmente la salida de la consola y esperar que cualquier tecla sea presionada import java.io.*; public class Try{ public static void main(String argv[]){ Try t = new Try(); t.go(); }//End of main public void go(){ try{ InputStreamReader isr = new InputStreamReader(System.in); BufferedReader br = new BufferedReader(isr); br.readLine(); } catch(Exception e){ /*No se hace nada cuando ocurre la excepción*/ } //End of try System.out.println("Continuing"); }//End of go } En este caso no se hace nada cuando ocurre un error, pero el programador aun reconoce que puede ocurrir un error. Si quitas las cláusulas try y catch el código simplemente no compilará. El compilador sabe que los métodos de I/O pueden causar excepciones y exige código para manejar esas posibles excepciones. Comparando con Visual Basic y C/C++ Esto es un poco más riguroso que en Visual Basic o en C/C++ que te permite lanzar "sucia y rápidamente" programas que fingen que están en un mundo donde los errores no 64 javaHispano. Tu comunidad, tu lenguaje. ocurren. No es exactamente una programación esclavizada y disciplinada, las excepciones sólo te animan persuasivamente a "hacer las cosas correctamente". La Cláusula finally Una de las rarezas que probablemente te preguntarán en el examen, es bajo qué circunstancias se ejecuta la cláusula finally de un bloque try/catch. La respuesta mas corta a esto es que la cláusula finally casi siempre se ejecuta, incluso cuando podrías pensar que no se ejecutaría. Hemos dicho que, la trayectoria de ejecución de las sentencias try/catch/finally es algo con lo que realmente necesitas jugar para convencerte de lo qué es lo que pasa bajo qué circunstancias. la cláusula finally de un bloque try/catch siempre se ejecutará, aun cuando haya cualquier sentencia de retorno en la parte try/catch Uno de las pocas ocasiones donde la cláusula finally no se ejecutará es si hay una llamada a: System.exit(0); El examen tiende a no intentar sorprenderte con esta excepción a la regla. El examen es más propenso a dar ejemplos que incluyan declaraciones de retorno para engañarte y hacerte pensar que el código retornará sin ejecutar la sentencia finally. No te despistes, la cláusula finally casi siempre se ejecutará. La cláusula try/catch debe capturar los errores en el orden natural de su jerarquía. Así, no puedes intentar capturar la excepción Exception, que es mas general, antes de intentar capturar la excepción IOException que es más específica.(NT: en otras palabras, debes de intentar capturar primero las más especificas y luego las más generales) El siguiente código no se compilará: try{ DataInputStream dis = new DataInputStream(System.in); dis.read(); }catch (Exception ioe) { }catch (IOException e) { //error en tiempo de compilación }finally{} 65 Tutorial para el examen de certificación: SCJP 1.2 Este código generará un mensaje de error en tiempo de compilación: IOException nunca se alcanzará. Sobrescribiendo métodos que lanzan excepciones Un método sobrescrito en una subclase solo puede lanzar excepciones declaradas en la clase padre o excepciones que sean hijas de las excepciones declarados en la clase padre. Esto es sólo valido para métodos sobrescritos y no para métodos sobrecargados. Así, si un método tiene exactamente el mismo nombre y argumentos puede lanzar únicamente excepciones declaradas en la clase padre, o excepciones que son hijos de excepciones de la declaración de la clase padre. No obstante puede lanzar ninguna o pocas excepciones. Así el ejemplo siguiente no se compilará: import java.io.*; class Base{ public static void amethod()throws FileNotFoundException{} } public class ExcepDemo extends Base{ //No se compilara, la excepción que se intenta lanzar no esta //en la versión del método amethod en la clase Padre(Base) public static void amethod()throws IOException{} } Si fuera el método de la clase padre el que estuviera lanzando a IOException y el método en la clase hijo el que estuviera lanzando a FileNotFoundException el código si compilaría(NT: debido a que FileNotFoundException es subclase de IOException). De nuevo, recuerda que esto sólo se aplica a los métodos sobrescritos, no hay ninguna regla similar para los métodos sobrecargados. Un método sobrescrito en una subclase también puede lanzar excepciones. Pregunta 1) 66 javaHispano. Tu comunidad, tu lenguaje. ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? import java.io.*; class Base{ public static void amethod()throws FileNotFoundException{} } public class ExcepDemo extends Base{ public static void main(String argv[]){ ExcepDemo e = new ExcepDemo(); } public static void amethod(){} protected ExcepDemo(){ try{ DataInputStream din = new DataInputStream(System.in); System.out.println("Pausing"); din.readChar(); System.out.println("Continuing"); this.amethod(); }catch(IOException ioe) {} } } 1)Error en tiempo de compilación causado por el constructor protegido. 2)Error en tiempo de compilación causado por que amethod ha declarado excepciones. 3)Error al ejecutar: amethod no ha declarado excepciones. 4)Se compilara y se ejecutara mostrando: “Pausing” y “Continuing” después de que una tecla es pulsada. Pregunta 2) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? import java.io.*; class Base{ public static void amethod()throws FileNotFoundException{} } public class ExcepDemo extends Base{ public static void main(String argv[]){ ExcepDemo e = new ExcepDemo(); } public static void amethod(int i)throws IOException{} 67 Tutorial para el examen de certificación: SCJP 1.2 private ExcepDemo(){ try{ DataInputStream din = new DataInputStream(System.in); System.out.println("Pausing"); din.readChar(); System.out.println("Continuing"); this.amethod(); }catch(IOException ioe) {} } } 1)Error en tiempo de compilación causado por el constructor privado. 2)Error en tiempo de compilación: amethod ha declarado excepciones que no están en la versión de la clase base. 3)Error al ejecutar: amethod ha declarado excepciones que no están en la versión de la clase base. 4)Se compilara y se ejecutara mostrando: “Pausing” y “Continuing” después de que una tecla es pulsada. Pregunta 3) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? import java.io.*; class Base{ public static void amethod()throws FileNotFoundException{} } public class ExcepDemo extends Base{ public static void main(String argv[]){ ExcepDemo e = new ExcepDemo(); } public static void amethod(int i)throws IOException{} private boolean ExcepDemo(){ try{ DataInputStream din = new DataInputStream(System.in); System.out.println("Pausing"); din.readChar(); System.out.println("Continuing"); this.amethod(); return true; }catch(IOException ioe) { }finally{ System.out.println("finally"); } 68 javaHispano. Tu comunidad, tu lenguaje. return false; } } 1)Se compilara y se ejecutara sin mostrar algo. 2)Mostrando: “Pausing”, “Continuing” y “Finally” 3)Error al ejecutar: amethod ha declarado excepciones que no están en la versión de la clase base. 4)Se compilara y se ejecutara mostrando: “Pausing” y “Continuing” después de que una tecla es pulsada. Pregunta 4) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? import java.io.*; class Base{ public static void amethod()throws FileNotFoundException{} } public class ExcepDemo extends Base{ public static void main(String argv[]){ ExcepDemo e = new ExcepDemo(); } public boolean amethod(int i){ try{ DataInputStream din = new DataInputStream(System.in); System.out.println("Pausing"); din.readChar(); System.out.println("Continuing"); this.amethod(); return true; }catch(IOException ioe) { }finally{ System.out.println("Doing finally"); } return false; } ExcepDemo(){ amethod(99); } } 69 Tutorial para el examen de certificación: SCJP 1.2 1)Error en tiempo de compilación: amethod no lanza FileNotFounfException 2)Se compila, se ejecuta y muestra “Pausing” y “Continuing” 3) Se compila, se ejecuta y muestra “Pausing” ,“Continuing” y “Doing finally” 4)Error en tiempo de compilación, la cláusula finally nunca se alcanza. Respuestas Respuesta 1) 4)Se compilara y se ejecutara mostrando: “Pausing” y “Continuing” después de que una tecla es pulsada. Un método sobrescrito en una subclase no debe lanzar Excepciones no lanzadas en la clase base. En el caso del método amethod este no lanza ninguna excepción y en consecuencia compilará, así sin alguna indicación. No hay razón alguna por la que un constructor no pueda protegerse. Respuesta 2) 4)Se compilara y se ejecutara mostrando: “Pausing” y “Continuing” después de que una tecla es pulsada. En este versión amethod se ha sobrecargado de manera que no hay ninguna restricción sobre qué excepciones pueden o no ser lanzadas. Respuesta 3) 1)Se compilara y se ejecutara sin mostrar algo. OK, aquí me he salido un poco fuera del tema. Observa que lo que antes era un constructor ahora tiene un valor del retorno. Esto lo convierte en un método ordinario y por lo tanto no se llama cuando se crea una instancia de la clase. Respuesta 4) 3) Se compila, se ejecuta y muestra “Pausing” ,“Continuing” y “Doing finally” La cláusula finally siempre se ejecuta Otras Fuentes para este tema Este tema es cubierto en el Tutorial de SUN http://java.sun.com/docs/books/tutorial/essential/exceptions/definition.html Richard Baldwin cubre este tema en: http://www.geocities.com/Athens/Acropolis/3797/Java030.htm 70 javaHispano. Tu comunidad, tu lenguaje. y http://www.geocities.com/Athens/Acropolis/3797/Java056.htm Jyothi Krishnan en http://www.geocities.com/SiliconValley/Network/3693/obj_sec2.html#obj7 Bruce Eckel Thinking in Java Capítulo 9) 71 Tutorial para el examen de certificación: SCJP 1.2 Sección 3. Recolector de basura –Garbage CollectorObjetivo 1. Exponer el comportamiento que se garantiza por el sistema de recolección de basura y escribir código que explícitamente haga objetos elegibles para la recolección. ¿Por qué querrías utilizar el recolector de basura? Puedes ser un programador de Java muy experimentado y nunca haber tenido que familiarizarte con los detalles del recolector de basura. Incluso la expresión recolección de basura es un poco rara. En este contexto significa la liberación de memoria que se ha asignado y que ya ha sido usada por el programa. Cuando ya no se necesita una parte de la memoria asignada ésta puede considerarse como basura, es decir algo que ya no se necesita; y que simplemente está desordenando espacio útil. Una de las grandes cualidades aclamadas de Java es que no tienes que preocuparte por la recolección de basura. Si vienes de programar con Visual Basic puede parecerte absurdo que algún sistema, por si mismo, no cuide este detalle. En C/C++ el programador tiene que seguir, de forma manual, el rastro la asignación y desasignación de la memoria. Como resultado de esto, las "fugas de memoria" son una gran fuente de errores difíciles de rastrear. Ésta es una de las razones por las que algunas versiones de aplicaciones de Microsoft como Word o Excel, se detienen apenas comienzan a ejecutarse. Las fugas de memoria a la larga, finalmente hacen que el sistema entero se cuelgue y que te veas en la necesidad de pulsar ese gran interruptor luminoso. En alguna parte, en esos centenares de miles de líneas de código en C++, un programador asignó un bloque de memoria pero se le olvido asegurarse de liberarla. Java y el recolector de basura. A diferencia de C/C++, Java libera automáticamente las referencias que no se están utilizando. No tienes que pasar por el trauma de buscar asignaciones de memoria que nunca son liberadas y por consiguiente, no necesita saber cómo asignar ó cómo saber el tamaño de un tipo de datos para asegurar la compatibilidad de la plataforma. Así qué ¿Por qué querrías conocer mas los detalles del recolector de basura? Dos respuestas saltan a mi mente, una es aprobar el examen y la otra es comprender lo que ocurre en circunstancias extremas. Por ejemplo, si escribes código que genere números muy grandes de objetos o de variables te puede ser útil saber cuando se liberan las referencias. Si lees algún newsgroups verás a personas informando en ocasiones de ciertas aplicaciones Java que agotan los recursos y hacen que todo se caiga. Esto no estaba en el folleto del SUN cuando lanzaron Java. 72 javaHispano. Tu comunidad, tu lenguaje. Siguiendo la filosofía de la recolección de basura automática, puedes sugerir o animar que la JVM realice la recolección de basura, pero no la puedes forzar. Permíteme remarcar el punto, no puedes forzar la ejecución de la recolección de basura, sólo puedes sugerirla. A primera vista el termino finalización me suena un poco como a los destructores en C++ usados para limpiar los recursos antes de que se destruya un objeto. La diferencia es qué en Java los recursos internos no necesitan ser liberados durante la finalización porque el recolector de basura cuida la asignación de memoria. Sin embargo si tienes recursos externos como un archivo de información, puedes usar la finalización para liberarlos. Escribir ejercicios o practicar con el recolector de basura es un poco complejo ya que no hay manera de conseguir código que indique cuando está disponible para la recolección de basura. No puedes escribir un fragmento de código con una sintaxis como la siguiente: if(EligibleForGC(Object){ //código no real System.out("Ready for Garbage"); } Debido a esto solo tienes que aprender las reglas. Una vez que una variable no es referenciada en mucho tiempo está disponible para ser recolectada por el recolector de basura. Puedes sugerir la ejecución del recolector de basura con el método System.gc(), pero esto no garantiza que se ejecutará inmediatamente. Las variables locales en los métodos pierden su alcance cuando el método que las ocupa termina. A partir de ese momento esos métodos son elegibles para el recolector de basura. Cada vez que el método vuelve a estar en el alcance del programa las variables locales se vuelven a crear. 73 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 1) ¿Cual de las siguientes líneas tiene la sintaxis correcta para sugerir que la JVM ejecute la recolección de basura? 1) System.free(); 2) System.setGarbageCollection(); 3) System.out.gc(); 4) System.gc(); Pregunta 2) ¿Qué código puedes escribir para asegurarte que las variables tipo int son recolectadas en un punto particular en este código? public class Rub{ int i=1; int j=2; int k=3; public static void main(String argv[]){ Rub r = new Rub(); r.amethod(); } public void amethod(){ i=0; j=0; k=0; } } 1) System.gc(); 2) System.free(); 3) Fijar el valor de cada variable int a null. 4) Ninguno de los anteriores. Respuestas Respuesta 1) 4) System.gc(); Respuesta 2) 4) Ninguno de los anteriores. 74 javaHispano. Tu comunidad, tu lenguaje. Únicamente puedes sugerir la ejecución del recolector de basura, por consiguiente no puedes estar seguro que se ejecutará en algún punto en particular de tu código. Otras Fuentes para este tema Jyothi Krishnan en http://www.geocities.com/SiliconValley/Network/3693/obj_sec3.html#obj8 75 Tutorial para el examen de certificación: SCJP 1.2 Sección 4. Fundamentos del lenguaje. Objetivo 1. Identificar las declaraciones correctas para la construcción de paquetes, conocer la sentencia import, la declaración de clases (todas las formas para incluir clases internas), conocer la declaración e implementación de interfaces (por ejemplo java.lang.Runnable u otra interfase descrita en el examen), declaración de métodos (incluyendo el método main que se usa para comenzar la ejecución de una clase), las declaraciones de variables e identificadores. Nota sobre este objetivo La frase de este objetivo es un poco extraña. Parece estar pidiéndote que entiendas donde, cómo y por qué puedes usar la sentencia import y package, donde debes colocar la declaración de una interface y las declaraciones de las variables. La sentencia package La palabra package implica una colección de clases, algo parecido a una librería. En el uso, un paquete es un poco semejante a un directorio. Si pones una declaración de un paquete en un archivo éste será sólo visible a otras clases en el mismo paquete. Puedes poner un comentario antes de la declaración del paquete pero nada más. Puedes encontrar preguntas del examen en donde pongan una sentencia import antes de la declaración del paquete. //Puedes colocar un comentario antes de la sentencia package package MyPack; public class MyPack{} Lo siguiente causará un error: import java.awt.*; //Error: colocar una sentencia import antes de la declaración //del paquete causará un error al compilar. package MyPack; public class MyPack{} Si un archivo fuente tiene una sentencia package, ésta debe ponerse antes de cualquier otra declaración aparte de los comentarios. La sentencia package puede incluir la anotación punto para indicar la jerarquía del paquete. Así el siguiente código compilará sin el error: package myprogs.MyPack; 76 javaHispano. Tu comunidad, tu lenguaje. public class MyPack{} Recuerda que si no pones una sentencia package al inicio del archivo fuente, se considerará que éste forma parte del paquete predefinido que corresponde al directorio actual. La sentencia import La sentencia import debe ir después de cualquier sentencia package y antes de cualquier otra línea de código. La sentencia import no puede ir dentro de las clases, ni después de la declaración de la clase o en cualquier otra lugar. La sentencia import te permite usar una clase directamente con su nombre en lugar de especificarla totalmente incluyendo el nombre completo del paquete al que pertenece. Un ejemplo de esto es que a la clase java.awt.Button normalmente se le referencia simplemente como Button, pero para ello requieres haber puesto al inicio del archivo fuente la siguiente línea: import java.awt.*; Toma en cuenta que aún usando la declaración completa de la clase no se afecta el desempeño ni se cambia el tamaño del archivo .class generado. Clases y la declaración de clases internas Un archivo puede tener únicamente una clase con modificador public. Si intentas crear un archivo con más de una clase pública el compilador te indicará un error específico. Un archivo puede contener múltiples clases no públicas, pero ten presente que esto generara distintos archivos .class separados por cada clase. Las clases internas se introdujeron con el JDK 1.1 La idea es permitir la definición de una clase dentro de otra, para que ésta pueda ser definida dentro de un método y para la creación de clases internas anónimas. Esto tiene algunos efectos interesantes, particularmente en la visibilidad. Lo siguiente es un ejemplo sencillo de una clase interna: class Outer{ class inner{} } Esto produce la generación de archivos .class con los nombres: 77 Tutorial para el examen de certificación: SCJP 1.2 Outer.class Outer$Inner.class La definición de la clase interna es sólo visible dentro del contexto de la existente clase Exterior. Así el siguiente código causará un error cuando se compile: class Outer{ class Inner{} } class Another{ public void amethod(){ Inner i = new Inner(); } } De modo que la clase Another no se podría crear, ya que, en ese contexto la clase inner no existe. Ésta sólo puede existir en el contexto de una instancia de la clase Outer. Así, el siguiente código se compilará sin problemas ya que sé esta instanciando la clase Inner dentro del ámbito de la clase Outer. class Outer{ public void mymethod(){ Inner i = new Inner(); } public class Inner{} } Pero que pasa si el contexto de la instancia Outer no existe. Para entender mejor como conseguir lo anterior, pensemos como podemos lograr el efecto anterior utilizando la palabra clave this. Podrías cambiar la línea que crea la instancia por lo siguiente: Inner i = this.new Inner(); Así, si necesitas crear una instancia de la clase Inner en un método estático o en alguna otra parte donde no tenga ámbito el objeto, puedes usar la palabra clave new como un método que pertenece a una clase exterior. class Outer{ public class Inner{} } 78 javaHispano. Tu comunidad, tu lenguaje. class another{ public void amethod(){ Outer.Inner i = new Outer().new Inner(); } } A pesar de mis locas explicaciones, encuentro poco intuitiva esta sintaxis y hasta yo me olvido de ella cinco minutos después de aprendérmela. Es muy probable que encuentres preguntas referentes a esto en el examen, así que préstale bastante atención. Puedes acceder a una clase interna utilizando el siguiente código: Outer.Inner i = new Outer().new Inner(); Uno de los beneficios de las clases internas es que una clase interna generalmente consigue acceso a los campos la clase que lo contiene(o una externa) Al contrario de una clase externa, una clase interna puede ser privada o estática. Los examinadores pueden hacer preguntas molestas como "¿Puede una clase interna ser estática o privada?". Haciendo estática una clase interna, ésta tiene algunos efectos interesantes con la manera en que se tienen acceso a los campos o miembros de la clase que la contiene. El efecto de hacer estáticos los recursos se refleja en que sólo hay una instancia de cualquier variable, no importa cuántas instancias se creen de la clase externa. En esta situación ¿Cómo puede la clase interna estática saber qué variables puede acceder de su clase externa no estática?. Por supuesto la respuesta es que no lo puede saber, y así una clase interna estática no puede acceder variables de instancia de la clase que la contiene. Por supuesto, los métodos de una clase interna estática pueden acceder a cualquier campo o miembro estático de la clase que lo contiene pero siempre será únicamente a una sola instancia de cualquiera de esos campos. Clases internas declaradas dentro de métodos Las clases internas se pueden crear dentro de los métodos. Esto es algo que las herramientas de desarrollo de aplicaciones como JBuilder de Borland hacen constantemente cuando generan un manejador de Eventos. Aquí hay un ejemplo de dicho código generado automáticamente: 79 Tutorial para el examen de certificación: SCJP 1.2 buttonControl1.addMouseListener(new java.awt.event.MouseAdapter() { public void mouseClicked(MouseEvent e) { buttonControl1_mouseClicked(e); } }); Fíjate en la palabra clave new que se encuentra justo después del primer paréntesis. Esta indica que una clase interna anónima(sin nombre) está definiéndose dentro del método addMouseListener. Esta clase normalmente podría definirse con un nombre que podría hacerla más fácil de leer, pero cuando ese fragmento de código no se utiliza para algún otro proceso en cualquier otra parte, darle un nombre no ayuda mucho. Si creas tal código de manera manual, es fácil confundirte por el número de niveles de las llaves y paréntesis. Observa cómo la estructura completa finaliza con punto y coma, ya que es realmente ahí en donde finaliza la llamada al método. Como podrías suponer una clase anónima no puede tener un constructor. Míralo de esa manera, un constructor es un método sin valor de retorno y con el mismo nombre de la clase. ¡Upss! Nosotros estamos hablando de clases sin nombres. Una clase anónima puede heredar otra clase o implementar una única interface. Este límite peculiar no parece ser tomado en cuenta en el examen. Visibilidad de campos para clases definidas dentro de un método Una clase definida dentro de un método puede acceder sólo a campos o miembros del método que la contiene si éstos se han definido con el modificador final. Esto se debe a que las variables definidas dentro de un método normalmente se consideran variables automáticas; es decir, sólo existen mientras el método se está ejecutando. Los Campos o miembros definidos dentro de una clase creada dentro de un método duran mas que el método que la contiene. Una clase definida dentro de un método solo puede acceder a campos con el modificador final del método que la contiene Debido que una variable con modificador final no puede ser cambiada por la JVM puedes estar seguro que el valor permanecerá constante aun después de que el método exterior ha dejado de ejecutarse. Es muy probable que encuentres preguntas referentes a esto en el examen, incluso preguntas que cuestionen el estado de las variables pasadas como parámetros al método. 80 javaHispano. Tu comunidad, tu lenguaje. Creando una interface Las interfaces son la manera en que Java maneja la ausencia de la herencia múltiple. Curiosamente Visual Basic tiene la palabra reservada interface y maneja el concepto de manera similar a Java. El tema de las interfaces es conocido en ocasiones como programación por contacto. Una interface se utiliza a través de la palabra clave implements. Así, una clase puede declararse de la siguiente manera: class Malvern implements Hill,Well{ public } Pregunta 1) Dado el siguiente código: public class FinAc{ static int l = 4; private int k=2; public static void main(String argv[]){ FinAc a = new FinAc(); a.amethod(); } public void amethod(){ final int i = 99; int j = 6; class CInMet{ public void mymethod(int q){ //Here }//fin de mymethod }//fin de CInMet CInMet c = new CInMet(); c.mymethod(i); 81 Tutorial para el examen de certificación: SCJP 1.2 }//fin de amthod } ¿Cuales de las siguientes variables son visibles en la línea marcada con el comentario //Here? 1)l 2)k 3)i 4)j Pregunta 2) ¿Cuales de los siguientes fragmentos de código compilarán correctamente? 1) //A Comment import java.awt.*; class Base{}; 2) import java.awt.*; package Spot; class Base(); 3) //Another comment package myprogs.MyPack; public class MyPack{} 4) class Base{} import java.awt.*; public class Tiny{} Pregunta 3) ¿Cuales de las siguientes afirmaciones son verdaderas? 1)Una clase interna puede definirse como estática 2)Una clase interna no puede ser defina como privada 3)Una clase anónima puede tener sólo un constructor 4)Una clase interna puede heredar otra clase 82 javaHispano. Tu comunidad, tu lenguaje. Pregunta 4) Desde código que no tenga activa la referencia this, ¿Como puedes crear una instancia de una clase interna? 1)Outer.Inner i = new Outer().new Inner(); 2)Sin una referencia this una clase interna no puede ser creada 3)Outer.Inner i = Outer().new new Inner(); 4)Outer i = Outer.new().Inner(); Respuestas Respuesta 1) 1) l 2) k 3) i Una clase definida dentro de un método solo puede ver campos con el modificador final desde el método que la contiene. Sin embargo, también puede ver los campos de la clase que contiene al método que la contiene a ella, incluyendo campos privados. El campo j no se definió como final. Respuesta 2) 1) //A Comment import java.awt.*; class Base{}; 3) //Another comment package myprogs.MyPack; public class MyPack{} La declaración de un paquete debe ser la primera línea de código en un archivo (aparte de los comentarios). Una declaración import debe estar después de cualquier declaración de un paquete y antes de cualquier otra línea de código. Respuesta 3) 1)Una clase interna puede definirse como estática 4)Una clase interna puede heredar otra clase Las clases anónimas no pueden tener un constructor. Las clases internas pueden definirse como privadas. Respuesta 4) 83 Tutorial para el examen de certificación: SCJP 1.2 1)Outer.Inner i = new Outer().new Inner(); Otras Fuentes para este tema: El Tutorial de SUN http://java.sun.com/docs/books/tutorial/java/more/nested.html Richard Baldwin http://www.Geocities.com/Athens/7077/Java094.htm y también http://www.Geocities.com/Athens/7077/Java095.htm Jyothi Krishnan http://www.geocities.com/SiliconValley/Network/3693/obj_sec4.html#obj9 Un Tutorial sobre paquetes http://v2ma09.gsfc.nasa.gov/JavaPackages.html Especificaciones sobre interfaces en Java http://java.sun.com/docs/books/jls/html/9.doc.html#238680 84 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2. Declarar la correspondencia entre los valores del arreglo pasado como argumento al método main y los argumentos de la línea de comandos. Nota: Este tema por sí solo difícilmente amerita ser tomado como un objetivo. Este objetivo puede sorprender al programador más experimentado en C/C++ ya que el primer elemento del arreglo argv[] es la primera cadena después del nombre del programa en la línea de comandos. Así si un programa se ejecutara como sigue: java myprog myparm El elemento argv[0] contendría "myparm". Si vienes de programar en C/C++ podrías esperar que el contenido fuera "java". Java no contiene una opción equivalente a Option Base de Visual Basic y, en consecuencia, los arreglos siempre empezarán desde el elemento cero. Veamos el siguiente código: public class MyParm{ public static void main(String argv[]){ String s1 = argv[1]; System.out.println(s1); } } He asignado el argumento 1 a un String sólo para resaltar que argv es un arreglo de String's. Si ejecutas este programa de la manera siguiente: java MyParm hello there Se mostrará there, y no hello o MyParm Pregunta 1) 85 Tutorial para el examen de certificación: SCJP 1.2 Dado el siguiente método main en una clase llamada cycle: public static void main(String bicycle[]){ System.out.println(bicycle[0]); } y una llamada desde la línea de comandos como la siguiente: java Cycle one two ¿Cuál será la salida? 1) Ninguna de estas opciones 2) Cycle 3) one 4) two Pregunta 1) ¿Cómo puedes recuperar los valores pasados desde la línea comandos al método main? 1) usando el método System.getParms() 2) asignando un elemento del argumento a un String 3) asignando un elemento del argumento a un arreglo de caracteres 4) ninguna de estas opciones Respuestas Respuesta 1) 3) one Respuesta 2) 2) asignando un elemento del argumento a un String Otras Fuentes para este tema: Este tema es cubierto en el tutorial de SUN en: http://java.sun.com/docs/books/tutorial/essential/attributes/cmdLineArgs.html 86 javaHispano. Tu comunidad, tu lenguaje. Objetivo 3. Identificar todas las palabras reservadas del lenguaje. Nota en este objetivo: Puedes asimilar fácilmente este objetivo en base a aprender las palabras reservadas frecuentemente menos utilizadas y asegurándote no tomar en cuenta a "falsos friendsπ" de otros lenguajes que puedes llegar a saber, particularmente C/C++. El examen pone énfasis especial en el reconocimiento de las palabras reservadas. Reconocerás la mayoría de la palabras reservadas de Java conforme uses el lenguaje, pero en el uso personal del lenguaje raramente utilizamos las excepciones, y las palabras reservadas que podrían venir en el examen. Ejemplos de las palabras reservadas más raramente utilizadas (indudablemente para un principiante) son: volatile transient native • • • Palabras reservadas abstract boolean break byte case catch char class const * continue default do double else extends final finally float for goto * if implements import instanceof int interface long native new null package private protected public return short static super switch synchronized this throw throws transient try void while volatile Las palabras con asteriscos son reservadas y actualmente no son utilizadas. Observa que todas las palabras reservadas están en minúsculas, así for es una palabra reservada pero FOR no la es. Hay un debate acerca de que si null es una palabra reservada, pero sugiero que para los propósitos del examen asumas que si la es. π NT: El autor utiliza esa expresión para referirse a la confusión que en ocasiones produce la palabra friend, la cual es palabra reservada en C++, pero no en Java. 87 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 1) ¿Cuáles de las siguientes son palabras reservadas de Java? 1) double 2) Switch 3) then 4) instanceof Pregunta 2) ¿Cuáles de las siguientes no son palabras reservadas de Java? 1)volatile 2)sizeOf 3)goto 4)try Respuestas Respuesta 1) 1) double 4) instanceof Nota que la letra mayúscula S en switch hace que ésta no sea una palabra reservada y la palabra then es parte de Visual Basic pero no de Java. Respuesta 2) 2)sizeOf 88 javaHispano. Tu comunidad, tu lenguaje. Esta es una palabra reservada en C/C++ para determinar el tamaño de un dato primitivo para una plataforma en particular. Debido a que los datos primitivos tienen el mismo tamaño en todas las plataformas en Java ésta palabra reservada no se necesita. Otras Fuentes para este tema Michael Thomas http://www.michael-thomas.com/java/JCP_Keywords.htm 89 Tutorial para el examen de certificación: SCJP 1.2 Objetivo 4. Conocer el efecto de usar una variable o un elemento de un arreglo de cualquier tipo cuando no se le ha hecho alguna asignación explicita. Variables Podrías aprender a programar en Java sin realmente entender la importancia que representa este objetivo, pero éste representa conocimiento valioso para el mundo real. Esencialmente a una variable de nivel de clase siempre se le asignará un valor predefinido y a una variable miembro (una variable contenida dentro de un método) no se le asignará un valor predefinido. Si intentas acceder a una variable sin asignar generarás un error. Por ejemplo: class MyClass{ public static void main(String argv[]){ int p; int j = 10; j=p; } } Este código producirá el siguiente error: "error variable p might not have been assigned" Esto puede verse como un cambio bienvenido de la tendencia de C/C++ de darle bastante cuerda a una variable sin asignar dejando un valor arbitrario en p. Si p se hubiera definido al nivel de clase se le habría asignado su valor predefinido y no se habría generado algún error. class MyClass{ static int p; public static void main(String argv[]){ int j = 10; j=p; System.out.println(j); } } El valor predefinido para un entero es 0, así el código anterior mostrará 0. El valor predefinido para los tipos de datos numéricos es cero, para un boolean es false y una referencia a un objeto es el único tipo de datos que tiene como valor predefinido null. 90 javaHispano. Tu comunidad, tu lenguaje. Antes de la inicialización de los arreglos, donde quiera que sean creados, estos siempre son fijados para contener sus valores predefinidos. Arreglos Aprender esta parte del objetivo requiere comprender una sencilla regla. El valor de los elementos de un arreglo de cualquier tipo base siempre se inicializará con un valor predefinido, dondequiera que se defina el arreglo. No importa si el arreglo se define a nivel de clase o método, a los valores del arreglo siempre se le pondrán sus valores predefinidos. Una pregunta que puedes hacerte es ¿Qué contendrá un elemento en particular de un arreglo sin asignar?. A menos que el arreglo sea de objetos, la respuesta será distinta de null. Pregunta 1) Dado el siguiente código, ¿Qué contendrá el elemento b[5]? public class MyVal{ public static void main(String argv[]){ MyVal m = new MyVal(); m.amethod(); } public void amethod(){ boolean b[] = new boolean[5]; } } 1) 0 2) null 91 Tutorial para el examen de certificación: SCJP 1.2 3) "" 4) Ninguna de estas opciones Pregunta 2) Dado el siguiente código, ¿Qué contendrá el elemento 1 del arreglo mycon? MyCon(){ int[] mycon= new int[5]; } 1) 0 2) null 3) "" 4) Ninguna de estas opciones Pregunta 3) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? public class MyField{ int i=99; public static void main(String argv[]){ MyField m = new MyField(); m.amethod(); } void amethod(){ int i; System.out.println(i); } } 1)Se mostrará 99 2)Se mostrará 0 3)Se generará un error al compilar 4)Se generará un error al ejecutar Pregunta 4) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? public class MyField{ 92 javaHispano. Tu comunidad, tu lenguaje. String s; public static void main(String argv[]){ MyField m = new MyField(); m.amethod(); } void amethod(){ System.out.println(s); } } 1)Error al compilar: s no ha sido inicializada 2)Error al ejecutar: s no ha sido inicializada 3)Salida en blanco 4)Se mostrará null Respuestas Respuesta 1) 4) Ninguna de estas opciones Atención en este punto. Los elementos de un arreglo comienzan a numerarse desde 0, por consiguiente, para éste arreglo, no hay un elemento 5. Si intentas ejecutar: System.out.println(b[5]) Generarás una excepción. Respuesta 2) 1)0 Un constructor actúa de manera parecida a cualquier otro método; así, un arreglo de enteros será inicializado conteniendo ceros dondequiera que sea creado. Respuesta 3) 3)Se generará un error al compilar Generarás un error al compilar el código indicando que la variable i no ha sido inicializada, la variable de nivel de clase i es un caso extraño, ya que es ensombrecida por su versión en el nivel del método. Las variables de nivel de método no obtienen ninguna inicialización por omisión. Respuesta 4) 4)Se mostrará null 93 Tutorial para el examen de certificación: SCJP 1.2 Una variable creada en el nivel de clase siempre tendrá un valor por omisión. El valor por omisión para una referencia a un objeto es null y el método toString implícitamente llamado con System.out.println mostrará null. Otras Fuentes para este tema: Este tema es cubierto ligeramente en el Tutorial de SUN en: http://java.sun.com/docs/books/tutorial/java/nutsandbolts/vars.htmll Richard Baldwin cubre el tema en: http://www.geocities.com/Athens/Acropolis/3797/Java020.htm#variables Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec4.html#obj12 94 javaHispano. Tu comunidad, tu lenguaje. Objetivo 5. Definir el rango de todos los tipos de datos primitivosφ, declarar valores literales para Strings y todos los tipos primitivos usando todos los formatos permitidos, conocer las bases numéricas y sus representaciones. Nota en este objetivo Éste es uno de los ligeramente molestos, pero bastante fácil, objetivos a cubrir. Puedes escribir una gran cantidad de código Java sin conocer el rango de los tipos primitivos pero familiarizarte con ellos no debe tomarte mucho tiempo. Ten cuidado con este tema para que puedas usar todos los formatos, y no pases por alto el formato octal. Tamaño de los tipos primitivos enteros Cuando este objetivo pide el rango de los tipos de datos primitivos asumo que para representarlo sólo se requiere el número 2 elevado a la potencia apropiada en lugar del número que representa. En mi mente hay sólo tres tipos enteros a aprender ya que, en mi experiencia basada en PC. El tamaño de un byte es intuitivo: 8 bits. Rango de los primitivos Enteros Nombre Tamaño Rango byte 8 bit -27 to 2 7-1 short 16 bit -215 to 215-1 int 32 bit -231 to 231-1 long 64 bit -2 63 to 2 63-1 Declarando literales enteros Hay tres maneras de declarar un literal entero. Como esperabas, por defecto, es un valor decimal. Aquí están las opciones φ NT: Algunos documentos utilizan el termino “sencillos” o “básicos” en lugar de “primitivos”. 95 Tutorial para el examen de certificación: SCJP 1.2 Declarando 18 como un literal entero Decimal 18 Octal 022 (Cero no la letra O) Hexadecimal 0x12 Si compilas y ejecutas esta pequeña clase obtendrás en cada ocasión el valor 18. public class Lit{ public static void main(String[] argv){ int i = 18; int j = 022;//Version Octal: dos veces ocho mas dos int k = 0x12;//Version Hex: una vez dieciséis mas dos System.out.println(i); System.out.println(j); System.out.println(k); } } Roberts y Heller describen 6 maneras de declarar literales enteros, ya que para Java la letra X para la declaración de un hexadecimal extraordinariamente no es sensible a mayúsculas, observa que no se utilizan las letras de la A a la F para la notación hexadecimal. A mí me es mas más fácil recordar únicamente las tres maneras descritas anteriormente y que las letras no son sensibles a mayúsculas. Tamaño de los tipos primitivos de coma flotante Los números de coma flotante son especimenes ligeramente extraños debido a que sus cálculos pueden tener algunos resultados inesperados. Citando a Peter Van Linden "La precisión exacta depende del número que se representa". Como compensación a esta precisión variable logras conseguir casi jugar con números grandes más allá los que puedas imaginar. Así una variable de tipo double puede almacenar un número como 17 seguido por 307 ceros. Un numero que sirve incluso para que puedas almacenar los estados financieros de Bill Gates. (hasta que Linux alcance la dominación mundial total, entonces un entero podrá hacer muy bien ese trabajo). Rango de los tipo de coma flotante float 32 bit double 64 bit 96 javaHispano. Tu comunidad, tu lenguaje. Ten presente que el tipo de dato predefinido para un literal con un componente decimal es un double y no un long. Esto es ligeramente confuso puesto que podrías pensar que el tipo predefinido para un número de coma flotante debería ser un float. Puedes encontrar preguntas en el examen similares a la siguiente: ¿Se compilará el siguiente código? float i = 1.0; Tu intuición debería decirte que esto debe compilarse sin problemas. Desgraciadamente el examen no esta diseñado para probar su intuición. Esto causará un error en tiempo de compilación ya que estas intentando asignar un double a un float. Puedes arreglar este código de la manera siguiente: float i = 1.0F; o incluso: float i = (float) 1.0; Indicando tipos de datos con un literal sufijo Como se mostró anteriormente puedes indicarle a Java que un literal numérico es de un tipo de dato en particular asignándole un literal sufijo. Estos son los sufijos disponibles: Sufijos para tipos de datos float F long L double D Los tipos boolean y char Los tipos de datos primitivos boolean y char son un poco curiosos. Si tienes experiencia programando con C/C++ presta atención particularmente al tipo boolean y a asegúrate de no traer contigo "falsos friends" de cualquiera de estos lenguajes. A un tipo boolean no se le puede asignarse otro valor distinto de true o false. Los valores true o false no corresponden al 0, al -1 o a algún otro número. 97 Tutorial para el examen de certificación: SCJP 1.2 Un dato de tipo boolean únicamente puede recibir los valores de true o false, y no se le puede asignar un número como –1 o 0. El tipo de dato primitivo char es el único tipo de dato primitivo sin signo. El tipo char se usa para denotar un carácter Unicode. Unicode es un código alternativo al ASCII que almacena caracteres en 2 bytes en lugar de 1 byte como lo hace el código ASCII. Esto te permite representar 65K caracteres, qué aunque no es una cantidad suficientemente grande para cubrir todas las escrituras del mundo, es mejor que los 255 caracteres que nos proporciona el código ASCII. La internacionalización es un tema completo por si mismo, y el hecho de que puedas representar caracteres en chino o Vietnamés, no significa que se desplegarán correctamente si tienes un sistema operativo en inglés normal. Un literal de tipo char se crea encerrando el carácter con comillas simples: char a = 'z'; Nota que son comillas simples ‘ y no comillas dobles “. Todo esto es funcional en mi pequeño mundo basado en el Ingles pero como Java es un sistema mundial un char puede contener cualquiera de los caracteres disponible en el sistema Unicode. Esto se hace usando cuatro dígitos hexadecimales precedidos por \u, con la expresión completa encerrada con comillas simples. Así, el espacio en blanco se representa de la siguiente manera: char c = ‘\u0020’; Si asignas un número sencillo a un char, éste será tomado como un carácter alfabético. Así que el siguiente código mostrará la letra A(valor 65 del código ASCII) seguido de una espacio en blanco. public class MyChar{ public static void main(String argv[]){ char i = 65; char c = '\u0020'; System.out.println(i); System.out.println("This"+c+"Is a space"); } } 98 javaHispano. Tu comunidad, tu lenguaje. Declarando literales String El tipo String no es un tipo de dato primitivo pero es tan importante que en ciertas áreas Java lo trata como si lo fuera. Uno de estos rasgos es la habilidad de declarar literales String en lugar de usar new para instanciar una copia de la clase. Las literales String son bastante sinceras. Asegúrate de recordar que las literales String son encerradas con comillas dobles mientras que un literal char utiliza comillas simples. Así: String name = "James Bond" Pregunta 1) ¿Cuáles de las siguientes líneas se compilarán correctamente? 1) float f=10f; 2) float f=10.1; 3) float f=10.1f; 4) byte b=10b; Pregunta 2) ¿Cuáles de las siguientes líneas se compilarán correctamente? 1) short myshort=99S; 2) String name='Excellent tutorial Mr Green'; 3) char c=17c; 4) int z=015; Pregunta 3) 99 Tutorial para el examen de certificación: SCJP 1.2 ¿Cuáles de las siguientes líneas se compilarán correctamente? 1) boolean b=-1; 2) boolean b2=false; 3)int i=019; 4) char c=99; Respuestas Respuesta 1) 1) float f=10f; 3) float f=10.1f; Para la opción 4; No existe una literal b(byte), y la opción 2 causará un error al compilarse debido a que el tipo de dato por omisión para un número con un componente decimal es double. Respuesta 2) 4)int z=015; Las letras c y s no existen como indicadores literales y un String debe estar encerrado por comillas dobles y no por comillas simples como en el ejemplo. Respuesta 3) 2) boolean b2=false; 4) char c=99; Te debe ser bastante obvio que la opción 1 es errónea, a un boolean únicamente se le pueden asignar los valores true y false. La opción 3 es ligeramente más difícil de detectar como errónea; ya que, esa es la manera correcta de declarar un literal octal; pero no puedes usar el número 9 si estas en base 8 donde tienes números del 0 al 7. Quizá sean un poco tramposas estas ultimas opciones. Otras Fuentes para este tema: Este tema es cubierto en el Tutorial de SUN en: http://java.sun.com/docs/books/tutorial/java/nutsandbolts/vars.html (No encontraras mención alguna sobre las literales.) 100 javaHispano. Tu comunidad, tu lenguaje. Richard Baldwin cubre este tema en: http://www.Geocities.com/Athens/Acropolis/3797/Java020.htm#primitive types (nuevamente no encontraras mención alguna sobre las literales.) Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec4.html#obj13 Bruce Eckel's Thinking in Java Capítulo 2 "Caso Especial: Tipos primitivos” Capítulo 3 "Literales" 101 Tutorial para el examen de certificación: SCJP 1.2 Sección 5. Operadores y Asignaciones Objetivo 1 Determinar el resultado de aplicar cualquier operador, incluyendo los operadores de asignación y el operador instanceof, sobre operandos de cualquier tipo de clase, dependiendo de su alcance, accesibilidad, o de la combinación de estos. El operador instanceof El operador instanceof es una bestia extraña, desde mi punto de vista debería ser un método en lugar de un operador. Probablemente has escrito bastante código con Java y no has tenido la necesidad de usarlo, pero para propósitos del examen necesitas conocerlo. Este operador retorna un valor booleano como resultado a una prueba sobre un tipo de clase en tiempo de ejecución. Y efectivamente, como estas pensando, se usa para preguntar: ¿Es esta_clase instanciade esta_otra_clase? Si lo utilizas en casos triviales como el siguiente, quizá no le veas mucha utilidad: public class InOf { public static void main(String argv[]){ InOf i = new InOf(); if(i instanceof InOf){ System.out.println("It's an instance of InOf"); }//End if }//End of main } Como habrás deducido el resultado será: It's an instance of InOf Sin embargo, pueden surgir situaciones donde únicamente tienes acceso a una referencia de un objeto, el cual se refiere a algo que esta más arriba o más abajo en la jerarquía de clases del programa. Así puedes tener un método que tome como parámetro un componente que puede referirse a un botón, a una etiqueta o a cualquier otra cosa. En estas circunstancias, el operador instanceof se usa para examinar el tipo del objeto, realizar una comparación y una conversión y así llamar al método más apropiado. El siguiente código ilustra un ejemplo de esto: import java.awt.*; 102 javaHispano. Tu comunidad, tu lenguaje. public class InOfComp { public static void main(String argv[]){ }//End of main public void mymethod(Component c){ if( c instanceof Button){ Button bc = (Button) c; bc.setLabel("Hello"); }else if (c instanceof Label){ Label lc = (Label) c; lc.setText("Hello"); } }//End of mymethod } Si en tiempo de compilación la examinacion del tipo de la clase y la conversión forzada no fueran realizadas, los métodos setLabel y setText no estarían disponibles. Observa que el operador instanceof se aplica a los nombres de las clases no a una referencia de un objeto para una clase. El operador + Como esperabas, el operador + adiciona 2 números entre si. Así la siguiente línea de código mostrará 10: int p=5; int q=5; System.out.println(p+q); El operador + es un ejemplo raro de la sobrecarga de operadores en Java. los programadores en C++ están acostumbrados a poder sobrecargar operadores para que estos signifiquen cualquier cosa que ellos definan. Esta facilidad no está disponible para un programador en Java, sin embargo esta opción esta disponible para los Strings, ya que el signo mas(+) se sobrescribe para ofrecer la concatenación de Strings. De esta manera el siguiente código sí se compilará: String s = "One"; String s2 = "Two" String s3 = ""; s3 = s+s2; System.out.println(s3); 103 Tutorial para el examen de certificación: SCJP 1.2 El resultado será "OneTwo", observa que no hay espacios entre las dos cadenas unidas. Bajo ciertas circunstancias Java realiza una llamada implícita el método toString. Este método, como su nombre lo dice, intenta convertir a un String a el objeto que lo llama. Para un entero con valor 10, la llamada a toString retornara la cadena "10". Lo anterior se aprecia mejor en el siguiente código int p = 10; String s = "Two"; String s2 = ""; s2 = s + p; System.out.printlns(s2); el resultado será: Two10 Recuerda que para los Strings, únicamente el operador + esta sobrecargado. Si intentas utilizar sobre un String los operadores división y menos(/,-) causaras un error. Asignando variables primitivas de tipos diferentes Un booleano única y exclusivamente puede recibir otro valor de tipo booleano. Para los programadores C/C++, recuerden que esto significa que a un booleano no se le puede asignar -1 o 0, ya que en Java un booleano no es substituible por el numero cero o un valor distinto de cero. Con esta principal excepción en el tipo de datos boolean, el principio general para entender este objetivo esta en saber que las conversiones por "ensanchamiento" están permitidas en Java, siempre y cuando no impliquen exactitud. Por ensanchamiento quiero decir que una variable como un byte que ocupa 1 byte (ocho bits) puede asignarse a una variable que ocupa más bits como por ejemplo un int. Sin embargo si intentas asignar un int a un byte conseguirás un error en tiempo de compilación: byte b= 10; int i = 0; b = i; 104 javaHispano. Tu comunidad, tu lenguaje. Los datos primitivos pueden ser asignados a tipos de datos mas "anchos", un booleano únicamente puede ser asignado a otro booleano. Como esperabas, no puedes asignar datos primitivos a objetos y viceversa. Así, el siguiente código no es valido: int j=0; Integer k = new Integer(99); j=k; //Asignacion ilegal de un objeto un tipo primitivo. Una diferencia importante entre la asignación de objetos y la asignación de tipos primitivos, es que los tipos primitivos son revisados en tiempo de compilación mientras que los objetos son revisados en tiempo de ejecución. Esto se cubrirá mas adelante ya que tiene implicaciones importantes cuando un objeto no esta completamente resuelto en tiempo de compilación. Por supuesto, puedes realizar una conversión forzada para hacer que una variable se ajuste a un tipo de datos mas estrecho, Esto a menudo no es aconsejable ya que el resultado perderá precisión, pero es valido si realmente lo quieres hacer. Java utiliza la convención de C/C++ encerrando entre paréntesis () el tipo de datos al que se quiere convertir. Por lo tanto el siguiente código se compilara y ejecutara sin problemas: public class Mc{ public static void main(String argv[]){ byte b=0; int i = 5000; b = (byte) i; System.out.println(b); } } el resultado es -120 Posiblemente no es un valor que te sea muy útil. Asignando referencias de objetos de diferentes tipos. Cuando asignamos referencias de objetos, otra regla general es que puedes realizar la asignación de abajo hacia arriba en la jerarquía de clases, pero no de arriba hacia abajo. Puedes ver esto de la siguiente manera. Si tu asignas una instancia de una clase hija a la 105 Tutorial para el examen de certificación: SCJP 1.2 clase Base, Java sabrá qué métodos están en la clase hija. Sin embargo una clase hija puede tener métodos adicionales de los que le proporciona la clase Base. Puedes solucionar este problema usando la conversión forzada. Las referencias de objetos se asignan hacia arriba en la jerarquía de clases. El siguiente ejemplo muestra que ocurre cuando fuerzas la asignación de una referencia de un objeto hacia una clase de jerarquía superior: class Base{} public class ObRef extends Base{ public static void main(String argv[]){ ObRef o = new ObRef(); Base b = new Base(); b=o;//Esta linea se compilara sin error /*o=b; Esto debe causar un error indicando que se requiere una conversion forzada para asignar una referencia de Base a ObRef */ } } Operadores de desplazamiento de bits. Odio un poco esto de trabajar con desplazamientos de bits. Ya que implica que llenes tu mente con una capacidad no intuitiva que es trabajar con números infinitamente pequeños, números que un programador quizá nunca utilizara. Pero debido a que todas las razones para aprenderlo son por que pueden aparecer en el examen, es mejor tratar el tema, pues es poco probable que lo aprendas de otra manera. El resultado de usar estos operadores puede sorprenderte. Para entenderlos tienes que tener un pensamiento binario bastante fluido, por ejemplo conocer el valor de un bit en cada posición: 32, 16, 8, 4, 2, 1 Los programadores C/C++, pueden estar tranquilos ya que la manera de trabajar del operador de desplazamiento de bits en Java es menos ambigua que en C/C++. En C/C++ el desplazamiento hacia la derecha puede ser con signo o sin signo dependiendo de la implementación del compilador. Si vienes de programar con Visual Basic, bienvenido a la programación de bajo nivel. 106 javaHispano. Tu comunidad, tu lenguaje. Observa que el objetivo únicamente pide que comprendas el resultado de aplicar estos operadores sobre valores de tipo int. Esto es también aplicable a un byte o a un short, particularmente si el valor es negativo, ya que se puede tener algunos resultados demasiado inesperados. Operadores de desplazamiento con signo << y >> Los operadores de desplazamiento a la derecha y a la izquierda mueven el patrón de bits hacia la derecha o hacia la izquierda y dejan solo al bit de signo del número. Para números positivos los resultados son bastante predecibles. Así el desplazamiento con signo para números positivos da los siguientes resultados: int x = 14; int y = 0; y = x >>2; System.out.println(y); El resultado será 3,se pierde un bit y el resto del patrón de bits se inclina a la derecha int x = 14; int y = 0; y = x <<2; System.out.println(y); El resultado será 56 De manera que, ¿Qué esperarías obtener al realizar un desplazamiento a la izquierda a un numero negativo? Podrías esperar el mismo resultado obtenido al desplazar a la derecha un número positivo excepto que el resultado mantiene el signo negativo. Si desplazamos 4 posiciones, lo que sucede es que cada una de las posiciones, debido al moviendo de los otros bits, toman el valor del bit más significativo (el bit de signo). El efecto de esto es que cada desplazamiento divide/multiplica el número negativo por dos. Esto te será fácil de entender hasta que te des cuenta de las implicaciones del almacenamiento de números binarios a través del complemento a dos. El complemento a dos funciona un poco como el odómetro físico que se encuentra en el tablero de los coches. Imagina que lo "rebobinas" hasta cero y después pasas a números negativos. El primer número que obtendrías no sería 1, sería uno por debajo del máximo que pudieras representar con las ruedecillas disponibles. Si esto te suena improbable ejecuta la calculadora de Windows, ponla en modo científico, introduce un número negativo y después pasa a modo binario. Esto te mostrará el patrón de bits del número que hayas introducido. 107 Tutorial para el examen de certificación: SCJP 1.2 Si toda esta charla sobre patrones de bits y representación en complemento a dos "hace que tu cabeza heche humo", podrías preferir pensar el desplazamiento de bits como un proceso repetitivo de multiplicación o división por dos. Esta aproximación funciona bien hasta que empiezas a desplazar un numero negativo a la derecha, de modo que pierde bits del lado derecho. Desplazamiento sin signo a la derecha >>> El desplazamiento sin signo a la derecha >>> realiza un desplazamiento sin tomar en cuenta el bit del signo. Así, en un int, los 32 bits son desplazados el numero de veces que se le indique al operador y son rellenados los espacios de la izquierda con ceros. De tal manera que esto da el efecto de convertir un número negativo a positivo Este desplazamiento puede darnos algunos resultados muy raros. Veamos la siguiente sentencia: System.out.println(-1 >>>1); El resultado es el siguiente: 2147483647 El examen probablemente no te pedirá que des el resultado exacto pero podría darle algunas alternativas como 0, -1, etc. y tendrás que elegir el resultado mas aproximado. ¿Cual crees que sea el resultado para la siguiente línea de código? System.out.println(-1 >>> 32); Si acabas de leer que el deslazamiento sin signo no toma en cuenta el bit del signo, el resultado de la línea anterior, que es -1, puede ser una sorpresa. La razón es que el modulo 32 se realiza sobre el operando antes del desplazamiento. Así, si divides 32 entre 32 obtienes 0 y si realizas un desplazamiento sin signo de cero lugares hacia la derecha tendrás todavía como valor -1. No deseches esto como si fuera una peculiaridad irrelevante ya que puede venir en el examen. He creado un applet que te permite probar varias operaciones de desplazamiento y ver los resultados tanto en decimal como el patrón de bits. También se incluye el código para que veas como trabaja, puedes revisarlo en: http://www.jchq.net/applets/BitShift/BitShiftAr.html Aquí hay una imagen que muestra el applet en funcionamiento. Applet BitShift 108 javaHispano. Tu comunidad, tu lenguaje. Pregunta 1) Dadas las siguientes clases: interface IFace{} class CFace implements IFace{} class Base{} public class ObRef extends Base{ public static void main(String argv[]){ ObRef ob = new ObRef(); Base b = new Base(); Object o1 = new Object(); IFace o2 = new CFace(); } } ¿Cuales de las siguientes líneas son válidas? 1) o1=o2; 2) b=ob; 109 Tutorial para el examen de certificación: SCJP 1.2 3) ob=b; 4) o1=b; Pregunta 2) Dadas las siguientes declaraciones: String s = "Hello"; long l = 99; double d = 1.11; int i = 1; int j = 0; ¿Cuales de las siguientes líneas compilaran sin error? 1) j=i <<s; 2) j=i<<j; 3) j=i<<d; 4) j=i<<l; Pregunta 3) Dadas las siguientes variables: char c = 'c'; int i = 10; double d = 10; long l = 1; String s = "Hello"; ¿Cuales de las siguientes líneas compilaran sin error? 1) c=c+i; 2) s+=i; 3) i+=s; 4) c+=s; Pregunta 4) ¿Cual será el resultado de la siguiente sentencia? System.out.println(-1 >>>1); 1) 0 2) -1 3) 1 110 javaHispano. Tu comunidad, tu lenguaje. 4) 2147483647 Pregunta 5) ¿Cual será el resultado de la siguiente sentencia? System.out.println(1 <<32); 1) 1 2) -1 3) 32 4)-2147483648 Pregunta 6) ¿Cuales de las siguientes sentencias son validas? 1) 2) 3) 4) System.out.println(1+1); int i= 2+'2'; String s= "on"+'one'; byte b=255; Respuestas Respuesta 1) 1)o1=o2; 2)b=ob; 4)o1=b; Respuesta 2) 2)j= i<<j; 4)j=i<<l; Respuesta 3) 2)s+=i; Respuesta 4) 4) 2147483647 Aún y cuando no logres grabarte este numero, únicamente entendiendo al desplazamiento sin signo a la derecha te puedes dar cuenta que todas las otras opciones no son correctas. Respuesta 5) 111 Tutorial para el examen de certificación: SCJP 1.2 1) 1 Con el desplazamiento a la izquierda, los bits son "envueltos alrededor" del patrón de bits. Así, el resultado de: System.out.println(1 <<31); Será: -2147483648 Respuesta 6) 1) System.out.println(1+1); 2) int i= 2+'2'; La opción 3 no es valida, ya que las comillas simples sirven para indicar un solo caracter constante y no a una cadena. La opción 4 no compilara ya que el valor 255 esta fuera del rango de un byte. Otras Fuentes para este tema: El tutorial de Sun http://java.sun.com/docs/books/tutorial/java/nutsandbolts/operators.html No encontré nada relacionado con el operador instanceof Richard Baldwin http://home.att.net/~baldwin.dick/Intro/Java022.htm#bitwiseoperations No encontré nada relacionado con el operador instanceof Jyothi Krishnan cubre este tema en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec5.html#obj15 112 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2. Determinar el resultado de aplicar el método booleano equals(Object) sobre Objetos, y cualquier combinación de las clases java.lang.String, java.lang.Boolean y java.lang.Object. Diferencia entre el método equals y el operador = = El método equals puede verse como la realización de una comparación profunda del valor de un objeto, por el contrario el operador = = realiza una comparación poco profunda. El método equals compara lo que un objeto apunta en lugar del apuntador en si (esto si consideramos que java tiene punteros ). Usando el método equals con Strings El método equals retorna un valor primitivo de tipo boolean. Esto significa que se puede usar para manejar sentencias if, while u otra sentencia de enciclamiento. Puede usarse también en lugar del operador = = con una primitiva. La utilización del método equals y el operador = = tiene extraños efectos laterales cuando se usa para comparar Strings. Esta es una de las ocasiones en donde la naturaleza inmutable de los Strings, y la manera en que java maneja esta característica, puede crearte confusión. Existen 2 maneras para crear un String en Java. Una de ellas implica la utilización del operador new. Así, normalmente un String se crea de la siguiente manera: String s = new String("Hello"); La otra manera, ligeramente más corta, es la siguiente: String s = "GoodBye"; Generalmente la diferencia entre estas dos maneras de crear un String es mínima, pero en el examen pueden haber preguntas que te exijan reconocer muy bien esta diferencia. La creación de 2 cadenas con la misma secuencia de letras sin ocupar la palabra clave new creara 2 apuntadores a la misma cadena en el pool de Strings de Java. El pool de Strings es la manera en que Java mantiene los recursos. Veamos lo siguiente para comprender mejor lo anterior: String s = "Hello"; String s2 = "Hello"; if (s = = s2){ 113 Tutorial para el examen de certificación: SCJP 1.2 System.out.println("Equal without new operator"); } String t = new String("Hello"); String u = new String("Hello"); if (t == u){ System.out.println("Equal with new operator"); } Tomando en cuenta lo que vimos en el objetivo anterior, esperarías que el primer mensaje "Equal without new operator", nunca se mostrara ya que s y s2 son objetos distintos, y además el operador = = examina lo que apunta a un objeto, no su valor. Sin embargo, debido a la manera en que Java maneja los recursos con la re-utilización de cadenas idénticas que son creadas sin el operador new, s y s2 tiene la misma "direccion" y en consecuencia el código mostrará el mensaje: "Equal without new operator" Por otro lado, en el segundo par de Strings t y u, el operador new fuerza a Java a crear cadenas separadas, es decir, dos objetos distintos. Y debido a que el operador = = únicamente compara la dirección del objeto, y no su valor, t y u tienen diferentes direcciones y así el mensaje "Equal with new operator" nunca se mostrará. El método equals aplicado a Strings, sin importar la manera en que fueron creadas, ejecuta una comparación caracter a caracter. La manera en que trabaja el pool de String y la diferencia entre usar el operador = = y el método equals no es muy obvio, particularmente si tienes antecedentes de programar con Visual Basic. La mejor manera de comprender esto es creando algunos ejemplos por ti mismo y viendo como es que trabajan. Prueba permutando cadenas idénticas creadas con y sin el operador new. Usando el método equals con la Clase Boolean El requerimiento para comprender el uso del operador equals sobre java.lang.Boolean es una confusión potencial. Boolean es un objeto envolvente para la primitiva boolean De acuerdo a la documentación del JDK el método equals en la clase Boolean: "Retorna true si y solo si el argumento que recibe es distinto de null y si el objeto Boolean que recibe tiene el mismo valor booleano que éste objeto" 114 javaHispano. Tu comunidad, tu lenguaje. Ejemplo: Boolean b1 = new Boolean(true); Boolean b2 = new Boolean(true); if(b1.equals(b2)){ System.out.println("We are equal"); } Como un tema aparte sobre el uso de boolean y Boolean, una vez que te familiarices con la sentencia if en Java te darás cuenta que no puedes realizar la conversión implícita tan apreciada por los programadores C/C++. Esa conversión es del tipo: int x =1; if(x){ //realiza } algo, pero no en Java Esto no funcionara en Java porque el parámetro para la sentencia if debe ser una evaluación booleana, ya que Java no tiene el concepto de C/C++ en donde cualquier valor no nulo se considera verdadero. Sin embargo puedes realizar lo siguiente en Java : boolean b1=true; if(b1){ //do something in java } Aunque esta es una practica de programación bastante mala es sintácticamente correcta, ya que el parámetro para la sentencia if es un booleano. Usando el método equals con objetos Debido al diseño fundamental de Java una instancia de cualquier clase es también una instancia de la clase java.lang.Object. Examinaremos esto utilizando el método toString. Para la clase Object, el método toString simplemente retorna la dirección de memoria. Así el resultado es el equivalente de ejecutar una comparación con el operador = =. Como Java no esta diseñado para manipular direcciones de memoria o apuntadores, ésta no es una operación particularmente útil. Veamos el siguiente ejemplo: public class MyParm{ public static void main(String argv[]){ Object m1 = new Object(); Object m2 = new Object(); System.out.println(m1); 115 Tutorial para el examen de certificación: SCJP 1.2 System.out.println(m2); if (m1.equals(m2)){ System.out.println("Equals"); }else{ System.out.println("Not Equals"); } } } Si intentas compilar y ejecutar lo anterior, obtendrás algo similar a lo siguiente: java.lang.Object@16c80b java.lang.Object@16c80a Not Equals Estos valores son direcciones de memoria , y probablemente no tienen nada que ver con lo que tu necesitas. Pregunta 1) ¿Que pasará cuando intentes compilar y ejecutar el siguiente código? public class MyParm{ public static void main(String argv[]){ String s1= "One"; String s2 = "One"; if(s1.equals(s2)){ System.out.println("String equals"); } boolean b1 = true; boolean b2 = true; if(b1.equals(b2)){ System.out.println("true"); } } } 116 javaHispano. Tu comunidad, tu lenguaje. 1)Error al compilar 2)No se mostrará ningún mensaje 3)Se mostrará únicamente el mensaje "String equals" 4)Se mostrará el mensaje "String equals", seguido de "true" Pregunta 2) ¿Que pasará cuando intentes compilar y ejecutar el siguiente código? String s1= "One"; String s2 = new String("One"); if(s1.equals(s2)){ System.out.println("String equals"); } Boolean b1 = new Boolean(true); Boolean b2 = new Boolean(true); if(b1==b2){ System.out.println("Boolean Equals"); } 1)Error al compilar 2)Se mostrará únicamente "String equals" 3)Se mostrará "String equals" seguido de "Boolean Equals" 4)Se mostrará únicamente "Boolean Equals" Pregunta 3) ¿Cuál es el resultado de intentar compilar y ejecutar el siguiente código? Object o1 = new Object(); Object o2 = new Object(); o1=o2; if(o1.equals(o2)) System.out.println("Equals"); } 1)Error al compilar 2)Mostrará "Equals" 3)No mostrará nada 4)Error al ejecutarse 117 Tutorial para el examen de certificación: SCJP 1.2 Respuestas Respuesta 1) 1)Error al compilar la línea b1.equals causará un error debido a que b1 es un valor primitivo y los valores primitivos no tienen métodos. Si b1 hubiera sido creado son la clase Boolean entonces si podrías ocupar el método equals. Respuesta 2) 2)Se mostrará únicamente "String equals" Probar una instancia de la clase Boolean con el operador = = únicamente examina la dirección de memoria, no el valor de la instancia. Respuesta 3) 2)Mostrará "Equals" Debido a que una instancia de Object a sido asignada a la otra en la línea: o1=o2 Estas instancias apuntan ahora a la misma dirección de memoria y en consecuencia, la aplicación del método equals retornará verdadero. Otras Fuentes para este tema Jyothi Krishnan cubre este tema en http://www.geocities.com/SiliconValley/Network/3693/obj_sec5.html#obj16 Michael Thomas http://www.michael-thomas.com/java/JCP_Operators.htm#equals() 118 javaHispano. Tu comunidad, tu lenguaje. Objetivo 3. En una expresión que involucre a los operadores &,|,&&,||, y variables, conocer el estado de los valores que son evaluados con estos operadores y el valor de la expresión. Es fácil olvidarse cuales son los símbolos que significan una operación lógica y cuales significan operaciones a nivel de bits, asegúrate de reconocer la diferencia para el examen. Si eres nuevo con el uso de estos operadores vale la pena que te asegures no confundirte con los operadores a nivel de bits y los operadores lógicos. El efecto corto circuito con los operadores lógicos. Los operadores lógicos (&&, ||) tienen un ligero efecto peculiar en una operación lógica llamado "corto circuito”, como ocurre de manera similar, en C/C++. Esto quizá te pueda sorprender si vienes de programar con Visual Basic ya que Visual Basic evalúa todos los operandos. En Java esto tiene sentido si consideras que en una operación AND, si el primer operando es falso ya no es necesario evaluar el segundo operando, el resultado final será, con certeza, falso. Lo mismo ocurre con el operador lógico OR, si el primer operando ha resultado verdadero, el cálculo global será verdadero ya que sólo un operando del OR debe retorna verdadero para que el resultado final sea verdadero. Esto puede traer código un efecto especial con esos cálculos reducidos que dependen de efectos laterales. Veamos el siguiente ejemplo: public class MyClass1{ public static void main(String argv[]){ int Output=10; boolean b1 = false; if((b1==true) && ((Output+=10)==20)) { System.out.println("We are equal "+Output); }else { System.out.println("Not equal! "+Output); } } } El resultado de este código será "Not equal 10". Esto muestra que la operación Ouput+=10 jamás será realizada debido a que el proceso se detiene después de que se evalúa la primera operación, debido a que retorna falso, y al estar aplicando el operador 119 Tutorial para el examen de certificación: SCJP 1.2 AND (&&) ya no es necesario evaluar el resto de la expresión, pues el resultado será falso. Si cambias el valor de b1 a verdadero, la operación se realizará y el resultado será el que esperabas: "We are equal 20". Esto a veces puede ser conveniente cuando realmente no quieras procesar otras operaciones si una de ellas retorna falso, pero puede tener un efecto lateral inesperado si no estás completamente familiarizado con el uso de estos operadores. Operadores a nivel de bits. Los operadores | y & se aplican para realizar operaciones AND y OR a nivel de bits. Puedes encontrarte preguntas en el examen que te den números en decimal y te pidan que realices operaciones AND o OR a nivel de bits. Para hacer esto, necesitas familiarizarte con la conversión de números decimales a binarios y conocer que ocurre con el patrón de bits. Este es un ejemplo típico ¿Cuál es el resultado de la siguiente operación? 3|4 El patrón binario para el decimal 3 es 11, el patrón binario para 4 es 100, para realizar una operación OR a nivel de bits, cada bit es comparado con el bit que se encuentra en la misma posición del otro número. Si alguno de los bits es 1 el bit del numero resultante, en la posición que se esta comparando, es también 1. Así para esta operación el resultado será: 011 100 -----111 y 111 es 7 en decimal. Los objetivos no especifican que conozcas la operación XOR a nivel de bits. realizada con el operador ˆ. Pensando en binario Si no te siente muy cómodo pensando en binario (yo me siento mucho mas cómodo pensando en decimal), puedes querer hacer algunos ejercicios para ayudarte a dominar este tema y también el relacionado con los operadores de desplazamiento de bits. Si estás 120 javaHispano. Tu comunidad, tu lenguaje. ejecutando Windows puedes encontrar útil usar la calculadora que proporciona este sistema operativo en modo científico. Para hacer esto solo elige la opción Ver y cambiar del modo Estándar predefinido al modo científico. En modo científico puedes cambiar entre ver los números en formato decimal y binario, esto despliega el patrón de bits de los números. Y si lo prefieres puedes realizar un programa que muestre el patrón de bits de un numero, utilizando la clase Integer, he aquí un pequeño programa que muestra un ejemplo: public class BinDec{ public static void main(String argv[]){ System.out.println(Integer.parseInt("100",2)); System.out.println(Integer.toString(64,2)); } } Si lo compilas y ejecutas este será el resultado: 4 1000000 Observa que el programa convierte el patrón de bits 100 en su equivalente en decimal, que es 4; y convierte el decimal 64 en su patrón de bits equivalente. Pregunta 1) ¿Que pasará cuando intentes compilar y ejecutar el siguiente código? int Output=10; boolean b1 = false; if((b1= =true) && ((Output+=10)==20)){ System.out.println("We are equal "+Output); }else{ System.out.println("Not equal! "+Output); } 1)Error al compilar, por intentar realizar una comparación binaria sobre tipos de datos lógicos 2)Se compilara y mostrará la cadena "We are equal 10" 121 Tutorial para el examen de certificación: SCJP 1.2 3)Se compilara y mostrará la cadena "Not equal! 20" 4)Se compilara y mostrará la cadena "Not equal! 10" Pregunta 2) Qué mostrará la siguiente línea de código System.out.println(010|4); 1) 14 2) 0 3) 6 4) 12 Pregunta 3) Cuales de los siguientes fragmentos de código se compilara sin errores 1) int i=10; int j = 4; System.out.println(i||j); 2) int i=10; int j = 4; System.out.println(i|j); 3) boolean b1=true; boolean b2=true; System.out.println(b1|b2); 4) boolean b1=true; boolean b2=true; System.out.println(b1||b2); Respuestas) Respuesta 1 4)Se compilara y mostrará la cadena "Not equal! 10" Respuesta 2) 122 javaHispano. Tu comunidad, tu lenguaje. 4)12 La respuesta a esta pregunta requiere que conozcas la notación octal, el cero que antecede al primer numero indica que dicho número esta en notación octal, 010 es la notación octal para el numero 8 decimal. Así, la instrucción: System.out.println(010|4) es equivalente a: System.out.println(8|4); que en binario es: 1000 0100 ----1100 y 1100 en decimal es 12 Repuesta 3) 2,3,4 La opción 1 no compilara ya que estas intentando ejecutar una aplicación OR lógica sobre tipos enteros. Y una operación lógica solo se puede realizar con argumentos de tipo boolean. Otras Fuentes para este tema Tutorial de Sun: http://java.sun.com/docs/books/tutorial/java/nutsandbolts/operators.html Richard Baldwin http://home.att.net/~baldwin.dick/Intro/Java022f.htm 123 Tutorial para el examen de certificación: SCJP 1.2 Objetivo 4 Determinar los efectos sobre los objetos y los valores primitivos al pasarlos como variables en métodos al aplicarles asignaciones u otras modificaciones dentro del método. Nota sobre este objetivo El objetivo parece estarte pidiendo que entiendas lo que ocurre cuando pasas un valor a un método. Si el código dentro del método modifica el valor de la variable, ¿El cambio es visible aun fuera del método? Si vienes de programar con C/C++ estarás familiarizado con el concepto "paso de parámetros por valor" y "paso de parámetros por referencia" con el operador &. En java no hay tal opción ya que todo es pasado por valor. Sin embargo esto no siempre es así. Si pasas un objeto este es una referencia a un objeto y no puedes manipular directamente una referencia a un objeto. Así, si manipulas un campo de un objeto que se pasa a un método, el campo se comportara como si lo hubieras pasado por referencia (esto quiere decir que cualquier cambio mantendrá su efecto aún después de terminada la llamada al método). Referencias de Objetos como parámetros de métodos. Veamos el siguiente ejemplo: class ValHold{ public int i = 10; } public class ObParm{ public static void main(String argv[]){ ObParm o = new ObParm(); o.amethod(); } public void amethod(){ ValHold v = new ValHold(); v.i=10; System.out.println("Before another = "+ v.i); another(v); System.out.println("After another = "+ v.i); }//End of amethod public void another(ValHold v){ v.i = 20; System.out.println("In another = "+ v.i); }//End of another 124 javaHispano. Tu comunidad, tu lenguaje. } El resultado será: Before another = 10 In another = 20 After another = 20 Observa como el valor de la variable i ha sido modificado. Si java siempre pasa por valor, ¿Cómo se ha modificado? Bueno, el método recibió una copia del manejador o referencia del objeto pero esa referencia actúa como un apuntador al valor real del objeto y las modificaciones se reflejan en los campos a los que apunta. Esto es parecido a la desreferenciación automática de punteros en C/C++. Primitivas como parámetros de métodos Cuando pasas valores primitivos a métodos, estos pasan por valor. El método obtiene su propia copia para utilizarla y en consecuencia, cualquier modificación no se refleja fuera del método. Veamos el siguiente ejemplo: public class Parm{ public static void main(String argv[]){ Parm p = new Parm(); p.amethod(); }//End of main public void amethod(){ int i=10; System.out.println("Before another i= " +i); another(i); System.out.println("After another i= " + i); }//End of amethod public void another(int i){ i+=10; System.out.println("In another i= " + i); }//End of another } La salida será: Before another i= 10 In another i= 20 After another i= 10 125 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 1) Dado el siguiente código: class ValHold{ public int i = 10; } public class ObParm{ public static void main(String argv[]){ ObParm o = new ObParm(); o.amethod(); } public void amethod(){ int i = 99; ValHold v = new ValHold(); v.i=30; another(v,i); System.out.println(v.i); }//End of amethod public void another(ValHold v, int i){ i=0; v.i = 20; ValHold vh = new ValHold(); v = vh; System.out.println(v.i+ " "+i); }//End of another } ¿Cual será la salida? 1)10,0, 30 2)20,0,30 3)20,99,30 4)10,0,20 Respuestas Respuesta 1 4)10,0,20 126 javaHispano. Tu comunidad, tu lenguaje. Otras Fuentes para este tema: Tutorial de SUN: http://java.sun.com/docs/books/tutorial/java/javaOO/arguments.html Jyothi Krishnan cubre este tema en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec5.html#obj18 127 Tutorial para el examen de certificación: SCJP 1.2 Sección 6. Sobrecarga, sobreescritura, tipos en tiempo de ejecución y orientación a objetos Objetivo 1) Mostrar los beneficios de la encapsulación en el diseño orientado a objetos y escribir código que implemente clases fuertemente encapsuladas y entender las relaciones “es un” y “tiene un”. Relaciones “es un” y “tiene un” Éste es un tema básico de OO y probablemente te encontrarás una pregunta sobre él en el examen. Esencialmente busca descubrir si entiendes cuándo algo se refiere al tipo de estructura de clases a la que pertenece un objeto y cuándo se refiere a un método o campo que posee una clase. Así, un gato ES UN tipo de animal y un gato TIENE UNA cola. Desde luego la distinción puede ser difusa. Si fueras un zoólogo y conocieras los nombres correctos para los grupos de tipos de animales podrías decir que un gato ES UN: Largapalabraenlatínparagrupodeanimalesconcolas Pero en lo que al examen se refiere esto no es un requisito. Las preguntas del examen tienden a ser del tipo en las que se te da un texto que describe una jerarquía potencial y te hace preguntas como: qué debería ser un campo y qué debería ser un nuevo tipo de clase como hijo. A primera vista estas cuestiones pueden parecer complejas, pero son bastante obvias si las lees con cuidado Encapsulamiento Los objetivos de Java 1.1 no mencionaban específicamente la encapsulación aunque sería raro estudiar Java y no encontrarse con el concepto. La encapsulación conlleva a separar la interfaz de una clase de su implementación. Esto significa que no se puede corromper el valor de un campo “accidentalmente”, hay que usar un método para cambiar un valor. La encapsulación conlleva esconder los datos de una clase y permitir el acceso solo mediante una interfaz pública. 128 javaHispano. Tu comunidad, tu lenguaje. Esto usualmente conlleva la creación de variables privadas (campos) cuyo valor se actualiza y recupera mediante métodos. La convención estándar para dar nombre a estos métodos es: • SetNombreCampo • GetNombreCampo Por ejemplo, si cambiaras el color de una figura podrías crear un par de métodos de la siguiente manera public void setColor(Color c){ cBack = c; } public Color getColor(){ return cBack; } Las principales palabras claves para controlar el acceso a las variables son: • public • private • protected No te confundas pensando que el sistema de control de acceso tiene que ver con la seguridad. Este no está diseñado para evitar que un programador acceda a variables, está diseñado para ayudar a evitar modificaciones no deseadas. El enfoque estándar al usar el ejemplo de arriba de color sería hacer privado el campo CBack de Color. Un campo privado sólo es visible desde la clase en cuestión. Esto significa que un programador no puede accidentalmente escribir código que cambie el valor desde otra clase. Esto puede ayudar a reducir la introducción de errores. La separación de la interfaz y la implementación hace más fácil modificar el código de una clase sin afectar a otro código que lo pueda usar. Para el diseñador de clases esto lleva a la posibilidad de modificar una clase, sabiendo que no estropeará programas que la usen. Un diseñador de clases puede añadir rutinas de comprobación adicionales para realizar “chequeos de cordura” relacionados con modificaciones de campos. He trabajado en proyectos de seguros en los que era posible que el cliente tuviera una edad menor de cero. Si un valor así se guarda en un campo simple como un entero, no hay un lugar obvio en el que se puedan guardar rutinas de comprobación. Si la edad es accesible sólo vía métodos set y get será posible insertar comprobaciones de edades cero o negativas de forma que no quede afectado código existente. Desde luego a medida que el desarrollo continúa se pueden descubrir más situaciones que necesiten chequeo. Para el usuario final de la clase esto significa que no tiene que comprender el funcionamiento interno y que se le presenta una interfaz claramente definida para manejar los datos. El usuario final puede tener la confianza de que las actualizaciones en el código de la clase no romperán su código existente. 129 Tutorial para el examen de certificación: SCJP 1.2 Tipos en Tiempo de ejecución Como el polimorfismo permite seleccionar en tiempo de ejecución la versión que se ejecuta de un método, a veces no es obvio cuál será el método ejecutado. Véase el siguiente ejemplo. class Base { int i=99; public void amethod(){ System.out.println("Base.amethod()"); } } public class RType extends Base{ int i=-1; public static void main(String argv[]){ Base b = new RType();//<= Nótese el tipo System.out.println(b.i); b.amethod(); } public void amethod(){ System.out.println("RType.amethod()"); } } Nótese cómo el tipo de la referencia es la variable b es la clase Base pero el tipo de la clase de hecho es RType. La llamada a amethod invocará la versión en RType pero la salida b.i referenciará el campo i en la clase Base. Pregunta 1) Supón que te han entregado el siguiente diseño: “Una persona tiene nombre, edad, dirección y sexo. Estás diseñando una clase para representar un tipo de persona llamada paciente. Este tipo de persona puede recibir un diagnóstico, tener cónyuge y está vivo". Dado que la clase persona ya ha sido creada, qué de entre lo siguiente sería apropiado incluir cuando se diseñe la clase paciente? 130 javaHispano. Tu comunidad, tu lenguaje. 1)fecha de registro 2)edad 3)sexo 4)diagnóstico Pregunta 2) ¿Qué sucederá cuando intentes compilar y ejecutar el siguiente código? class Base { int i=99; public void amethod(){ System.out.println("Base.amethod()"); } Base(){ amethod(); } } public class RType extends Base{ int i=-1; public static void main(String argv[]){ Base b = new RType(); System.out.println(b.i); b.amethod(); } public void amethod(){ System.out.println("RType.amethod()"); } } 1) RType.amethod -1 RType.amethod 2) RType.amethod 99 RType.amethod 3) 99 RType.amethod 131 Tutorial para el examen de certificación: SCJP 1.2 4) Error en tiempo de compilación Pregunta 3) Tu jefe diseñador de software te ha mostrado un esquema del nuevo sistema de partes de ordenador que está a punto de crear. En lo alto de la jerarquía está una clase llamada Ordenador y bajo ella hay dos clases hijas. Una se llama LinuxPC y la otra WindowsPC. La principal diferencia entre las dos es que una ejecuta el Sistema Operativo Linux y la otra ejecuta el sistema Windows (desde luego otra diferencia es que una necesita ser rearrancada constantemente y la otra funciona fiablemente). Bajo WindowsPC hay dos subclases, una llamada Server y la otra llamada Workstation. ¿Cómo valorarías el trabajo de tu diseñador? 1)Dar la luz verde para posteriores diseños usando el esquema actual 2)Pedir un rediseño de la jerarquía cambiando el Sistema Operativo a un campo de la clase antes que a un tipo de clase 3)Pedir que la opción WindowsPC se elimine ya que pronto quedará obsoleta 4) Cambiar la jerarquía para eliminar la necesidad de la superflua clase Ordenador. Pregunta 4) Dada la siguiente clase class Base{ int Edad=33; } ¿Cómo mejorarías la clase con respecto al acceso al campo Edad? 1)Definir la variable Edad como privada 2)Definir la variable Edad como protegida 3)Definir la variable Edad como privada y crear un método ‘get’ que la devuelva y un método ‘set’ que la actualice 4)Definir la variable Edad como protegida y crear un método ‘set’ que la devuelva y un método ‘get’ que la actualice Pregunta 5) ¿Cuáles de los siguientes son beneficios de la encapsulación? 1)Todas las variables pueden manipularse como objetos en vez de con primitivas 2)Haciendo todas las variables protegidas están protegidas de una corrupción accidental 3)Se puede cambiar la implementación de una clase sin estropear el código que la usa 4)Hacer todos los métodos protegidos previene la corrupción accidental de datos. 132 javaHispano. Tu comunidad, tu lenguaje. Pregunta 6) Nombre tres características principales de la Programación Orientada a Objetos: 1)Encapsulación, ligadura dinámica, polimorfismo 2)Polimorfismo, sobrecarga, redefinición 3)Encapsulación, herencia, ligadura dinámica 4)Encapsulación, herencia, polimorfismo Pregunta 7) ¿Como implementas la encapsulación en una clase? 1)Haciendo todas las variables protegidas y sólo permitiendo el acceso vía métodos 2)Haciendo todas las variables privadas y sólo permitiendo el acceso vía métodos 3)Asegurando que todas las variables son representadas por clases envolventes 4)Asegurando que todas las variables son accedidas a través de métodos de una clase ancestra. Respuestas Respuesta 1) 1) fecha de registro 4) diagnóstico La fecha de registro es razonable como campo adicional para un paciente, y el diseño específicamente dice que un paciente debería tener un diagnóstico. Como el paciente es un tipo de persona, debería tener disponibles los campos edad y sexo (asumiendo que no fueran declarados como privados). Respuesta 2) 2) RType.amethod 99 RType.amethod Si esta respuesta te parece improbable, intenta compilar y ejecutar el código. La razón es que este código crea una instancia de la clase Rtype pero la asigna a una referencia a la clase Base. En esta situación una referencia a cualquiera de los campos como es el caso de ‘i’se referirá al valor en la clase Base, pero una llamada a un método hará referencia al método en la clase Rtype antes que a su manejador de referencia. Respuesta 3) 2) Pedir un rediseño de la jerarquía cambiando el Sistema Operativo a un campo de la clase antes que a un tipo de clase Respuesta 4) 133 Tutorial para el examen de certificación: SCJP 1.2 3) Definir la variable Edad como privada y crear un método get que la devuelva y un método set que la actualice. Respuesta 5) 3) Se puede cambiar la implementación de una clase sin estropear el código que la usa. Respuesta 6) 4) encapsulación, herencia, polimorfismo Me hicieron esta pregunta en una entrevista de trabajo una vez. Conseguí el trabajo. No puedes tener la seguridad de encontrar algo así en el examen, pero bien viene saberlo. Respuesta 7) 2)Haciendo todas las variables privadas y sólo permitiendo el acceso vía métodos Otras Fuentes para este tema: Este tema se cubre en el Tutorial de Sun en: http://java.sun.com/docs/books/tutorial/java/concepts/index.html Richard Baldwin cubre este tema en: http://www.Geocities.com/Athens/Acropolis/3797/Java004.htm#an initial description of oop (Esto es material general sobre OOP más que sobre "es un" / "tiene un") Jyothi Krishnan sobre este tema en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec6.html#obj19 Java 1.1 Unleashed http://www.itlibrary.com/reference/library/1575212986/htm/ch05.htm (Ver la sección sobre encapsulación) http://www.geocities.com/SiliconValley/Network/3693/obj_sec6.html - obj19 134 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2) Escribir código para invocar métodos sobrecargados y sobrescritos; constructores paternos o sobrecargados y describir el efecto de invocar estos métodos. Comentario sobre el objetivo Los términos overloaded (sobrecargado) y overriden (sobrescrito, redefinido o sobrepasado) son en inglés lo bastante similares como para llevar a la confusión. Mi forma de recordarlo es imaginar que algo que ha sido redefinido o sobrepasado es algo a lo que le ha pasado encima un vehículo pesado y ya no existe por sí mismo. Algo que ha sido sobrecargado todavía se mueve pero está cargado con montones de funcionalidades que le causan un gran esfuerzo. Esto es sólo un pequeño truco para distinguir a ambos, no tiene ninguna relación con la realidad del funcionamiento de Java. Sobrecarga de métodos La sobrecarga de métodos es un truco del compilador para permitirte usar el mismo nombre para realizar acciones distintas dependiendo de los parámetros. Entonces, imagina que estás diseñando la interfaz para un sistema que haga exámenes de certificación de Java falsos (¿quién podría saber?). Una respuesta podría venir como un entero, un booleano o una cadena de texto. Podrías crear una versión del método para cada tipo de parámetro y darle el nombre correspondiente como: marcarespuestaboolean(boolean respuesta){ } marcarespuestaint(int respuesta){ } marcarespuestaString(String respuesta){ } Esto funcionaría pero supone que los futuros usuarios de tus clases tendrán que saber más nombres de métodos de lo estrictamente necesario. Sería más útil si pudieras usar un solo nombre de método y el compilador resolviera a qué código llamar según el tipo y el número de parámetros de la llamada. Este es el núcleo de la sobrecarga de métodos, parte de lo cual se conoce como polimorfismo. No hay palabras claves que recordar para sobrecargar métodos, sólo se crean múltiples métodos con el mismo nombre pero distinto número y/o tipo de parámetros. Los nombres 135 Tutorial para el examen de certificación: SCJP 1.2 de los parámetros no son importantes pero el número y los tipos sí deben ser diferentes. Así, el siguiente es un ejemplo de un método marcarespuesta sobrecargado void marcarespuesta(String respuesta){ } void marcarespuesta(int respuesta){ } El siguiente no es un ejemplo de sobrecarga y causará un error en tiempo de compilación indicando una declaración duplicada de método. void marcarespuesta(String respuesta){ } void marcarespuesta(String titulo){ } El tipo de retorno no forma parte de la identificación en cuanto a la sobrecarga. Por tanto, cambiárselo a uno de los métodos de arriba por int aún producirá un error en tiempo de compilación, pero esta vez indicando que no se puede redefinir un método con un tipo de retorno distinto. Los métodos sobrecargados no tienen restricciones en cuanto a qué excepciones se pueden hacer lanzar (hacer throw). Esto sí es algo a tener en cuenta con la redefinición. Los métodos se diferencian sólo en el número, tipo y orden de los parámetros, no en el tipo de retorno del método Sobrescritura de métodos Sobrescribir un método significa que se reemplaza su funcionalidad entera. Se hace en una clase hija con un método definido en una clase padre. Para redefinir un método se define uno nuevo en la clase hija con exactamente la misma identificación que el de la clase padre. Esto tiene el efecto de obscurecer el método en la clase padre y de que la funcionalidad de éste ya no sea directamente accesible. Java proporciona un ejemplo de redefinición en el caso del método equals que cada clase hereda de la gran clase antecesora Object. La versión heredada de equals 136 javaHispano. Tu comunidad, tu lenguaje. simplemente compara a qué lugar de la memoria referencia la instancia de la clase. A menudo esto no es lo que se desea, particularmente en el caso de un String. Para una cadena generalmente querrías una comparación carácter a carácter para ver si las dos cadenas son la misma. Para permitir esto la versión de equals que viene con String es una versión redefinida que efectúa esta comparación carácter a carácter. Invocando constructores de clases base Un constructor es un método especial que se ejecuta automáticamente cada vez que se crea una instancia de una clase. Java sabe que un método es un constructor porque tiene el mismo nombre que la clase misma y no tiene valor de retorno. Un constructor puede recibir parámetros como cualquier otro método y puedes necesitar diferentes parámetros según cómo quieras que la clase se inicialice. Así si te fijas en el ejemplo en la clase Button del paquete AWT su constructor está sobrecargado para que tenga dos versiones. • • Button() Button(String label) Así, puedes crear un botón sin etiqueta y dársela después, o usar la opción más común y asignarle la etiqueta en el momento de la creación. Sin embargo, los constructores no se heredan, así que si quieres sacar un constructor útil de una clase ancestro no está disponible por defecto. Así, el siguiente código no compilará class Base{ public public } Base(){} Base(int i){} public class MiRedef extends Base{ public static void main(String argvp[]){ MiRedef m = new MiRedef(10);// NO compilará } } La palabra mágica que necesitas para acceder a un constructor de un ancestro es super. Esta palabra clave se puede usar como si fuera un método y recibir parámetros que concuerden con la versión deseada del constructor padre. En este ejemplo modificado del código previo se usa la palabra clave super para llamar a la versión con el entero del constructor en la clase base y el código se compila sin problemas. 137 Tutorial para el examen de certificación: SCJP 1.2 class Base{ public Base(){} public Base(int i){} } public class MiRedef extends Base{ public static void main(String arg[]){ MiRedef m = new MiRedef(10); } MiRedef(int i){ super(i); } } Invocando constructores con this() De la misma forma que se puede llamar al constructor de una clase base usando super() se puede llamar a otro constructor de la clase en cuestión usando this como si fuera un método. Así en el ejemplo anterior se podría definir otro constructor de la siguiente manera MiRedef(String s, int i){ this(i); } Tanto this como super se pueden llamar en la primera línea desde un constructor, pero no ambos a la vez. Como habrás adivinado esto llamará al otro constructor de la clase actual que toma un solo entero como parámetro. Si usas super() o this() en un constructor debe ser la primera llamada a un método. Como sólo uno de los dos puede ser el primer método llamado, no puedes usar a la vez super() y this() en un constructor. Así, lo siguiente causará un error en tiempo de compilación. MiRedef (String s, int i){ this(i); super();// Causa un error en tiempo de compilación. } 138 javaHispano. Tu comunidad, tu lenguaje. Basándonos en el conocimiento de que los constructores no se heredan, debe resultar obvio que la redefinición es irrelevante. Si tienes una clase llamada Base y creas una clase hija que la extiende, para la clase que la extiende redefinir el constructor debe tener el mismo nombre. Esto causaría un error en tiempo de compilación. Aquí hay un ejemplo de esta jerarquía absurda. class Base{} class Base extends Base{} // ¡Error en tiempo de compilación! El constructor y la jerarquía de clases Los constructores siempre se llaman hacia abajo desde lo alto de la jerarquía. Es muy probable que en el examen te hagan preguntas relativas a una jerarquía de clases con varias llamadas a this y super y tengas que elegir cuál será el resultado. Ten cuidado con las preguntas en las que tienes una jerarquía compleja que resulta irrelevante a causa de un constructor que hace una llamada tanto a this como a super y por tanto se produce un error en tiempo de compilación. Los constructores se llaman desde la base (ancestro) de la jerarquía hacia abajo Sea el siguiente ejemplo: class Mamífero{ Mamífero (){ System.out.println("Creando Mamífero "); } } public class Humano extends Mamífero { public static void main(String argv[]){ Humano h = new Humano(); } Humano(){ System.out.println("Creando Humano"); } } 139 Tutorial para el examen de certificación: SCJP 1.2 Cuando se ejecute este código la cadena “Creando Mamífero” aparece primero debido a la llamada implícita al constructor sin argumentos de la base de la jerarquía. Pregunta 1) Dada la siguiente definición de clase, ¿cuál de los siguientes métodos podría colocarse legalmente después del comentario con la palabra comentada “//Aquí”? public class Rid{ public void amethod(int i, String s){} //Aquí } 1) public void amethod(String s, int i){} 2) public int amethod(int i, String s){} 3) public void amethod(int i, String mystring){} 4) public void Amethod(int i, String s) {} Pregunta 2) Dada la siguiente definición de clase, ¿cuál de los siguientes métodos podría colocarse legalmente después del comentario con la palabra comentada “//Aquí”? 140 javaHispano. Tu comunidad, tu lenguaje. public class MyOver extends Base{ public static void main(String arg[]){ MyOver m = new MyOver(10); } MyOver(int i){ super(i); } MyOver(String s, int i){ this(i); //Aquí } } 1) MyOver m = new MyOver(); 2) super(); 3) this("Hola",10); 4) Base b = new Base(10); Pregunta 3) Dada la siguiente definición de clases: class Mamífero{ Mamífero (){ System.out.println("Mamífero "); } } class Perro extends Mamífero { Perro(){ System.out.println("Perro"); } } public class Collie extends Perro { public static void main(String argv[]){ Collie c = new Collie(); } Collie(){ this("Buen Perro"); System.out.println("Collie"); 141 Tutorial para el examen de certificación: SCJP 1.2 } Collie(String s){ System.out.println(s); } } ¿Cuál será la salida? 1)Error al compilar 2)Mamífero, Perro, Buen Perro, Collie 3)Buen Perro, Collie, Perro, Mamífero 4)Buen Perro Collie Pregunta 4) ¿Cuáles de las siguientes sentencias son verdaderas? 1)Los constructores no se pueden heredar 2)Un constructor se puede sobrescribir 3)Un constructor padre se puede invocar utilizando this 4)Cualquier método puede contener una llamada a this o super Pregunta 5) ¿Qué pasara cuando intentes compilar y ejecutar el siguiente código? class Base{ public void amethod(int i, String s){ System.out.println("Base amethod"); } Base(){ System.out.println("Base Constructor"); } } public class Child extends Base{ int i; String Parm="Hello"; public static void main(String argv[]){ Child c = new Child(); c.amethod(); } void amethod(int i, String Parm){ 142 javaHispano. Tu comunidad, tu lenguaje. super.amethod(i,Parm); } public void amethod(){} } 1)Error al compilar 2)Error causado por la sintaxis ilegal en: super.amethod(i,Parm); 3)Mostrará “Base Constructor” 4)Error causado por los nombres incorrectos de los parámetros en la llamada a; super.amethod Pregunta 6) ¿Cuál será la salida del siguiente código si intentas compilarlo y ejecutarlo? class Mammal{ Mammal(){ System.out.println("Four"); } public void ears(){ System.out.println("Two"); } } class Dog extends Mammal{ Dog(){ super.ears(); System.out.println("Three"); } } public class Scottie extends Dog{ public static void main(String argv[]){ System.out.println("One"); Scottie h = new Scottie(); } } 1) One, Three, Two, Four 2) One, Four, Three, Two 3) One, Four, Two, Three 4) Error al compilar Respuestas Respuesta 1) 143 Tutorial para el examen de certificación: SCJP 1.2 1) public void amethod(String s, int i){} 4) public void Amethod(int i, String s) {} La letra A mayúscula en Amethod nos indica que es un método distinto a amethod Respuesta 2) 4) Base b = new Base(10); Cualquier llamada a this o super debe estar en la primera línea del constructor. Como en el método ya se utilizó this, ninguna otra opción puede ser utilizada Respuesta 3) 2) Mamífero, Perro, Buen Perro, Collie Respuesta 4) 1)Los constructores no se pueden heredar Los constructores padres con invocados utilizando this o super Repuesta 5) 1)Error al compilar Esto causa un error que indica algo como lo siguiente: “no puedes sobrescribir métodos para hacerlos privados”, amethod en la clase Base esta definido como privado, y en Child al no ponerle modificador de acceso antes de void, estas intentando alterar el modificador public. Respuesta 6) 3) One, Four, Two, Three Las clases son creadas desde la raíz del la jerarquía de herencia hacia abajo. 144 javaHispano. Tu comunidad, tu lenguaje. Otras Fuentes para este tema: Este tema es cubierto el Tutorial de SUN en: http://java.sun.com/docs/books/tutorial/java/javaOO/methoddecl.html Richard Baldwin cubre este tema en: http://www.Geocities.com/Athens/Acropolis/3797/Java004.htm#polymorphism in general (Material sobre POO concerniente a "es un" y “tiene un") Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec6.html#obj20 145 Tutorial para el examen de certificación: SCJP 1.2 Objetivo 3) Escribir código que construya instancias de una clase en concreto, incluyendo clases de nivel superior, clases internas, clases internas estáticas y clases anónimas. Nota sobre este objetivo Parte de este material es cubierto en la sección 4. Instanciando una clase Concretamente, las clases pueden ser instanciadas como una referencia a un objeto (también llamada simplemente objeto). De esta manera, una clase abstracta no puede ser instanciada y por lo tanto no se puede crear una referencia a un objeto. Recuerda que una clase que contiene un método abstracto es abstracta por si misma y no puede ser instanciada. La manera para instanciar una clase es usando la palabra clave new. Típicamente se utiliza de la siguiente manera: Button b = new Button(); La sintaxis indica que la variable name es del tipo Button y contiene una referencia a una instancia de Button. Sin embargo aunque el tipo de la referencia es frecuentemente el mismo que el tipo de la clase instanciada, esto no necesariamente tiene que se así. El siguiente código es valido: Object b = new Button(); La sintaxis anterior indica que el tipo de la referencia de b es Object en lugar de Button. La declaración e instanciación puede no ocurrir en la misma línea, Así, puedes construir una instancia de una clase de la siguiente manera: Button b; b=new Button(); Clases internas 146 javaHispano. Tu comunidad, tu lenguaje. Las clases internas fueron introducidas con la liberación del JDK 1.1. Las clases internas permiten definir clases dentro de otras clases, en ocasiones estas clases son conocidas como clases anidadas. Son usadas extensivamente a partir del modelo de eventos de JDK 1.1. Con certeza encontraras en el examen preguntas acerca de las clases anidadas y del alcance de ellas. Este es un ejemplo sencillo: class Nest { class NestIn{} } La salida cuando este código se compile serán dos archivos .class. El primero, que es el que esperas es: Nest.class el otro es Nest$NestIn.class Esto ilustra que el anidamiento de clases es generalmente una convención para nombrar clases en lugar de una nueva clasificación de archivos .class. Las clases internas te permite hacer una agrupación lógica de clases. También tienen beneficios para controlar el alcance de las variables. Clases Anidadas de nivel Superior (Top Level) Una clase anidada de nivel superior es un miembro estático contenido dentro de una clase de nivel superior. class Nest{ static class NestIn{} } Este tipo de anidamiento es usado frecuentemente para agrupar de una manera sencilla clases que guardan una relación entre si. Debido a que la clase es estática no requiere una instancia de la clase externa para existir. Clases Miembro Veo una clase miembro como una “clase interna ordinaria”. Una clase miembro es semejante a otros miembros de una clase, debes instanciar la clase exterior antes de crear una instancia de la clase interna. Debido a la necesidad que tiene la clase interna de ser asociada con una instancia de la clase externa, Sun introduce una nueva sintaxis para 147 Tutorial para el examen de certificación: SCJP 1.2 permitir la creación simultánea de una instancia de la clase externa y al mismo tiempo la creación de la clase interna. Es de esta forma: Outer.Inner i = new Outer().new Inner(); Para darle sentido a esta nueva sintaxis, intenta pensar que la primera aparición de la palabra clave new es usada en el ejemplo anterior para designar la pertenencia a la instancia actual this. Así, podrías cambiar la línea que crea la instancia de ésta clase y leer lo siguiente: Inner i = this.new Inner(); Debido a que una clase miembro no puede existir sin un instancia de la clase exterior, ésta puede tener acceso a las variables de la clase exterior. Clases creadas dentro de métodos Un nombre más correcto para este tipo de clases es clase local. Las clases locales solo pueden acceder a campos finales o parámetros del método que la contiene Una clase local es visible únicamente dentro del bloque de código del método. El código dentro de la clase local puede usar únicamente variables locales que tengan el modificador final en el bloque del método o parámetros del método. Es muy probable que te encuentres con preguntas de este tipo en el examen. Clases anónimas Tu primera reacción con la idea de una clase anónima debe ser "¿por qué querría yo tener una clase sin nombre? y ¿cómo podría referirme a una clase si no tiene nombre?" Para contestar estas preguntas, considera lo siguiente. Podrías estar en una situación en donde constantemente tienes que pensar en nombres para instancias de clases en donde el nombre es por si mismo evidente. Por ejemplo en el manejo de eventos las dos cosas mas importantes que necesitas saber para manejar el evento son: el evento que se va a manejar y el nombre del componente que esta utilizando dicho manejador. Tener el nombre de la instancia la clase que maneja el evento no aporta mucho valor. 148 javaHispano. Tu comunidad, tu lenguaje. Acerca de la pregunta de como puedes hacerle referencia, pues, simplemente no puedes y si necesitas referirte a la clase por su nombre, entonces no deberías crear una clase anónima. La falta del nombre de la clase tiene efectos adicionales, como el hecho de que no puedes darle a la clase anónima un constructor. Las clases anónimas no pueden tener constructores Este es un ejemplo de una clase interna anónima: class Nest{ public static void main(String argv[]){ Nest n = new Nest(); n.mymethod(new anon(){}); } public void mymethod(anon i){} } class anon{} Nota que la declaración y definición de la clase anónima se realiza dentro de los paréntesis de la llamada a mymethod. Pregunta 1) ¿Cuáles de las siguientes afirmaciones son verdaderas? 1)Una clase definida dentro de un método puede únicamente acceder métodos estáticos del método que la contiene 2)Una clase definida dentro de un método puede únicamente acceder variables finales del método que la contiene 3)Una clase definida dentro de un método no puede acceder a algún campo del método que la contiene 149 Tutorial para el examen de certificación: SCJP 1.2 4)Una clase definida dentro de un método puede acceder cualquier campo accesible del método que la contiene Pregunta 2) ¿Cuáles de las siguientes afirmaciones son verdaderas? 1)Una clase anónima no puede tener un constructor 2)Una clase anónima puede ser creada únicamente dentro del cuerpo de un método 3)Una clase anónima puede únicamente acceder a campos estáticos del método que la contiene 4)El tipo de clase de una clase anónima puede ser obtenido utilizando el método getName Pregunta 3) ¿Cuáles de las siguientes afirmaciones son verdaderas? 1)Una clase interna no puede ser marcada como privada 2) Una instancia de una clase anidada de nivel superior puede crearse sin necesidad de crear una instancia de la clase que la contiene 3)Un archivo que contenga una clase externa y una interna únicamente producirá un único archivo .class 4) Para crear una instancia de una clase miembro se requiere crear una instancia de la clase que la contiene Respuestas Respuesta 1) 2)Una clase definida dentro de un método puede únicamente acceder variables finales del método que la contiene Respuesta 2) 1)Una clase anónima no puede tener un constructor Respuesta 3) 2) Una instancia de una clase anidada de nivel superior puede crearse sin necesidad de crear una instancia de la clase que la contiene 4) Para crear una instancia de una clase miembro se requiere crear una instancia de la clase que la contiene 150 javaHispano. Tu comunidad, tu lenguaje. Una clase anidada genera por si misma un archivo .class, usando el formato: Outer$Inner.class Una clase anidada de nivel superior es como una clase estática así que no requiere una instancia de la clase que la contiene. Una clase miembro es como una clase no estática, por consiguiente se requiere una instancia de la clase que la contiene,. Otras Fuentes para este tema: Tutorial de Sun: http://java.sun.com/docs/books/tutorial/java/more/nested.html Richard Baldwin http://www.geocities.com/Athens/7077/Java094.htm Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec6.html#obj21 151 Tutorial para el examen de certificación: SCJP 1.2 Sección 7. Threads Objetivo 1) Escribir código para definir, instanciar y comenzar nuevos threads usando java.lang.Thread y java.lang.Runnable. ¿Qué es un Thread ? Los Threads son procesos ligeros que aparentan correr en paralelo con tu programa principal. A diferencia de los procesos un thread comparte memoria y datos con el resto del programa. La Palabra Thread es una contracción de “Thread de ejecución” (Hilo de ejecución), puedes imaginar una cuerda, a la cual le has deshilachado el final y has tomado un hilo. El hilo todavía es parte de la misma cuerda, pero puede ser separado de la principal y ser manipulado por si solo. Un ejemplo de donde pueden ser útiles los threads, es en la impresión. Cuando das un click en el botón imprimir, probablemente no quieras que el programa principal se detenga hasta que finalice la impresión. Lo deseable, seria que el proceso de impresión comenzara a ejecutarse en segundo plano y te permitiera continuar usando la parte principal del programa. También sería útil si el programa principal pudiera responder si el hilo (thread) de impresión encontrara algún problema. Un ejemplo común utilizado para ilustrar los threads es el crear una aplicación GUI que lanzara una pelota rebotando cada vez que un botón es presionado. A diferencia de otros lenguajes el threading esta embebido en el corazón de Java, mucho de él al nivel de la ultima clase ancestro llamada Object. Las dos formas de crear un Hilo (Thread) De las dos maneras de crear un Hilo (algunos documentos ocupan la palabra hebra en lugar de hilo) el uso de Runnable es probablemente el mas común, pero debes de conocer las dos para propósitos del examen. El siguiente es un ejemplo de una clase creada con la interface Runnable. class MyClass implements Runnable { public void run() { //Blank Body } } Creando el hilo de ejecución. MyClass mc = new MyClass(); 152 javaHispano. Tu comunidad, tu lenguaje. Cualquier clase que implemente una interfase debe crear un método correspondiente por cada uno de los que se encuentren en la interfase. No es necesario que los métodos realicen algo, pueden tener el cuerpo del método en blanco, pero deben estar ahí. De esta manera incluí el método run incluso en este pequeño ejemplo, ya que si implementas Runnable debes incluir el método run No incluir el método run causaría un error en tiempo de compilación. Para hacer algo útil cuando se crea un hilo de ejecución de una clase deberías de, por supuesto, colocar algo de código donde puse : //Blank body La otra manera de crear un hilo es crear una clase que descienda de Thread. Esto es fácil de hacer pero significa que no puedes heredar de ninguna otra clase, debido a que Java solo soporta herencia simple. De esta manera, si estas creando un botón no le puedes agregar threading a de esta manera porque un botón hereda de la clase AWT Button y eso utiliza tu única opción de herencia. Existe debate sobre cual de las dos formas de crear un hilo es verdaderamente más orientada a objetos, pero no necesitas llegar a esa conclusión para propósitos del examen. Instanciando e Iniciando un hilo. Aunque el código que se ejecuta en tu thread esta en un método llamado run, no necesitas llamar a este método directamente, en vez de eso llamas al método start de la clase Thread. La interfase Runnable no contiene un método start, entonces, para llegar a este método y a los otros útiles métodos para threads (sleep, suspend, etc), pasas tu clase con la interfase Runnable como el constructor de una instancia de la clase Thread. De este modo para hacer que el hilo se ejecute desde una clase que implemente Runnable deberás llamar lo siguiente: MyClass mc = new MyClass(); Thread t = new Thread(mc); t.start(); De nuevo observa que fue una llamada al método start, no una llamada al método run, aunque es el código en el método run en tu clase el que actualmente se ejecuta. Aunque es el método run el que ejecuta, un hilo se inicia a través del método start. 153 Tutorial para el examen de certificación: SCJP 1.2 Si creas una clase que sea una subclase de Thread simplemente tienes que llamar al método start. El inconveniente de heredar de la clase Thread, es que debido al soporte de herencia simple no puedes heredar funcionalidad de ninguna otra clase. Preguntas Pregunta 1) ¿Qué ocurrirá cuando intentes compilar y ejecutar el siguiente código? public class Runt implements Runnable{ public static void main(String argv[]){ Runt r = new Runt(); Thread t = new Thread(r); t.start(); } } public void start(){ for(int i=0;i<100;i++) System.out.println(i); } 1)Compila y tiene como salida una cuenta de 1 a 99 2)Compila pero no tiene salida 3)Error en tiempo de compilación: La clase Runt es una clase abstracta. No puede ser instanciada 4)Error en tiempo de compilación, el método start no puede ser llamado directamente Pregunta 2) ¿Cuáles de las siguientes frases es cierta? 1)Heredar directamente de la clase Thread te da acceso a más funcionalidad de las capacidades de tareas de Java que implementar la interfase Runnable 2)Usar la interfase Runnable significa que no tienes que crear una instancia de la clase Thread y llamar al método run directamente. 154 javaHispano. Tu comunidad, tu lenguaje. 3)Utilizando ambas, La interfase Runnable y heredar de la clase Thread requiere el llamado al método start para comenzar la ejecución 4)La interfase Runnable requiere únicamente implementar un método, que es llamado run Pregunta 3) ¿Qué ocurrirá cuando intentes compilar y ejecutar el siguiente código? public class Runt extends Thread{ public static void main(String argv[]){ Runt r = new Runt(); r.run(); } } public void run(){ for(int i=0;i<100;i++) System.out.println(i); } 1)Compila y tiene como salida una cuenta de 1 a 99 2)Compila pero no tiene salida 3)Error en tiempo de compilación: La clase Runt es una clase abstracta. No puede ser instanciada 4)Error en tiempo de compilación, el método start no ha sido definido Pregunta 4) ¿Cuáles de las siguientes frases es cierta? 1)Para implementar el Threading en un programa debes importar la clase java.io.Thread 2)El código que se ejecuta cuando inicias un hilo es colocado en el método run 3)Los Threads pueden compartir datos entre unos y otros 4)Para comenzar la ejecución de un Thread, llamas al método start y no al método run. Respuestas Respuesta 1) 3) Error en tiempo de compilación: La clase Runt es una clase abstracta. No puede ser instanciada 155 Tutorial para el examen de certificación: SCJP 1.2 La clase implementa Runnable pero no define el método run. Respuesta 2) 3) Usar la interfase Runnable significa que no tienes que crear una instancia de la clase Thread y llamar al método run directamente. 4) La interfase Runnable requiere únicamente implementar un método, que es llamado run Respuesta 3) 1)Compila y tiene como salida una cuenta de 1 a 99 Sin embargo, observa que este código no inicia la ejecución del hilo y el método run no debe ser llamado de esta forma. Respuesta 4) 2)El código que se ejecuta cuando inicias un hilo es colocado en el método run 3)Los Threads pueden compartir datos entre unos y otros 4)Para comenzar la ejecución de un Thread, llamas al método start y no al método run. No necesitas importar ninguna clase ya que el Threading es una parte integral de Java. Otras Fuentes sobre este tema Este tema esta cubierto en el tutorial de sun en http://java.sun.com/docs/books/tutorial/essential/threads/customizing.html Richard Baldwin cubre este tema en http://www.Geocities.com/Athens/Acropolis/3797/Java058.htm#two ways to thread Jyothi Krishnan sobre este tema en http://www.geocities.com/SiliconValley/Network/3693/obj_sec7.html#obj22 La parte de Threads del Tutorial de Elliot Rusty Harold http://www.ibiblio.org/javafaq/course/week11/index.html 156 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2) Reconocer las condiciones que puedan impedir que un Thread se ejecute Comentario de este objetivo La expresión “impedir que un thread se ejecute” es ligeramente ambigua, ¿significa que un hilo ha sido pausado deliberadamente, o también incluye hilos que han muerto?. Se dice que un hilo a sido bloqueado cuando se le ha impedido ejecutarse. Razones por las que un thread puede ser bloqueado Un hilo puede ser bloqueado por que: Ha sido puesto a dormir por una cantidad de tiempo • Ha sido suspendido por una llamada al método suspend() y estará bloqueado hasta una llamada al método resume() • El hilo ha sido suspendido por una llamada al método wait(), y se volverá ejecutable con un mensaje notify o notifyAll. Para propósitos del examen sleep(), y wait/notify son probablemente las más importantes de las situaciones por las que un hilo pueda ser bloqueado. El método sleep es estático y detiene la ejecución de un hilo por un número fijo de milisegundos. Existe una versión que supuestamente debe pausar por un número fijo de nanosegundos, aunque encuentro difícil de creer que alguien pueda trabajar en una máquina o implementación de Java que trabaje a ese nivel de precisión. El siguiente es un ejemplo para poner a dormir un hilo, nota como el método sleep lanza InterruptedException. El hilo: public class TSleep extends Thread{ public static void main(String argv[]){ TSleep t = new TSleep(); t.start(); } public void run(){ try{ while(true){ this.sleep(1000); System.out.println("looping while"); } }catch(InterruptedException ie){} } } 157 Tutorial para el examen de certificación: SCJP 1.2 Con la salida de Java 2, los métodos de la clase Thread : stop, suspend y resume han quedado obsoletos (depracated, no se recomienda que se usen y al utilizarlos se envía una advertencia en tiempo de compilación). Las notas del JDK indican lo siguiente: //Quote Obsoleto. Este método ha sido censurado, ya que es inherentemente propenso a crear deadlocks. Si el hilo objetivo mantiene un bloqueo sobre el monitor protegiendo un recurso de sistema crítico cuando queda suspendido, ningún otro hilo puede acceder a este recurso hasta que el hilo objetivo es reiniciado. Si el Hilo que intenta resumir al hilo objetivo intenta realizar un bloqueo antes de llamar a resume(), ocurre un deadlock. Tales deadlocks típicamente se manifiestan como procesos “congelados”. Para mas información vea ¿Por que son Obsoletos los métodos: Thread.stop, Thread.suspend y Thread.resumen? //End Quote Una fuente generalmente confiable (Kathy Kozel) ha indicado que tienes que estar conciente de esto para propósitos del examen. Yo asumiré que actualmente no necesitas conocer como usarlos. El bloqueo del Thread vía el protocolo wait/notify es cubierto en el siguiente tema Pregunta 1) ¿Qué ocurrirá cuando intentes compilar y ejecutar el siguiente código? public class TGo implements Runnable{ public static void main(String argv[]){ TGo tg = new TGo(); Thread t = new Thread(tg); t.start(); } public void run(){ while(true){ Thread.currentThread().sleep(1000); System.out.println("looping while"); } } } 1)Compila y no tiene salida 158 javaHispano. Tu comunidad, tu lenguaje. 2)Compila y tiene como salida el texto “looping while” repetidamente 3)Compila y tiene como salida el texto “looping while” una sola vez 4)Error en tiempo de compilación Pregunta 2) ¿Cuál de las siguientes formas es una manera recomendada para que un Thread pueda ser bloqueado? 1)sleep() 2)wait/notify 3)suspend 4)pause Pregunta 3) ¿Cuáles de las siguientes frases son verdaderas? 1)El método sleep toma parámetros del Hilo y el número de segundos que debe dormir 2)El método sleep toma un solo parámetro que indica el número de segundos que debe dormir 3)El método sleep toma un solo parámetro que indica el número de milisegundos que debe dormir 4)El método sleep es un método estático de la clase Thread Respuestas Respuesta 1) 4 )Error en tiempo de compilación El método sleep lanza InterruptedException y de esta manera este código no compilara hasta que el ciclo while quede envuelto en un bloque try/catch Respuesta 2) 1)sleep() 2)wait/notify Para la plataforma Java 2 el método suspend ha sido censurado y de esta manera es válido pero no recomendado. Respuesta 3) 159 Tutorial para el examen de certificación: SCJP 1.2 3)El método sleep toma un solo parámetro que indica el número de milisegundos que debe dormir 4)El método sleep es un método estático de la clase Thread Otras Fuentes sobre este tema Este tema esta cubierto en el tutorial de sun en http://java.sun.com/docs/books/tutorial/essential/threads/waitAndNotify.html Richard Baldwin cubre este tema en http://www.geocities.com/Athens/Acropolis/3797/Java058.htm#the notify() and wait() methods Jyothi Krishnan sobre este tema en http://www.geocities.com/SiliconValley/Network/3693/obj_sec7.html#obj23 Comentario de métodos deprecados en http://java.sun.com/docs/books/tutorial/post1.0/preview/threads.html 160 javaHispano. Tu comunidad, tu lenguaje. Objetivo 3) Escribir código utilizando: synchronized, wait, notify, y notifyAll, para proteger el código contra los problemas de acceso concurrente y la comunicación entre los hilos. Definir la interacción de bloqueos entre hilos y entre hilos y objetos cuando ejecutan los métodos: synchronized, wait, notify, y notifyAll. ¿Por qué necesitarías el protocolo wait/notify? Una manera de pensar en una utilidad para el protocolo wait/notify es imaginar un dato, por ejemplo una variable entera, e imaginar que es un campo en una base de datos. Si no tienes algún mecanismo de bloqueo en la base de datos existe la posibilidad de corromper los datos. Así un usuario podría obtener los datos, realizar un cálculo y escribir los datos. Si en el entretanto alguien más ha recuperado los datos, ha realizado un cálculo y lo ha escrito, los segundos cálculos de los usuarios se perderán cuando la primera persona escriba en la base de datos. synchronized La palabra clave synchronized se utiliza para marcar una sentencia o bloque de código e indicar que únicamente un hilo a la vez puede ejecutar una instancia del código. La entrada al código esta protegida por un supervisor o monitor de bloqueos. Este proceso es implementado por un sistema de bloqueos. Se asigna un bloqueo al objeto y así se asegura que únicamente un hilo a la vez pueda acceder al código. Así cuando un hilo empieza a ejecutar un bloque sincronizado este adopta el bloqueo. Cualquier otro hilo no podrá ejecutar el código hasta que el primer hilo haya finalizado y liberado el bloqueo. Nota que el bloqueo esta basado en el objeto y no en el método. Para un método la palabra clave synchronized se coloca antes de la definición del método: synchronized void amethod() { /* method body */} Para un bloque de código la palabra va antes de la apertura y cerradura de llaves. synchronized (ObjectReference) { /* Block body */ } El valor entre paréntesis indica el objeto o clase que necesita obtener el supervisor. Es generalmente más común sincronizar un método que un bloque de código. 161 Tutorial para el examen de certificación: SCJP 1.2 Cuando un bloque de código sincronizado se ejecuta, el objeto es bloqueado y no puede ser llamado por cualquier otro código hasta que el bloqueo sea liberado: synchronized void first(); synchronized void second(); De esta manera el beneficio por la sincronización es mayor que colocando la palabra reservada synchronized antes de un bloque de código. wait/notify Además de tener un bloqueo que puede fijarse y luego liberarse, cada objeto tiene un sistema que le permite hacer una pausa o esperar (wait) aunque otro hilo tome el bloqueo. Esto permite a los hilos poder comunicarse la condición de disponible para poder ejecutarse Debido a la naturaleza de herencia simple en Java, cada objeto es hijo de una gran clase ancestro llamado Object desde la cual obtienen la capacidad de comunicación entre hilos. wait y notify deben colocarse dentro del código sincronizado para asegurar que el código actual posee al supervisor de threads. Una llamada a wait dentro del código sincronizado causa que el hilo active su bloqueo y se ponga a dormir. Esto normalmente ocurre para permitir a otro hilo obtener el bloqueo y continuar algún proceso. El método wait es incoherente sin el uso de notify o notifyAll los cuales permiten que el código que está esperando ser notificado pueda despertarse y continuar ejecutándose. Un ejemplo típico del uso del protocolo wait/notify es para permitir la comunicación entre los Hilos que aparentemente involucran un ciclo infinito, como el siguiente: //codigo producto while(true){ try{ wait(); }catch (InterruptedException e) {} } // alguna otra accion va aqui notifyAll(); Como la condición a evaluar es notoriamente verdadera esta permanecerá así, el código a primera vista muestra que el ciclo será infinito. El método wait sin embargo 162 javaHispano. Tu comunidad, tu lenguaje. efectivamente significa activa el bloqueo sobre el objeto y espera hasta que el método notify o notifyAll sea llamado para liberarlo. La planificación de hilos es dependiente de la implementación y no puede asegurarse una actuación similar en cada JVM. A diferencia del resto de Java, la planificación de hilos es dependiente de la plataforma. Existen dos maneras conocidas para la planificación de hilos: • • Preemtive (Preventivo o apropiativo) Time Slicing (Division o fragmentación de tiempos) En el primer sistema un programa puede "apropiarse" de otro para conseguir su porción de tiempo de CPU. En el segundo sistema cada hilo consigue un "pedazo" del tiempo de CPU y entonces se mueve al estado de listo. Esto asegura que un solo hilo no pueda obtener todo el tiempo del CPU. La desventaja es que no puedes estar seguro cuánto tiempo tardará un Hilo en ejecutarse. Aunque Java define prioridades para los hilos, de la mas baja 1 a la más alta que es 10, debes tomar en cuenta que algunas plataformas reconocerán estas prioridades con precisión pero otras no. El método notify despertará un hilo que espera ser reactivado por el supervisor de hilos para el objeto. No puedes estar seguro de qué hilo se levantara. Si tienes sólo un hilo en espera entonces no tendrás ningún problema. . Si tienes en espera múltiples hilos entonces se levantara el hilo que lleva más tiempo esperando. Sin embargo no puedes asegurarlo, ya que las prioridades de los hilos influirán en el resultado. Como resultado generalmente se te aconseja que uses notifyAll en lugar de notify para no hacer suposiciones con la planificación o las prioridades. Por supuesto esto no siempre es posible y tendrás que intentar probar tu código en tantas plataformas como te sea posible. Preguntas Pregunta 1) ¿Cuales de las siguientes palabras claves indican a un hilo que libere el objeto que esta bloqueando? 163 Tutorial para el examen de certificación: SCJP 1.2 1) release 2) wait 3) continue 4) notifyAll Pregunta 2) ¿Cuál de las siguientes frases define mejor la función de la palabra clave synchronized? 1)Permite que más de un Hilo acceda a un método simultáneamente. 2)Permite que más de un Hilo obtenga el bloqueo de un Objeto sobre una referencia. 3)Da acceso exclusivo al monitor a las palabras clave notify/notifyAll. 4)Significa que sólo un hilo en un momento puede acceder a un método o bloque de código. Pregunta 3) ¿Que pasará cuando intentes compilar y ejecutar el siguiente bloque de código? public class WaNot{ int i=0; public static void main(String argv[]){ WaNot w = new WaNot(); w.amethod(); } public void amethod(){ while(true){ try{ wait(); }catch (InterruptedException e) {} i++; }//End of while }//End of amethod }//End of class 1)Error en tiempo de compilación; no hay una llamada a notify() dentro del método 2)Se compila y ejecuta, pero se crea un ciclo infinito dentro del ciclo while 3)Se compila y ejecuta 4)Se genera un error en tiempo de Ejecución: “IllegalMonitorStatException” Pregunta 4) 164 javaHispano. Tu comunidad, tu lenguaje. Cómo puedes especificar qué hilo es el notificado con el protocolo wait/notify 1)Pasando la referencia del objeto como parámetro al método notify 2)Pasando el nombre del método como parámetro al método notify 3)Usando el método notifyAll y pasando la referencia del objeto como referencia 4)Ninguna de las anteriores Pregunta 5) ¿Cuáles de las siguientes afirmaciones son verdaderas? 1)Java usa un sistema de planificación basado en la división de tiempos para determinar qué Hilo se ejecutará 2)Java usa un sistema apropiativo cooperativo para determinar qué Hilo se ejecutará 3)La planificación de hilos en Java es dependiente de plataforma y puede variar de una implementación a otra 4)Tu puedes fijar la prioridad de un hilo en el código Respuestas Respuesta 1) 2) wait Respuesta 2) 4)Significa que sólo un hilo en un momento puede acceder a un método o bloque de código. Respuesta 3) 4)Se genera un error en tiempo de Ejecución: “IllegalMonitorStatException” El protocolo wait/notify se puede usar solamente dentro del código sincronizado. En este caso la llamada al código no tiene bloqueo sobre el objeto y de esta manera se generará un error en tiempo de compilación. Respuesta 4) 4)Ninguna de las anteriores 165 Tutorial para el examen de certificación: SCJP 1.2 El protocolo wait/notify no ofrece un método para especificar que hilo se notificará. Respuesta 5) 3)La planificación de hilos en Java es dependiente de plataforma y puede variar de una implementación a otra 4)Tu puedes fijar la prioridad de un hilo en el código Otras Fuentes para este tema: Este tema es cubierto en tutorial de Sun: http://java.sun.com/docs/books/tutorial/essential/threads/waitAndNotify.html Richard Baldwin Covers lo cubre en: http://www.geocities.com/Athens/Acropolis/3797/Java058.htm Jyothi Krishnan en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec7.html#obj24 Bruce Eckel Thinking in Java Capítulo 14 166 javaHispano. Tu comunidad, tu lenguaje. Sección 8. El paquete java.awt y su disposición Objetivo 1. Escribir código que use las clases administradoras de disposición (Layout Manager) y los contenedores (Container) de componentes del paquete java.awt para presentar interfaces graficas de usuario (GUI) con una apariencia especifica, redimensionar su comportamiento y distinguir las responsabilidades entre los administradores de disposición y los contenedores. Nota sobre este objetivo Aunque el enunciado anterior no lo menciona específicamente, este objetivo involucra un nuevo tema comparado con el examen para la versión1.1. Éste tema es lo referente al GridBaglayout. Tiene sentido cubrirlo ya que es un administrador de disposición muy útil, pero debido a la utilidad que proporciona puede ser un poco complejo su aprendizaje. Peter va der Linden en su libro "Just Java and Beyond" 3a Edición lo describe como algo excesivamente complicado y no lo recomienda. Core Java dice: "el uso de GridBagLayout puede ser increíblemente complejo". Aunque es complejo utilizarlo de manera manual, varias herramientas de desarrollo como VisualCafe, Visual Age, JBuilder etc etc. lo hacen más fácil de entender. Comparando la disposición de componentes de Java y Visual Basic Comparada con herramientas como Visual Basic o Delphi, Java usa una filosofía distinta para la disposición de componentes. La mayoría de las herramientas de diseño usan una filosofía basada en la posición XY de píxeles para colocar un componente. Así en Visual Basic puedes tomar una caja de texto de la paleta de componentes y colocarla en una posición especifica en la forma o área de diseño, y esa posición será fija. En contraste Java usa clases de disposición para controlar donde será colocado un componente de acuerdo a la pantalla actual. La filosofía de los administradores de disposición El administrador de disposición FlowLayout es un buen lugar para empezar ya que es el administrador de disposición por defecto para los applets. El administrador de disposición FlowLayout simplemente coloca los componentes sobre el fondo uno después de otro de izquierda a derecha. Si se acaba el espacio hacia la derecha entonces envuelve los componentes a la próxima línea El siguiente código crea una aplicación sencilla y agrega una serie de botones. 167 Tutorial para el examen de certificación: SCJP 1.2 import java.awt.*; public class FlowAp extends Frame{ public static void main(String argv[]){ FlowAp fa=new FlowAp(); //Change from BorderLayout default fa.setLayout(new FlowLayout()); fa.setSize(400,300); fa.setVisible(true); } FlowAp(){ add(new Button("One")); add(new Button("Two")); add(new Button("Three")); add(new Button("Four")); add(new Button("Five")); add(new Button("Six")); add(new Button("Seven")); add(new Button("Eight")); add(new Button("Nine")); add(new Button("Ten")); }//End of constructor }//End of Application La siguiente imagen tiene la apariencia por defecto que tendrá la aplicación cuando la ejecutes desde la línea de comandos: Apariencia por defecto de la aplicación FlowAp 168 javaHispano. Tu comunidad, tu lenguaje. Apariencia de la aplicación FlowAp después de redimensionar su tamaño. Ten presente que ambas imágenes visualizan exactamente el mismo código Java. Lo único que ha cambiado es la anchura del Frame. El administrador de disposición FlowLayout automáticamente cambia la disposición de los componentes cuando el Frame se redimensiona. Si haces el Frame más pequeño FlowLayout cambiaría la disposición para que los botones se envuelvan alrededor en varias filas. Cuando te encuentras por primera vez con estos administradores de componentes pueden parecerte un poco arbitrarios. Algunas herramientas de desarrollo de aplicaciones como Symantec Visual Café o Borland/Inprise Jbuilder ofrecen maneras para especificar la posición de los componentes. Para los propósitos del examen por lo menos debes familiarizarte con el manejo de los administradores de disposición para la creación de GUI. Administradores de disposición que debes conocer para el examen. Para el examen, debes conocer los siguientes administradores de disposición: • • • • FlowLayout BorderLayout GridLayout GridBagLayout Responsabilidades de los Administradores de Disposición VS los Contenedores 169 Tutorial para el examen de certificación: SCJP 1.2 Los contenedores y los administradores de disposición trabajan en equipo. Los administradores de disposición generalmente controlan donde se colocarán los componente. Un contenedor controlará la fuente predefinida para estos componentes. Un componente puede asignarse una fuente especifica para sí mismo. Te pueden dar un texto que describa la configuración de un contenedor de componentes y preguntarte que color de fondo y que fuente tendrá un botón o una etiqueta al visualizarse. Curiosidades del administrador BorderLayout Si agregas múltiples componentes a un contenedor que utiliza el administrador BorderLayout pero no pasas una de sus constantes como parámetro (North, South, etc), puedes conseguir resultados inesperados. Aquí una muestra que ilustra esto: import java.awt.*; public class FlowAp extends Frame{ public static void main(String argv[]){ FlowAp fa=new FlowAp(); // fa.setLayout(new FlowLayout()); fa.setSize(400,300); fa.setVisible(true); } FlowAp(){ add(new Button("One")); add(new Button("Two")); add(new Button("Three")); add(new Button("Four")); add(new Button("Five")); add(new Button("Six")); add(new Button("Seven")); add(new Button("Eight")); add(new Button("Nine")); add(new Button("Ten")); }//End of constructor }//End of Application Usando el administrador BorderLayout por defecto 170 javaHispano. Tu comunidad, tu lenguaje. La razón por la que obtienes este botón grande en el centro es que el administrador de disposición BorderLayout usa un conjunto de coordenadas al colocar los componentes. Divide el área de la superficie en: • • • • • North (Norte) South (Sur) East (Este) West (Oeste) Center (Centro) Podrías suponer que el valor por defecto para los componentes cuando los vas colocando es irse colocando en el sentido de las agujas del reloj alrededor de los puntos de su circunferencia o en algún otro orden similar. En cambio los diseñadores decidieron hacer a el centro del área del esquema el valor por defecto. Así en el ejemplo anterior cada botón se ha colocado encima del botón anterior y ha ocupado enteramente el área disponible. Como resultado la apariencia es que sólo tienes un botón, el último agregado. Debido a que el administrador de disposición BorderLayout sólo divide el área en las cinco coordenadas mencionadas anteriormente, no es el más utilizado de los administradores de disposición. Sin embargo necesitas conocerlo para el examen y necesitas ser consciente de que tiene como valor predefinido poner todos los componentes en el centro. El administrador de disposición GridLayout El administrador GridLayout hace aproximadamente lo que esperabas que hiciera. Divide el área de la superficie en una cuadrícula y cuando agregas componentes los pone uno después el otro de izquierda a derecha y de arriba hacia abajo. A diferencia de BorderLayout y de FlowLayout ignora cualquier tamaño predefinido del componente. Por ejemplo el tamaño predefinido de un botón será el ancho suficiente para mostrar su texto. El administrador FlowLayout intenta asegurarse que un botón mantenga su tamaño predefinido. El GridLayout de la única cosa de la que se preocupa es asegurarse que el componente se ajuste en la cuadrícula El código siguiente pone un conjunto de botones dentro de un Frame que utiliza un GridLayout que ha sido fijado para tener 2 filas y 5 columnas. import java.awt.*; public class GridAp extends Frame{ public static void main(String argv[]){ GridAp fa=new GridAp(); //definimos el GridLayout con 2 filas y 5 columnas fa.setLayout(new GridLayout(2,5)); fa.setSize(400,300); 171 Tutorial para el examen de certificación: SCJP 1.2 fa.setVisible(true); } GridAp(){ add(new Button("One")); add(new Button("Two")); add(new Button("Three")); add(new Button("Four")); add(new Button("Five")); add(new Button("Six")); add(new Button("Seven")); add(new Button("Eight")); add(new Button("Nine")); add(new Button("Ten")); }//fin del constructor }//fin de la Aplicacion Ejemplo con GridLayout Observa como los botones son alargados para ocupar todo el espacio disponible del Frame. GridBagLayout Peter van der Linden en su libro Just Java and Beyond describe al administrador de disposición GridBagLayout como “excesivamente complicado”. Core Java solo dice: “usar GidBagLayout puede ser increíblemente complicado”. Aunque es complejo manejarlo manualmente, varias herramientas de desarrollo como Visual Café, Visual Age, JBuilder, etc. Lo hacen más fácil de usar, sin tener que entenderlo. JBuilder 172 javaHispano. Tu comunidad, tu lenguaje. modifica fácilmente el método add para incluir los detalles siguientes en una instancia de la clase GridBagLayout. add(pAps,new GridBagConstraints2(1, GridBagConstraints.RELATIVE, GridBagConstraints.RELATIVE, 3, 0.0, 0.0,GridBagConstraints.CENTER, GridBagConstraints.NONE, new Insets(0, 0, 0, 0), -3, 45)); Pero cuando creas tu código manualmente no necesita parecer tan complejo como este. Comentarios de las personas que han realizado el examen indican que las preguntas sobre GridBagLayout no son muy profundas y una comprensión básica de los distintos campos de la clase GridBagConstraints puede ser adecuada para realizar el examen. GridBagLayout es un poco parecido a GridLayout sólo que las filas de la cuadrícula pueden tener alturas diferentes, y las columnas pueden tener anchuras distintas. La documentación de Java2 viene con un applet de demostración que muestra lo que puede hacerse con el administrador de disposición GridBagLayout.. Uno de los problemas con el GridBagLayout es que en lugar de estar basado estrictamente en la cuadrícula subyacente, Java intenta definir las celdas de acuerdo a la información dada. El administrador de disposición GridBagLayout usa una clase auxiliar llamada GridBagConstraints que tiene un conjunto de variables miembro que pueden ajustarse para afectar la apariencia de cada componente agregado. Los campos que modificas en la clase GridBagConstraints "sugieren" donde irán los componentes. Una instancia de GridBagConstraints se pasa como un parámetro al método add, en la forma: • add(component, GridBagConstraint); GridBagConstraints contradice la convención general en Java, ya que tu esperarías poder configurarlo con el método: setFooParam() Donde FooParam podría ser el valor X/Y o un relleno entre los componentes. En cambio toma la forma: GridBagLayout gbl=new GridBagLayout(); gbl.weightx=100; 173 Tutorial para el examen de certificación: SCJP 1.2 Si usas GridBagLayout sin usar la clase GridBagConstraints éste actúa un poco como FlowLayout, colocando simplemente uno por uno los componentes en el fondo. He creado un applet de demostración sencillo con su código fuente que muestra cómo no cambia la apariencia a menos que juegues con los valores de la clase GridBagConstraints http://www.software.u-net.com/applets/GridBagDemo/GridBagTest.htm El administrador GridBagLayout actúa un poco más como GridLayout si usas las clases GridBagConstraints y utilizas los campos gridx y gridy para asignar una posición en una cuadrícula "virtual" para cada componente que agregues. El applet demuestra esta posibilidad. Esto todavía es un poco torpe y muy parecido a los otros administradores de disposición. Las cosas empiezan a ponerse mucho más interesante cuando empiezas a modificar otros campos de la clase GridBagConstraints para modificar la apariencia de los distintos componentes dentro de esta cuadrícula "virtual". Recuerda que aunque necesitas entender esto para los propósitos del examen, podría ser más fácil que cuando estés programando en el mundo real una combinación de controles de contenedores con otros administradores de disposición. Un ejemplo de cuando esto no es una buena opción es cuando necesita redimensionar dinámicamente componentes. Ésta es una situación en donde herramientas de desarrollo como Visual Cafe o JBuilder no son de mucha ayuda y una buena comprensión del GridBagLayout puede ser esencial. He creado un applet de demostración: http://www.software.u-net.com/applets/GridBagDemo/GridBagDynamic.htm que muestran el efecto de cambiar los parámetros de relleno dinámicamente para un solo botón en un grupo de botones desplegados con GridbagLayout Los campos de la clase GridBagConstraints son: • • • • • • • gridx gridy gridwidth y gridheight fill ipadx e ipady insets anchor weightx y weighty Hojeando el excelente libro de Bill Brogdens Java2 Exam Cram, encontré una demostración comprensiva del GridBagLayout en: http://www.austria.eu.net/java/programs/applets/missgable/index.htm 174 javaHispano. Tu comunidad, tu lenguaje. Usando gridx y gridy para sugerir la posición de un componente Para este ejemplo supondremos que estas escribiendo el código necesario para diseñar el programa de una agenda. Mostrarás la hora de la cita abajo a la izquierda y los detalles de la cita abajo a la derecha. Las unidades de tiempo serán de media hora. Debido que una cita puede cubrir más de una unidad de tiempo; por ejemplo puede durar una hora y media, necesitaras poder cambiar la altura de los detalles de la cita dinámicamente para cubrir más de una unidad de tiempo. Debido a este requerimiento, de tener una altura variante para los detalles de las citas, utilizar GridLayout no es muy conveniente. Estarás poniendo paneles en el Frame como contenedores. El primer paso es asegurarse que cada panel este lado a lado en el Frame principal de la Aplicación. import java.awt.*; import java.awt.event.*; public class GBCal extends Frame{ Panel pTimes=new Panel(); Panel pAps=new Panel(); TextField txTimes=new TextField("09.00"); TextField txAps=new TextField("Meet the boss"); GridBagLayout gbl=new GridBagLayout(); GridBagConstraints gbc=new GridBagConstraints(); public static void main(String argv[]){ GBCal gbc=new GBCal(); gbc.setLayout(new FlowLayout()); } public GBCal() { pTimes.add(txTimes); pAps.add(txAps); setLayout(gbl); gbc.gridx=0; gbc.gridy=0; pTimes.setBackground(Color.pink); add(pTimes,gbc); gbc.gridx=1; gbc.gridy=0; pAps.setBackground(Color.lightGray); add(pAps,gbc); setSize(300,300); setVisible(true); } } 175 Tutorial para el examen de certificación: SCJP 1.2 Observa cómo las clases GridBagLayout y GridBagConstraints trabajan juntas. La instancia de GridBagConstraints gbc se reutiliza cada vez que se agrega un componente. En ningún punto especificas el número de filas y columnas para la cuadrícula, puesto que la clase GridBagLayout lo deduce de los campos gridx y gridy de la instancia de la clase GridBagConstraints. Este es el resultado: ipadx y ipady para controlar el relleno interior de los componentes El código ha puesto un color del fondo distinto para que puedas ver la magnitud de cada panel en lugar de ver simplemente la anchura de los campos de texto. Esto está bien, pero pensemos que ahora quieres que los campos de texto se estiren y abarquen de izquierda a derecha todo el Frame de la aplicación principal. Esto se puede realizar modificando el campo ipadx de la clase GridBagConstraints. Esto puede ejecutarse poniendo: gbc.ipadx=30; para las horas y gbc.ipadx=100; para los detalles El código queda: import java.awt.*; import java.awt.event.*; 176 javaHispano. Tu comunidad, tu lenguaje. public class GBCal extends Frame{ Panel pTimes=new Panel(); Panel pAps=new Panel(); TextField txTimes=new TextField("09.00"); TextField txAps=new TextField("Meet the boss"); GridBagLayout gbl=new GridBagLayout(); GridBagConstraints gbc=new GridBagConstraints(); public static void main(String argv[]){ GBCal gbc=new GBCal(); gbc.setLayout(new FlowLayout()); } public GBCal() { pTimes.add(txTimes); pAps.add(txAps); setLayout(gbl); gbc.gridx=0; gbc.gridy=0; gbc.ipadx=30; pTimes.setBackground(Color.pink); add(pTimes,gbc); gbc.gridx=1; gbc.gridy=0; gbc.ipadx=100; pAps.setBackground(Color.lightGray); add(pAps,gbc); setSize(300,300); setVisible(true); } } y este es el resultado: 177 Tutorial para el examen de certificación: SCJP 1.2 Componentes dentro de un panel utilizando GridBagLayout Para el próximo paso quiero proporcionar a cada panel su propio administrador de disposición GridBagLayout y agregar hendiduras para unidades de tiempo adicionales. Para propósitos de este ejemplo agregaré solo una hendidura de tiempo mas y simplemente estiraré el cuadro de texto del detalle de la cita para cubrir las hendiduras de las unidades de tiempo entre las 9.00 y 9.30. Para hacer esto crearé una nueva instancia de la clase GridBagLayout llamada gbBut y la usare para preparar la cuadrícula para el panel pTimes para colocar las hendiduras de las unidades de tiempo una encima de otra verticalmente. El código para realizar lo anterior es el siguiente: //Controlar el panel de los tiempos con GridBagLayout pTimes.setLayout(gbBut); gbc.gridx=0; gbc.gridy=0; pTimes.add(txTimes9,gbc); gbc.gridx=0; gbc.gridy=1; pTimes.add(txTimes930,gbc); Y este es el código completo: import java.awt.*; import java.awt.event.*; public class GBCal extends Frame{ Panel pTimes=new Panel(); Panel pAps=new Panel(); TextField txTimes9=new TextField("09.00"); TextField txTimes930=new TextField("09.30"); TextField txAps=new TextField("Meet the boss"); GridBagLayout gbl=new GridBagLayout(); GridBagLayout gbBut=new GridBagLayout(); GridBagConstraints gbc=new GridBagConstraints(); public static void main(String argv[]){ GBCal gbc=new GBCal(); gbc.setLayout(new FlowLayout()); } public GBCal() { setLayout(gbl); //Controlar el panel de tiempo con //GridBagLayout pTimes.setLayout(gbBut); 178 javaHispano. Tu comunidad, tu lenguaje. gbc.gridx=0; gbc.gridy=0; pTimes.add(txTimes9,gbc); gbc.gridx=0; gbc.gridy=1; pTimes.add(txTimes930,gbc); pTimes.setBackground(Color.pink); //Reutilizamos gbc para la disposición del panel principal gbc.gridx=0; gbc.gridy=0; gbc.ipadx=30; add(pTimes,gbc); pAps.setLayout(gbBut); gbc.gridx=0; gbc.gridy=1; pAps.add(txAps,gbc); gbc.gridx=1; gbc.gridy=0; gbc.ipadx=100; pAps.setBackground(Color.lightGray); add(pAps,gbc); setSize(300,300); setVisible(true); } }// fin de la clase El resultado es el siguiente: Esto esta trabajando bien hasta cierto punto. Tenemos ahora dos hendiduras para las unidades de tiempo, pero desafortunadamente el espacio para el detalle de la cita tiene 179 Tutorial para el examen de certificación: SCJP 1.2 como valor predefinido el centro del campo de las citas y obtenemos una fila gruesa. Lo que quiero ahora es fijarla hasta arriba y estirarla para que cubra las dos hendiduras de tiempo. Fijando componentes dentro de la cuadrícula Si un componente no-llena el área completa, puedes especificar en que parte del área lo quieres usando el campo anchor de la clase GridBagConstraints. Los posibles valores son: GridBagconstraints.CENTER GridBagconstraints.NORTH GridBagconstraints.NORTHEAST GridBagconstraints.EAST GridBagconstraints.SOUTHEAST GridBagconstraints.SOUTH GridBagconstraints.SOUTHWEST GridBagconstraints.WEST GridBagconstraints.NORTHWEST En este caso quiero posicionar los campos en la parte superior (North) de los paneles que los contienen. He aumentado el espesor del campo del detalle de la cita aumentando el valor de ipady para la dirección del campo. Aquí esta el código: import java.awt.*; import java.awt.event.*; public class GBCal extends Frame{ Panel pTimes=new Panel(); Panel pAps=new Panel(); TextField txTimes9=new TextField("09.00"); TextField txTimes930=new TextField("09.30"); TextField txAps=new TextField("Meet the boss"); GridBagLayout gbl=new GridBagLayout(); GridBagLayout gbBut=new GridBagLayout(); GridBagConstraints gbc=new GridBagConstraints(); public static void main(String argv[]){ GBCal gbc=new GBCal(); gbc.setLayout(new FlowLayout()); } public GBCal() { setLayout(gbl); //Controlamos el panel de tiempos con GridBagLayout pTimes.setLayout(gbBut); 180 javaHispano. Tu comunidad, tu lenguaje. //Aseguramos que los components se coloques //en la parte superior gbc.anchor=GridBagConstraints.NORTH; gbc.gridx=0; gbc.gridy=0; pTimes.add(txTimes9,gbc); gbc.gridx=0; gbc.gridy=1; pTimes.add(txTimes930,gbc); pTimes.setBackground(Color.pink); //Reutilizamos gbc para el panel principal gbc.gridx=0; gbc.gridy=0; gbc.ipadx=30; add(pTimes,gbc); pAps.setLayout(gbBut); gbc.gridx=0; gbc.gridy=1; gbc.ipady=12; pAps.add(txAps,gbc); gbc.gridx=1; gbc.gridy=0; gbc.ipadx=100; pAps.setBackground(Color.lightGray); add(pAps,gbc); setSize(300,300); setVisible(true); } }//fin de la clase Y este es el resultado: 181 Tutorial para el examen de certificación: SCJP 1.2 Elementos GridBag no cubiertos por este ejercicio Este ejercicio ha cubierto los siguientes campos de la clase GridBagConstraints : • • • ipadx/y gridx/y anchor La clase GridBagConstraints tiene los siguiente campos importantes: • • • • weightx/y fill gridwidth/height insets Los campos weightx/y controlan cómo un área crece o se contrae más allá de su tamaño inicial. Así que si pones el campo weighty a cero el campo seguirá manteniendo su altura constante aun cuando redimensiones la ventana. El campo fill controla cómo un componente se estira para llenar el área. Al igual que el campo anchor puedes definir los valores del campo fill usando constantes ya definidas de la clase GridBagConstraints. Éstas son: GridBagConstraints.NONE GridBagConstraints.HORIZONTAL GridBagConstraints.VERTICAL GridBagConstraints.BOTH Los campos del gridwidth/height determinan cuántas columnas y filas tendrá un componente. El campo insets indica el relleno "externo" que tendrá a lo largo de los límites de la celda de la cuadrícula. 182 javaHispano. Tu comunidad, tu lenguaje. Preguntas Pregunta 1) ¿Cuál será la apariencia de un applet con el código siguiente? import java.awt.*; public class FlowAp extends Frame{ public static void main(String argv[]){ FlowAp fa=new FlowAp(); fa.setSize(400,300); fa.setVisible(true); } FlowAp(){ add(new Button("One")); add(new Button("Two")); add(new Button("Three")); add(new Button("Four")); }//fin del constructor }//fin de la Application 1)Un Frame con botones etiquetados con "One", "Two", "Three","Four" colocados encada esquina 2)Un Frame con botones etiquetados con "One", "Two", "Three","Four" colocados de arriba hacia abajo 3)Un Frame con un botón grande etiquetado con "Four" en el Centro 4)Se generará un error al momento de la ejecución indicando que no has puesto un LayoutManager Pregunta 2) ¿Cómo indicas dónde se posicionará un componente utilizando Flowlayout? 1)Utilizando las constantes North, South, East, West 2)Asignando una referencia fila/columna 3)Pasando un porcentaje X/Y como parámetro al método add 4)No es necesario indicarlo, el administrador FlowLayout posicionará el componente 183 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 3) ¿Cómo le haces para cambiar al administrador de disposición actual a un contenedor? 1)Usando el método setLayout 2)Una vez creado no puedes cambiar el administrador de disposición de un componente 3)Usando el método setLayoutManager 4)Usando el método updateLayout Pregunta 4) ¿Qué pasa si agregas una Scrollbar en la parte Norte de un Frame? 1)El Frame se agrandará para permitir que el Scrollbar ocupe su tamaño predefinido. 2)El Scrollbar se colocará en la parte Norte del Frame, será muy ancho, muy grueso y no muy útil 3)No puedes agregar un Scrollbar en la parte Norte de un Frame, sólo al Este o al Oeste 4)El Scrollbar se estirará desde arriba hasta abajo del Frame Pregunta 5) ¿Qué pasa si agregas más botones de los que definiste en un GridLayout y las etiquetas de cada botón son demasiado largas? 1)El tamaño del contenedor se aumentara para permitir que las etiquetas de cada botón se desplieguen completamente 2)El GridLayout ignorará el largo de cada etiqueta y éstas se truncarán 3)Se generará un error al compilar indicando que los botones no pueden alcanza su tamaño predefinido 4)Se generará un error al ejecutarse indicando que los botones no puede alcanzar su tamaño predefinido Pregunta 6) ¿Cuales de las siguientes declaraciones son verdaderas? 184 javaHispano. Tu comunidad, tu lenguaje. 1)Puedes controlar la posición un componente llamando al método setLayout(new GridBagConstraints ()) 2)El administrador de disposición FlowLayout puede ser usado para controlar la posición de componentes utilizando GridBagLayout 3)El administrador de disposición GridBagLayout toma las constantes North, South, East, West y Center 4)Ninguna de estas respuestas es verdadera Pregunta 7) ¿Cuales de los siguiente son campos de la clase de GridBagConstraints? 1) ipadx 2) fill 3) insets 4) width Pregunta 8) ¿Cual será la apariencia mas probable del siguiente código después de ejecutarse? import java.awt.*; public class CompLay extends Frame{ public static void main(String argv[]){ CompLay cl = new CompLay(); } CompLay(){ Panel p = new Panel(); p.setBackground(Color.pink); p.add(new Button("One")); p.add(new Button("Two")); p.add(new Button("Three")); add("South",p); setLayout(new FlowLayout()); setSize(300,300); setVisible(true); } } 1)Los botones se colocarán de izquierda a derecha a lo largo de la parte inferior del Frame 2)Los botones se colocarán de izquierda a derecha a lo largo de la parte superior del Frame 3)Los botones no se desplegarán 185 Tutorial para el examen de certificación: SCJP 1.2 4)Solo se mostrarán los tres botones ocupando todo el Frame Pregunta 9) ¿Cuáles de estas declaraciones son ciertas respecto al campo anchor? 1)Es un campo del administrador de disposición GridBagLayout para controlar la colocación de componentes 2)Es un campo de la clase GridBagConstraints para controlar la colocación de componentes 3)Una constante valida para el campo anchor es GridBagConstraints.NORTH 4)El campo anchor controla la altura de los componentes agregados a un contenedor Pregunta 10) Al usar el administrador de disposición GridBagLayout, cada nuevo componente agregado requiere una nueva instancia de la clase GridBagConstraints. Es esto : 1)Verdadero 2)Falso Respuestas Respuesta 1 3)Un Frame con un botón grande etiquetado con "Four" en el Centro Si no especificas una constante, cualquier componente agregado a un contenedor que utilice el administrador de disposición BorderLayout se colocará en el centro. El administrador de disposición de un Frame por omisión es BorderLayout. Respuesta 2 4)No es necesario indicarlo, el administrador FlowLayout posicionará el componente Respuesta 3 1)Usando el método setLayout Respuesta 4 186 javaHispano. Tu comunidad, tu lenguaje. 2)El Scrollbar se colocará en la parte Norte del Frame, será muy ancho, muy grueso y no muy útil Respuesta 5 2)El GridLayout ignorará el largo de cada etiqueta y éstas se truncarán Respuesta 6 4)Ninguna de estas respuestas es verdadera Respuesta 7 1) ipadx 2) fill 3) insets Respuesta 8 2)Los botones se colocarán de izquierda a derecha a lo largo de la parte superior del Frame Cuando un administrador de disposición se cambia a FlowLayout el BorderLayout predefinido no se aplica y el panel se coloca en la parte superior del Frame. Respuesta 9 2)Es un campo de la clase GridBagConstraints para controlar la colocación de componentes 3)Una constante valida para el campo anchor es GridBagConstraints.NORTH Respuesta 10 2)Falso Otras Fuentes para este tema: Este tema es cubierto por el tutorial de sun en: http://java.sun.com/docs/books/tutorial/uiswing/layout/using.html Richard Baldwin cubre este tema en: BorderLayout http://www.geocities.com/Athens/7077/Java114.htm FlowLayout http://www.geocities.com/Athens/7077/Java116.htm GridLayout 187 Tutorial para el examen de certificación: SCJP 1.2 http://www.geocities.com/Athens/7077/Java118.htm Richard parece no cubrir el GridBagLayout Jan Newmarsh en Australia a creado esta pagina http://pandonia.canberra.edu.au/java/xadvisor/gridbag/gridbag.html Jyothi Krishnan en http://www.geocities.com/SiliconValley/Network/3693/obj_sec8.html#obj25 188 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2. Escribir código que implemente clases oyentes (Listener) y en métodos oyentes extraer información del evento para determinar el componente afectado ya sea para conocer la posición actual del ratón o el número de veces que se llama a un evento. Determinar el nombre de clase de un evento para cualquier interface oyente especificada en el paquete java.awt.event. El modelo oyente -Listener- de eventos Para escribir cualquier aplicación útil con interfaz gráfica de usuario (GUI) utilizando Java necesitas entender el funcionamiento las clases oyentes y la manera de extraer información de los eventos que ellas procesan. El sistema de manejo o gestión de eventos en Java cambió significativamente entre las versiones 1.0x y 1.1. En la versión 1.0x el código para manejar eventos se parecía bastante al código C llano para Windows, es decir bastante horrible. Requería la creación de grandes sentencias case donde colocabas el código para procesar un evento en particular de acuerdo con los parámetros pasados. Este sistema es bastante sencillo de entender para ejemplos triviales pero no es recomendable para programas más grandes. Tengo la impresión de que lo más relevante que necesita saber para el examen es que la gestión de eventos de la versión 1.1 no es compatible con versiones anteriores. En teoría, el código escrito con el estilo de código de la versión 1.0x para el manejo de eventos debe trabajar bien en versiones posteriores del JDK. El modelo de eventos del JDK 1.1. El sistema de gestión de eventos de Java 1.1 involucra el uso de clases oyentes -Listenerque son eficazmente "ligadas" a los componentes para procesar eventos específicosσ. Esto favorece a las herramientas de desarrollo de aplicaciones GUI para que generen automáticamente código que maneje los eventos. Si examinas el código generado por una de estas herramientas puede que te parezca confuso, esto es en parte debido a que tienden a involucrar clases internas creadas dentro de los métodos. Con el propósito de aprenderlo puedes manejar las clases de gestión de eventos como si fueran clases de alto nivel. Uno de los factores complicados para la gestión de eventos es que ésta se basa en Interfaces, pero es mucho más fácil de usar utilizando una serie de clases conocidas como las clases Adaptadoras -Adapter- , qué simplemente implementan las interfaces del evento. Cuando usas una interface necesitas implementar todos sus métodos, así el uso directo de la interface EventListener requiere la creación de métodos con cuerpos σ NT. El modelo de la versión 1.1 se conoce en algunos documentos como gestión de eventos delegados. 189 Tutorial para el examen de certificación: SCJP 1.2 vacíos para cualquier evento de la interface que no se utilice. Usando las clases Adaptadoras sólo necesitas crear los cuerpos de los métodos de los eventos que realmente utilizas. Las clases Adaptadoras -Adapter- te permiten usar las Interfaces oyentes sin tener que crear un cuerpo para cada uno de los método de la interface. Uno de los eventos esenciales más manejados para una aplicación sencilla es la habilidad de cerrar la aplicación como respuesta a seleccionar la opción cerrar proporcionada por el sistema operativoγ. Te puede sorprender que el manejo de este evento no venga ya implementado por omisión en un Frame. Si crea una aplicación que herede un Frame, pero no creas código para cerrarlo, tendrás qué o bien terminar el proceso desde el gestor de tareas o, en Windows, ir a la línea de comandos y pulsar CTRL-C. El equivalente Swing del componente Frame es el JFrame, el JFrame procesa su cierre como una acción predefinida, pero la certificación no cubre los componentes Swing. El funcionamiento del Frame de AWT es un buen lugar para comenzar a cubrir el tema de gestión de eventos. Los métodos para manejar los eventos de las ventanas (WindowEvent) no son tan intuitivos como los métodos de otros eventos. Así al principio no es obvio a cual de los métodos siguientes necesitas responder para cerrar el Frame: windowClosed o windowClosing De hecho es el método windowClosing el que necesita ser procesado. La manera más simple de destruir la ventana es usando la línea: System.exit(0); Aquí un ejemplo de una aplicación sencilla que muestra un Frame que responderá desapareciéndose cuando se accione la opción cerrar proporcionada por el sistema operativo: import import public public java.awt.event.*; // necesario para el manejo de eventos java.awt.*; class ShutHello extends Frame{ static void main(String argv[]){ ShutHello h = new ShutHello(); } ShutHello(){ γ NT: en Windows es la x que aparece en la esquina superior derecha o la pulsación de las teclas ALT-F4 190 javaHispano. Tu comunidad, tu lenguaje. Button b = new Button("ShutHello"); //creamos una instancia de la clase WindowCloser WindowCloser wc = new WindowCloser(); //Ligamos el oyente con el programa addWindowListener(wc); this.add(b); setSize(300,300); setVisible(true); } } class WindowCloser extends WindowAdapter{ //sobreescribimos uno de los metodos del Adaptador public void windowClosing(WindowEvent e){ System.exit(0); } } La segundo mitad del objetivo pide que conozcas el nombre de clase por cada evento de cada interface oyente. La tabla que aparece mas adelante lista todas las interfaces oyentes junto con sus métodos. Así la interface MouseListener ofrece métodos para los eventos: • • • • • clicked pressed released entered exited Si comparas estos con los eventos manejados en Visual Basic 5 la única área significante que no se cubre es un conjunto de métodos para manejar los eventos de arrastre con el ratón(Drag&Drop) El nombre de la clase del Evento pasado a cada método es bastante intuitivo y basado en el nombre de la clase Oyente. Así todos los métodos de ActionListener toman un parámetro de tipo ActionEvent, los métodos de ComponentListener toman un tipo de ComponentEvent, un ContainerListener toma un ComponentEvent, etc, etc, etc. Existen 11 interfaces oyentes en total, pero sólo 7 de ellas tienen múltiples métodos. Como lo relevante de los adaptadores es eliminar la necesidad de implementar métodos en blanco, sólo se implementan las clases Adaptadoras para estas 7 Interfaces Éstas son las siguientes: • ContainerAdapter 191 Tutorial para el examen de certificación: SCJP 1.2 • • • • • FocusAdapter KeyAdapter MouseAdapter MouseMotionAdapter WindowAdapter La siguiente tabla muestra la lista completa de interfaces para gestionar eventos. Interfaces para la gestión de eventos ActionListener actionPerformed(ActionEvent) AdjustmentListener adjustmentValueChanged(AdjustmentEvent) addAdjustmentListener() ComponentListener addComponentListener() componentHidden(ComponentEvent) componentMoved(ComponentEvent) componentResized(ComponentEvent) componentShown(ComponentEvent) ContainerListener componentAdded(ContainerEvent) componetRemoved(ContainerEvent) FocusListener focusGained(FocusEvent) focusLost(FocusEvent) ItemListener itemStateChanged(ItemEvent) KeyListener keyPressed(KeyEvent) keyReleased(KeyEvent) keyTyped(KeyEvent) MouseListener mouseClicked(MouseEvent) mouseEntered(MouseEvent) mouseExited(MouseEvent) mousePressed(MouseEvent) mouseReleased(MouseEvent) MouseMotionListener mouseDragged(MouseEvent) mouseMoved(MouseEvent) TextListener textValueChanged(TextEvent) WindowListener windowActivated(WindowEvent) windowClosed(WindowEvent) windowClosing(WindowEvent) windowDeactivated(WindowEvent) windowDeiconified(WindowEvent) windowIconified(WindowEvent) |windowOpened(WindowEvent) addActionListener() addContainerListener() addFocusListener() addItemListener() addKeyListener() addMouseListener() addMouseMotionListener() addTextListener() addWindowListener() 192 javaHispano. Tu comunidad, tu lenguaje. Preguntas Pregunta 1) ¿Cuáles de las siguientes afirmaciones son ciertas? 1)Para un componente dado sus eventos se procesará en el orden en que los oyentes fueron agregados. 2)Usar las clases Adaptadoras para el manejo de eventos significa crear cuerpos vacíos para todos los métodos que no se utilicen. 3)Un componente puede tener múltiples oyentes asociados a él 4)Pueden quitarse oyentes una vez agregados. Pregunta 2) ¿Cuáles de las siguientes líneas son métodos validos para el manejo de eventos? 1) 2) 3) 4) mousePressed(MouseEvent e){} MousePressed(MouseClick e){} functionKey(KeyPress k){} componentAdded(ContainerEvent e){} Pregunta 3) ¿Que pasara cuando intentes compilar y ejecutar el siguiente código? import java.awt.*; import java.awt.event.*; public class MClick extends Frame implements MouseListener{ public static void main(String argv[]){ MClick s = new MClick(); } MClick(){ this.addMouseListener(this); } public void mouseClicked(MouseEvent e){ System.out.println(e.getWhen()); } } 1)Se generará un error al compilar. 193 Tutorial para el examen de certificación: SCJP 1.2 2)Se generará un error al ejecutarse. 3)Se compilará y durante su ejecución mostrara la fecha y la hora por cada click 3)Se compilará y durante su ejecución la hora por cada click Pregunta 4) ¿Cuáles de las siguientes afirmaciones, respecto al manejo de eventos, son verdaderas? 1) El modelo de Eventos de la versión 1.1 es totalmente compatible con modelo de evento de la versión 1.0 2)El código escrito para el manejo de eventos en la versión 1.0x se ejecutará sin problemas sobre la versión 1.1 de la JVM. 3)El modelo de Eventos de la versión 1.1 está particularmente ajustado a las herramientas para la construcción de aplicaciones GUI. 4) El manejo del evento DragDrop fue agregado con la versión 1.1 del modelo de manejo de eventos. Respuestas Respuesta 1) 3)Un componente puede tener múltiples oyentes asociados a él 4)Pueden quitarse oyentes una vez agregados. Respuesta 2) 1)mousePressed(MouseEvent e){} 4)componentAdded(ContainerEvent e){} Respuesta 3) 1)Se generará un error al compilar. Debido a que este código utiliza una interface oyente de eventos, se deben crear cuerpos vacíos para cada uno de los métodos de la interface. Este código causará errores que advierten que MClick es una clase abstracta. Respuesta 4) 2)El código escrito para el manejo de eventos en la versión 1.0x se ejecutará sin problemas sobre la versión 1.1 de la JVM. 3)El modelo de Eventos de la versión 1.1 está particularmente ajustado a las herramientas para la construcción de aplicaciones GUI. 194 javaHispano. Tu comunidad, tu lenguaje. El código para el modelo de manejo de eventos de la versión 1.1 no funcionara con la versión 1.0x de la JVM. El método DragDrop no existe, yo lo invente. Otras Fuentes sobre este tema: Tutorial de Sun http://java.sun.com/docs/books/tutorial/uiswing/events/intro.html Richard Baldwin http://www.Geocities.com/Athens/7077/Java080.htm#design goals of the jdk 1.1 delegation Jyothi Krishnan http://www.geocities.com/SiliconValley/Network/3693/obj_sec8.html#obj25 David Reilly http://www.davidreilly.com/jcb/java107/java107.html 195 Tutorial para el examen de certificación: SCJP 1.2 Sección 9. El paquete java.lang.Math Objetivo 1. Escribir código que utilice los siguientes métodos de la clase java.lang.Math: abs, ceil, floor, max, min, random, round, sin, cos, tan y sqrt. Nota sobre este objetivo La clase java.lang.Math tiene el modificador final, por consiguiente sus métodos son estáticos. Esto significa que no puedes heredar la clase para crear o modificar las versiones de estos métodos. Esto es probablemente favorable ya que reduce la posibilidad de ambigüedad. Con certeza encontraras preguntas sobre estos métodos y sería una verdadera pena que no las contestes correctamente simplemente porque los pasaste por alto. abs Hasta que comencé ha estudiar para el examen de certificación no tenía ni idea de lo que hace el método abs. Elimina el signo de un número y retorna el numero simplemente. Así la siguiente línea de código mostrará simplemente 99. Si el número es positivo obtendrás como resultado el mismo número. System.out.println(Math.abs(-99)); ceil Este método retorna el próximo número entero hacia arriba. Si tienes lo siguiente: ceil(1.1) El resultado sea 2.0. Si cambias lo anterior por: ceil(-1.1) el resultado será -1 196 javaHispano. Tu comunidad, tu lenguaje. floor De acuerdo a la documentación del JDK este método devuelve: El valor double más grande que no sea mayor que el argumento pasado y que sea igual a un entero. Si tienes aun dudas de lo que significa lo anterior, aquí hay un pequeño programa y sus resultados: public class MyMat{ public static void main(String[] argv){ System.out.println(Math.floor(-99.1)); System.out.println(Math.floor(-99)); System.out.println(Math.floor(99)); System.out.println(Math.floor(-.01)); System.out.println(Math.floor(0.1)); } } los resultados son: -100.0 -99.0 99.0 -1.0 0.0 max y min Toma nota de lo siguiente: los siguientes dos métodos toman dos parámetros. Puedes encontrar preguntas con ejemplos incorrectos que les pasan sólo un parámetro. Como podrías esperar estos métodos son el equivalente de decir: "Cual es más grande: ESTE parámetro o ESTE parámetro" El siguiente código muestra como funcionan estos métodos: public class MaxMin{ public static void main(String argv[]){ System.out.println(Math.max(-1,-10)); System.out.println(Math.max(1,2)); System.out.println(Math.min(1,1)); System.out.println(Math.min(-1,-10)); System.out.println(Math.min(1,2)); } } 197 Tutorial para el examen de certificación: SCJP 1.2 Estos son los resultados: -1 2 1 -10 1 random Retorna un número aleatorio entre 0.0 y 1.0 (0.0 <=n<1) A diferencia de algunos sistemas que generan números aleatorios Java no parece ofrecer la funcionalidad de pasar un número como semilla para aumentar la aleatoriedad. Para propósitos del examen uno de los aspectos importantes a considerar de este método es que el valor de retorno está entre 0.0 y 1.0. Así los resultados de una típica secuencia de ejecuciones podrían ser: 0.9151633320773057 0.25135231957619386 0.10070205341831895 A menudo un programa necesita producir un número aleatorio entre 0 y 10 o 0 y 100. El código siguiente combina código matemático para producir un número aleatorio entre 0 y 100. System.out.println(Math.round(Math.random()*100)); round Redondea al entero más cercano. Para qué, si el valor tiende más de la mitad hacia el entero más alto, el valor es redondeado al entero siguiente. En caso contrario el valor retornado es el entero anterior. Si por ejemplo el valor a redondear es x entonces: 2.0 <=x < 2.5. entonces Math.round(x)==2.0 2.5 <=x < 3.0 entonces Math.round(x)==3.0 Aquí hay otros ejemplo con sus resultados: System.out.println(Math.round(1.01)); System.out.println(Math.round(-2.1)); System.out.println(Math.round(20)); 198 javaHispano. Tu comunidad, tu lenguaje. 1 -2 20 sin cos tan Estos métodos elegantes toman un parámetro de tipo double y hacen justo las mismas funciones elegantes que realizarían en cualquier otro lenguaje que hayas usado. En mi caso, que son 12 años de programación, nunca he usado una de estas funciones. Así quizás lo único que debemos recordar es que el parámetro que recibe es de tipo double. sqrt Retorna un valor tipo double que es la raíz cuadrada del valor pasado como parámetro. Resumen Los métodos max y min toman dos parámetros El método random retorna un valor entre 0 y 1 El método abs elimina el signo del valor pasado como parámetro El método round redondea un numero eliminando así la parte que esta después del punto decimal, pero conserva el signo del número. Pregunta 1) ¿Cuales de las siguientes líneas se compilarán correctamente? 1) 2) 3) 4) System.out.println(Math.max(x)); System.out.println(Math.random(10,3)); System.out.println(Math.round(20)); System.out.println(Math.sqrt(10)); 199 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 2) ¿Cuál de las siguientes líneas mostrara un valor aleatorio con valor entre 1 y10? 1) 2) 3) 4) System.out.println(Math.round(Math.random()* 10)); System.out.println(Math.round(Math.random() % 10)); System.out.println(Math.random() *10); Ninguna de las anteriores Pregunta 3) ¿Cuál será la salida de la siguiente línea? System.out.println(Math.floor(-2.1)); 1) -2 2) 2.0 3) -3 4) -3.0 Pregunta 4) ¿Cuál será la salida de la siguiente línea? System.out.println(Math.abs(-2.1)); 1) -2.0 2) -2.1 3) 2.1 4) 1.0 Pregunta 5) ¿Cuál será la salida de la siguiente línea? System.out.println(Math.ceil(-2.1)); 1) -2.0 2) -2.1 200 javaHispano. Tu comunidad, tu lenguaje. 3) 2.1 3) 1.0 Pregunta 6) ¿Qué pasará cando intentes compilar y ejecutar el siguiente código? class MyCalc extends Math{ public int random(){ double iTemp; iTemp=super(); return super.round(iTemp); } } public class MyRand{ public static void main(String argv[]){ MyCalc m = new MyCals(); System.out.println(m.random()); } } 1)Se generará un error al compilarse 2)Se generará un error al ejecutarse 3)Se ejecutará y mostrara un valor aleatorio entre 0 y 1 4) Se ejecutará y mostrara un valor aleatorio entre 1 y 10 Respuestas Respuesta 1) 3) System.out.println(Math.round(20)); 4) System.out.println(Math.sqrt(10)); La opción 1 es incorrecta porque el método max recibe dos parámetros no uno, y la opción 2 es incorrecta ya que el método random recibe solo un parámetro no dos Respuesta 2) 4) Ninguna de las anteriores La opción más cercana a la correcta es la 1 pero el detalle esta en recordar que el método random incluye el 0 y la pregunta pide valores entre 1 y 10. 201 Tutorial para el examen de certificación: SCJP 1.2 Respuesta 3) 4) -3.0 Respuesta 4) 3) 2.1 Respuesta 5) 1) -2.0 Respuesta 6) 1)Se generará un error al compilarse La clase Math tiene el modificador final y por eso no puede ser heredada. Otras Fuentes para este tema Jyothi Krishnan en http://www.geocities.com/SiliconValley/Network/3693/obj_sec9.html#obj28 202 javaHispano. Tu comunidad, tu lenguaje. Objetivo 2 Nota sobre este objetivo La teoría de la inmutabilidad de la clase String dice que una vez creado, un String nunca puede modificarse. En nuestra experiencia programando con Java en la vida real nos damos cuenta que esto no es del todo cierto. Veamos el siguiente código: public class ImString{ public static void main(String argv[]){ String s1 = new String("Hello"); String s2 = new String("There"); System.out.println(s1); s1=s2; System.out.println(s1); } } ¿Si un String no puede cambiarse entonces s1 debería mostrar siempre "Hello" , pero si pruebas el código te darás cuenta que en la segunda ejecución del println se muestra la cadena "There". Entonces ¿Que pasa? La inmutabilidad se refiere realmente a los puntos de referencia de los Strings. Cuando se asignan s2 a s1 en el ejemplo, el String que contiene "Hello" deja de ser referenciado y ahora s1 apunta a la misma referencia que s2. De hecho la cadena "Hola" realmente no se ha modificado es solo que teóricamente ya no puedes "llegar a ella". El objetivo te pide que reconozcas las implicaciones de la inmutabilidad de las cadenas, y la principal parece ser qué si quieres eliminar y cambiar el contenido de cadenas de texto debes utilizar la clase StringBuffer que viene construida con más métodos para esos propósitos. Debido a que la concatenación de Strings causa que un nuevo String sea instanciado "fuera de escena", puede haber un desempeño poco eficiente si estás manipulando grandes cantidades de Strings, como sucede al leer un archivo con gran cantidad de texto. Generalmente la inmutabilidad de la clase String no altera nuestra programación diaria, pero se cuestionará en el examen. Recuerda que de cualquier manera que te hagan la pregunta, una vez creado un String no puede cambiarse aún cuando la referencia a él se cambie para apuntar a algún otro String. Aunque ni los objetivos específicos del 203 Tutorial para el examen de certificación: SCJP 1.2 examen para Java2 ni para Java 1.1 lo mencionan, estoy bastante seguro que algunas preguntas requieren conocimiento sobre la clase StringBuffer. Preguntas Pregunta 1) Has creado dos Strings que contienen nombres. Así: String fname="John"; String lname="String" ¿Cómo puedes cambiar estos Strings para que tomen nuevos valores dentro del mismo bloque de código? 1) fname="Fred"; lname="Jones"; 2) String fname=new String("Fred"); String lname=new String("Jones"); 3) StringBuffer fname=new StringBuffer(fname); StringBuffer lname=new StringBuffer(lname); 4) Ninguna de las anteriores Pregunta 2) Estás creando un programa para leer en archivo del texto de 8 MB. Cada nueva línea leída se agrega a un objeto String pero te estás dando cuenta que, tristemente, le falta desempeño. ¿Cuál es probablemente la explicación? 204 javaHispano. Tu comunidad, tu lenguaje. 1)El sistema de Entrada/Salida(I/O) de Java se diseña alrededor de un común denominador de mas bajo nivel y es inherentemente lento 2)La clase String es inapropiada para operaciones de I/O, un arreglo de caracteres sería más conveniente. 3)Debido a que los Strings son inmutables un nuevo String se crea con cada línea leída, Cambiándolo por un StringBuffer puedes aumentar el desempeño. 4)Ninguna de las anteriores. Respuestas Respuesta 1) 4) Ninguna de las anteriores Una vez creado un String sólo se puede leer y no puede modificarse. Respuesta 2) 3)Debido a que los Strings son inmutables un nuevo String se crea con cada línea leída, Cambiándolo por un StringBuffer puedes aumentar el desempeño. Espero que ninguno de los programadores en C hayan sugerido un arreglo de caracteres. Otras Fuentes para este tema: Este tema es cubierto en el tutorial de Sun en: http://java.sun.com/docs/books/tutorial/essential/strings/stringsAndJavac.html (no entra mucho en detalle) Jyothi Krishnan cubre este tema en: http://www.geocities.com/SiliconValley/Network/3693/obj_sec9.html#obj29 205 Tutorial para el examen de certificación: SCJP 1.2 Sección 10. El paquete java.util Objetivo 1. Hacer una selección apropiada de las clases/interfaces de colecciones para satisfacer requisitos de comportamientos específicos Nota sobre este objetivo Aunque no lo menciona específicamente, este objetivo involucra uno de los nuevos objetivos para la versión del examen de Java 2, conocer las clases de colecciones. Lo que el examen cuestiona sobre estas nuevas clases es bastante básico, pide un conocimiento sobre donde y cómo podrías usarlas, en lugar de conocer detalladamente sus campos y métodos. Las colecciones antiguas La API Java 2 incluye nuevas interfaces y clases para reforzar las colecciones ya disponibles. Las primeras versiones de Java incluyeron: vector hashtable array BitSet De éstos, sólo el array era incluido en los objetivos para el examen de certificación de la versión 1.1. Una de las omisiones notables de Java 1.1 era el soporte para ordenar, un requisito muy común en cualquier situación de programación. Las nuevas colecciones La raíz de la API de colecciones es la interface Collection. Esta te proporciona una serie de métodos comunes que tendrán todas las clases de colecciones. Probablemente nunca crearas tu propia clase para implementar la interface Collection ya que Java te proporciona una serie de subclases y clases que usan esta interface. Java 2 incluye las siguiente nuevas clases de colecciones: Sets Maps Y una nueva clase llamada Set. 206 javaHispano. Tu comunidad, tu lenguaje. Las clases que implementan la interface Collection más que almacenar tipos de datos primitivos, almacenan objetos. Todo esto tiene inconvenientes ya que la creación de objetos reduce el rendimiento y a los elementos antes de usarse se les aplica un cast, al tipo apropiado, para que puedan ser utilizarlos. Esto también significa que las colecciones no verifican que los elementos sean todos el mismo tipo, ya que un objeto puede ser casi cualquier cosa. Set Set es una interface de colecciones que no puede contener elementos duplicados. De esta manera se presta para manejar conceptos como el de un conjunto de registros retornados de una base de datos relacional. Parte de la magia de la interface Set esta en el método add. add(Object o) Cualquier objeto pasado al método add puede implementar el método equals para que el valor pueda compararse con objetos existentes en la clase. Si la clase ya contiene éste objeto la llamada al método add no modifica el contenido y retorna false. La idea de retornar false cuando intentas agregar un elemento que ya existe se parece más a la manera en que trabaja C/C++ que la manera en que trabaja Java. La mayoría de los métodos similares en Java lanzan una Excepción en este tipo de situaciones. List List es una interface de colecciones ordenadas que puede contener elementos duplicados. Algunos de sus métodos más importantes son: add remove clear La documentación de JDK da como ejemplo usar List para manejar una GUI que controle una lista que contiene todos los nombres de los Planetas. Map Map es una interface, las clases que la implementen no pueden tener llaves duplicadas y es similar a Hashtables. ¿Por qué utilizar colecciones en lugar de arreglos? 207 Tutorial para el examen de certificación: SCJP 1.2 La gran ventaja de las colecciones sobre los arreglos es que las colecciones son desarrollables , es decir, no tienes que asignarles un tamaño al momento de crearlas. El inconveniente de las colecciones es que sólo guardan objetos y no tipos primitivos e inevitablemente afectan el rendimiento de la aplicación. Los arreglos no soportan directamente el ordenamiento, pero esto puede ser superado usando los métodos estáticos de las colecciones. Aquí hay un ejemplo: import java.util.*; public class Sort{ public static void main(String argv[]){ Sort s = new Sort(); } Sort(){ String s[] = new String[4]; s[0]="z"; s[1]="b"; s[2]="c"; s[3]="a"; Arrays.sort(s); for(int i=0;i< s.length;i++) System.out.println(s[i]); } } Las colecciones Set y Map aseguran la individualidad de sus elementos, List no asegura esta individualidad pero permite que sus elementos se ordenen. Usando Vectores El siguiente ejemplo muestra cómo puedes agregar objetos de diferentes clases a un Vector. Esto contrasta con los arreglos donde cada elemento debe ser del mismo tipo. El código avanza por cada objeto mostrando su contenido. Al mostrarse el contenido, implícitamente se accede al método toString() de cada objeto. import java.awt.*; import java.util.*; public class Vec{ public static void main(String argv[]){ Vec v = new Vec(); v.amethod(); }//fin de main public void amethod(){ 208 javaHispano. Tu comunidad, tu lenguaje. Vector mv = new Vector(); //nota como un Vector puede almacenar //objetos de distintos tipos mv.addElement("Hello"); mv.addElement(Color.red); mv.addElement(new Integer(99)); } //la siguiente linea causaría un error //ya que un vector no almacena tipos primitivos //mv.addElement(99) //Ahora avanzamos por cada elemento del vector for(int i=0; i< mv.size(); i++){ System.out.println(mv.elementAt(i)); } }//fin de amethod Antes de Java2 el uso de la clase del Vector era la manera principal de crear una estructura de datos re-dimensionable. Los elementos de esta clase pueden quitarse con el método remove . Usando Hashtables Las Hashtables son parecidas al concepto en Visual Basic de una colección que usa una llave. Actúa como un Vector, sólo que en lugar de referirte a los elementos por un número, te refieres a ellos por una llave. La palabra hash de su nombre se refiere a un término matemático que hace referencia al manejo de índices. Una Hashtable puede ofrecer un beneficio superior sobre un Vector, su rapidez BitSet Un BitSet como su nombre lo indica, almacena una secuencia de bits. No te confundas por la parte "set" de su nombre, no es un conjunto matemático o algo relacionado con bases de datos, tampoco se relaciona con los conjuntos(Sets) disponibles en Java2. Es más apropiado pensar en él como un vector de bits. Un BitSet puede ser útil para el almacenamiento eficaz de bits, en donde los bits se usen para representar los valores de true/false. La alternativa de usar alguna clase de colecciones que contenga valores tipo Boolean puede ser menos eficaz. Según Bruce Eckel en “Thinking in Java” Sólo es eficaz desde el punto de vista del tamaño; si estás buscando un acceso eficaz, es ligeramente más lento que el uso de un arreglo de algún tipo primitivo. 209 Tutorial para el examen de certificación: SCJP 1.2 Un BitSet es un poco una clase novedosa de la que nunca puedes llegar a necesitar. Sospecho que podría ser conveniente para propósitos criptográficos o para el procesamiento de imágenes. Por favor avísame si te encuentras con alguna pregunta relacionada con esta clase en el examen. Pregunta 1) ¿Cuáles de las siguientes clases son colecciones? 1) Collection 2) Iterator 3) HashSet 4) Vector Pregunta 2) ¿Cuáles de los siguientes enunciados son verdaderos respecto a la interface Collection? 1)La clase Vector a sido modificada para implementar la interface Collection 2)La interface Collection ofrece métodos individuales y métodos colectivos como addAll 3)La interface Collection es compatible con versiones anteriores y todos sus métodos están disponibles dentro de las clases del JDK 1.1. 4)Las clases de colecciones hacen innecesario el uso de arreglos. Pregunta 3) ¿Cuáles de las siguientes declaraciones son verdaderas? 1)La interface Set se diseño para asegurar que la implementación de clases tengan miembros únicos 2)Las clases que implementan la interface List no pueden contener elementos duplicados 210 javaHispano. Tu comunidad, tu lenguaje. 3) la interface Set se diseño para almacenar registros retornados de una consulta a una base de datos 4) la Interface Map no es parte del marco de trabajo de las colecciones Pregunta 4) ¿Cuáles de las siguientes declaraciones son verdaderas? 1) los elementos de clase de colecciones pueden ser ordenados usando el método sort de la interface Collection 2)Puedes crear una colección ordenada instanciando una clase que implemente la interface List 3)El método sort de la interface Collection toma como parámetros una A o una D para indicar el orden, Ascendente/Descendente 4)Los elementos de una clase de colecciones pueden ser ordenados usando el método order de la interface Collection Pregunta 5) Deseas almacenar una cantidad pequeña de datos y hacerlos disponibles para un acceso rápido. No necesitaras que los datos sean ordenados, la singularidad no es un problema y los datos permanecerán bastante estáticos ¿Qué estructura de datos podría ser la mas conveniente para este requisito? 1)TreeSet 2)HashMap 3)LinkedList 4)Un arreglo Pregunta 6) ¿Cuales de las siguientes son clases Collection? 1) ListBag 2) HashMap 3) Vector 4) SetList 211 Tutorial para el examen de certificación: SCJP 1.2 Pregunta 7) ¿Cómo puedes remover un elemento de un Vector? 1)Con el Método delete 2)Con el Método cancel 3)Con el Método clear 4)Con el Método remove Respuestas Respuesta 1) 3) HashSet 4) Vector Las otras 2 son interfaces. Respuesta 2) 1)La clase Vector a sido modificada para implementar la interface Collection 2)La interface Collection ofrece métodos individuales y métodos colectivos como addAll Las clases de colecciones son nuevas en Java 2. Con excepción de las clases que han sido actualizadas como Vector y BitSet, si ejecutas cualquiera de las Colecciones a través de un JDK antiguo, conseguirás un error en tiempo de compilación. Respuesta 3) 1)La interface Set se diseño para asegurar que la implementación de clases tengan miembros únicos Los elementos de una clase que implementa la interface List pueden contener elementos duplicados. Aunque una clase que implementa la interface Set podría usarse para guardar archivos retornados por una consulta a una base de datos, no se diseño particularmente para ese propósito. Respuesta 4) 2)Puedes crear una colección ordenada instanciando una clase que implemente la interface List Respuesta 5) 212 javaHispano. Tu comunidad, tu lenguaje. 4)Un arreglo Para semejante requisito sencillo probablemente un arreglo será la mejor solución. Respuesta 6) 2) HashMap 3) Vector A partir de Java 2 la clase Vector fue actualizado para ser parte del marco de trabajo de las colecciones. Respuesta 7) 4)Con el Método remove Otras Fuentes para este tema: El tutorial de SUN http://java.sun.com/docs/books/tutorial/collections/index.html Jyothi Krishnan cubre este tema en http://www.geocities.com/SiliconValley/Network/3693/obj_sec10.html#obj30 213 Tutorial para el examen de certificación: SCJP 1.2 Sección 11. El paquete java.io Objetivo 1 Escribir código que utilice objetos de la clase File para navegar dentro del sistema de archivos. En su excelente libro "Just Java and Beyond" Peter van der Linden comienza su capítulo sobre I/O de ficheros diciendo: "No es justo remarcar, como han hecho algunos, que el soporte para I/O en Java es un punto muy importante". Yo creo que él estaba indicando que es levemente importante, y así, es un área en la que hay que redoblar la comprobación de tu conocimiento antes de ir al examen. Cuando lo estas aprendiendo te queda al menos la compensación de que comprenderás un área útil del lenguaje. El sistema I/O de java esta basado en el concepto de flujos – streams – . El termino computacional stream fue popularizado primeramente por el sistema operativo UNÍX y puedes considerar tomarlo como analogía de un flujo de agua. Puedes tener un stream viniendo de una sola salida y de ahí aplicarle un filtro para procesarlo. Los nombres de las clases de flujos de I/O no son intuitivos y las cosas no siempre funcionan como las podrías imaginar. La clase File La clase File no es del todo descriptiva, puesto que una instancia de la clase File representa el nombre de un archivo o directorio más que el archivo en si mismo. Mi primera suposición al pedírseme navegar en un sistema de archivos sería buscar un método para cambiar de directorio. Desafortunadamente la clase File no tiene ése método y parece que simplemente tienes que crear una nueva instancia de File con un directorio diferente en su constructor. Así el examen podría hacerte preguntas sobre la habilidad de crear y eliminar archivos y directorios, qué podría ser considerado bajo la cabecera de "navegación por el sistema de ficheros". Crear una instancia de la clase File no crea un archivo en el sistema operativo subyacente. 214 javaHispano. Tu comunidad, tu lenguaje. La clase File ofrece el método: delete() Para borrar un archivo o directorio. mkdir() y mkdirs() Para crear directorios. La clase File contiene el método list(), que devuelve un arreglo de String´s conteniendo todos los archivos de un directorio. Esto es muy practico para comprobar si un archivo esta disponible antes de intentar abrirlo. Un ejemplo del uso de list(): import java.io.*; public class FileNav{ public static void main(String argv[]){ String[] filenames; File f = new File("."); filenames = f.list(); for(int i=0; i< filenames.length; i++) System.out.println(filenames[i]); } } Esto sencillamente muestra una lista de todos los archivos en el directorio actual ("*.*") Independencia de plataforma La clase File esta escrita en Java puro. Esto significa que no incluye código nativo, pero también significa que esta escrito para funcionar independientemente de la plataforma sobre la cual se ejecute. Debido a las diferencias en la manera en que trabajan los distintos sistemas de archivos es importante estar consciente de particularidades de la plataforma como el carácter separador de los directorios. En Win/DOS es la barra invertida “\” , en UNÍX es la barra normal “/” y sobre MAC es otro. Puedes saber cual es este carácter dependiente de la plataforma utilizando la constante File.separator en lugar de obtenerlo con alguna codificación compleja. Un programa para navegar por el sistema de archivos El siguiente archivo es bastante largo (más de 90 líneas) pero para entenderlo, necesitaras conocer la mayoría de los objetivos que se explican en esta sección. El programa te permite navegar entre los archivos de un directorio y cambiar de directorio. Esta inspirado en parte en código del libro "Java in a Nutshell" de O´reilly. Un libro que recomiendo ampliamente. Aquí una muestra del programa en funcionamiento. 215 Tutorial para el examen de certificación: SCJP 1.2 import java.awt.*; import java.awt.event.*; import java.io.*; public class Filer extends Frame implements ActionListener{ /************************************************************** Marcus Green October 2000 Part of the Java Programmer Certification tutorial available at http://www.jchq.net. Addressing the objective to be able to use the File class to navigate the File system.This program will show a list of files in a directory .Clicking on a directory will change to the directory and show the contentsNote the use of File.separator to allow this to work on Unix or PC (and maybe even the Mac) ****************************************************************/ List lstFiles; File currentDir; String[] safiles; int iEntryType = 6; int iRootElement = 0; int iElementCount = 20; public static void main(String argv[]){ Filer f = new Filer(); f.setSize(300,400); f.setVisible(true); } 216 javaHispano. Tu comunidad, tu lenguaje. Filer(){ setLayout(new FlowLayout()); lstFiles = new List(iElementCount); lstFiles.addActionListener(this); //set the current directory File dir = new File(System.getProperty("user.dir")); currentDir = dir; listDirectory(dir); add(lstFiles); addWindowListener( new WindowAdapter(){ public void windowClosing(WindowEvent e){ System.exit(0); } } ); } public void actionPerformed(ActionEvent e){ int i = lstFiles.getSelectedIndex(); if(i==iRootElement){ upDir(currentDir); }else{ String sCurFile = lstFiles.getItem(i); //Find the length of the file name and then //chop of the filetype part (dir or file) int iNameLen = sCurFile.length(); sCurFile = sCurFile.substring(iEntryType,iNameLen); File fCurFile = new File(currentDir.toString()+File.separator sCurFile); if(fCurFile.isDirectory()){ listDirectory(fCurFile); } } + } public void upDir(File currentDir){ File fullPath = new File(currentDir.getAbsolutePath()); String sparent = fullPath.getAbsoluteFile().getParent(); if(sparent == null) { //At the root so put in the dir separator to indicate this lstFiles.remove(iRootElement); lstFiles.add(" "+File.separator+" ",iRootElement); return; }else{ File fparent = new File(sparent); listDirectory(fparent); } } public void listDirectory(File dir){ String sCurPath = dir.getAbsolutePath()+File.separator ; //Get the directorie entries safiles = dir.list(); //remove the previous lis and add in the entry //for moving up a directory 217 Tutorial para el examen de certificación: SCJP 1.2 lstFiles.removeAll(); lstFiles.addItem("[ .. ]"); String sFileName = new String(); //loop through the file names and //add them to the list control for(int i=0; i< safiles.length; i++){ File curFile = new File(sCurPath + safiles[i]); if(curFile.isDirectory()){ sFileName = "[dir ]" + safiles[i]; }else{ sFileName = "[file]"+safiles[i]; } lstFiles.addItem(sFileName); } add(lstFiles); currentDir=dir; } } Preguntas Pregunta 1) ¿Cuál de los métodos siguientes sirve para distinguir entre un directorio y un archivo? 1) FileType() 2) isDir() 3) isDirectory() 4) getDirectory() Pregunta 2) ¿Cuál de los siguientes métodos de la clase File borrará un directorio o un archivo al ejecutarse? 1) La clase File no te permite borrar un archivo o un directorio 2) remove() 3) delete() 4) del() Pregunta 3) 218 javaHispano. Tu comunidad, tu lenguaje. Cómo puedes obtener los nombres de los archivos contenidos dentro de una instancia de la clase File con el contenido del directorio actual 1) dir.list() 2) dir.list 3) dir.files() 4) dir.FileNames() Pregunta 4) ¿Cuál de las siguientes líneas llenará una instancia de la clase File con el contenido del directorio actual? 1) File 2) File 3) File 4) File f f f f = = = = new new new new File(); File("*.*"); File('*.*'); File("."); Pregunta 5) Dada la siguiente línea de código: File f = new File("myfile.txt"); ¿Qué método creará el archivo “myfile.txt” en el sistema operativo subyacente? 1) f.write(); 2) f.close(); 3) f.flush(); 4) Ninguno de los anteriores Pregunta 6) ¿Cuál de las siguientes líneas cambiará el directorio actual al que se encuentra en un nivel superior? 1) chDir(".."); 2) cd("."); 3) up(); 4) Ninguno de los anteriores Pregunta 7) 219 Tutorial para el examen de certificación: SCJP 1.2 ¿Cuáles de los siguientes son métodos o campos de la clase File? 1) getParent() 2) separator 3) dirname 4) getName(); Respuestas Respuesta 1) 3) isDirectory() Respuesta 2) 3) delete() Respuesta 3) 1) dir.list() El método list retorna un arreglo de String´s que contendrá el contenido del directorio actual. Respuesta 4) 4) File f = new File("."); Este constructor para la clase File obtendrá el contenido del directorio actual sobre sistemas de archivos como DOS o UNIX, pero no estoy seguro que ocurrirá sobre otros sistemas con una estructura de archivos exótica como Mac OS. Respuesta 5) 4) Ninguno de los anteriores La clase File principalmente describe un archivo que debe de existir. Para crear un archivo en el sistema operativo subyacente necesitas pasar una instancia de la clase File a una instancia de una de las clases OutputStream. Respuesta 6) 4) Ninguno de los anteriores 220 javaHispano. Tu comunidad, tu lenguaje. Java no tiene una manera directa para cambiar el directorio actual. Una manera indirecta es crear una nueva instancia de la clase File apuntando al directorio deseado. Respuesta 7) 1) getParent() 2) separator 4) getName(); Otras Fuentes para este tema: Puedes darle un vistazo a los ejemplos del libro de I/O de O'Reilly: http://metalab.unc.edu/javafaq/books/javaio/index.html Este tema se cubre en el tutorial de SUN en http://java.sun.com/docs/books/tutorial/essential/io/ La API de la clase File http://java.sun.com/products/jdk/1.2/docs/api/java/io/File.html El JLS sobre IO, un poco académico http://www.infospheres.caltech.edu/resources/langspec-1.0/javaio.doc.html Richard Baldwin sobre I/O http://home.att.net/~baldwin.rg/Intermediate/Java060.htm Joyothi tiene varias útiles tablas sobre las clases de I/O en http://www.geocities.com/SiliconValley/Network/3693/io.html 221 Tutorial para el examen de certificación: SCJP 1.2 Objetivo 2 Escribir código que use objetos de las clases InputStreamReader y OutputStreamWriter para traducir entre códigos de caracteres UNICODE y el nativo de plataforma o el ISO 8859-1. Me sorprendió que este objetivo no se enfatizara en el examen de JDK 1.2 puesto que se había introducido la internacionalización y ésta es una de las grandes características de Java. Es bonito vender software a mil millones de europeos y americanos pero mil millones de chinos serían un agradable mercado adicional (incluso si sólo el 10% lo compran) Éste es el tipo de objetivo con el que ni siquiera los programadores de Java experimentados pueden tener poca experiencia, ¡Así qué toma nota!. Codificación de caracteres en Java: UTF y UNICODE Java utiliza dos sistemas muy parecidos de codificación UTF y UNICODE. Java fue diseñado desde abajo para tenérselas que ver con caracteres multibyte y puede tratar con inmensos números de caracteres que pueden guardarse usando el sistema de codificación UNICODE. Los caracteres UNICODE son almacenados en 2 bytes permitiendo almacenar arriba de 65K caracteres. Esto le permite ser usado con alfabetos no occidentales. Esto significa que puede manipular chino y japonés, y casi cualquier otro conjunto de caracteres conocido. Estarás agradecido de saber que no tienes que dar ejemplos de nada de esto en el examen. Aunque UNICODE puede representar casi cualquier carácter que llegues a usar en tu vida, no es un método eficiente de codificación para programar. La mayoría del texto dentro de un programa usa el código ASCII, la mayoría de este puede guardarse fácilmente dentro de un byte. Por razones de tamaño, Java utiliza un sistema de codificación llamado UTF8 para las literales de cadena, identificadores y otro texto dentro del programa. Esto puede dar como resultado un ahorro considerable de espacio comparándolo con el uso de UNICODE donde cada carácter requiere 2bytes. Las clases StreamReader La clase StreamReader convierte una entrada de bytes (no relacionada con un conjunto de caracteres) en un flujo de entrada de caracteres, uno que tenga el concepto de conjunto de caracteres. Si solo te las tienes que ver con conjuntos de caracteres de tipo ASCII, probablemente solo usarás clases Reader en la forma de: InputStreamReader(InputStream in) 222 javaHispano. Tu comunidad, tu lenguaje. Esta versión usa el código de caracteres nativo de la plataforma. En el JDK1.1 este valor por defecto se identifica por medio de la propiedad del sistema file.encoding. La codificación predeterminada generalmente es ISO-Latin 1 excepto en Mac donde es MacRoman. Si esta propiedad del sistema no esta definida, el identificador del código estándar es 8859_1 (ISO-LATIN-1). La suposición parece ser: si todo lo demás falla, debes volver al inglés. Experimentar con otros juegos de caracteres es problemático, debido que los caracteres no pueden mostrarse correctamente si el entorno no se configura apropiadamente. Así si intentas mostrar un carácter del conjunto de caracteres chinos tu sistema quizá no lo soportará Si te las tienes que ver con otros conjuntos de caracteres puedes usar InputStreamReader(InputStream in, String encoding); Las clases StreamReader y Writer pueden tomar cualquier codificación de caracteres como parámetro o puedes usar la codificación predeterminada. Recuerda que el InputStream viene primero y el código de caracteres después. Los métodos read y write La clase InputStreamReader tiene un método llamado read() y la clase OutputStreamWriter tiene un método llamado write() que permiten leer y escribir caracteres respectivamente. Cuando se llama al método read este lee bytes desde el stream de entrada y los convierte a caracteres UNICODE usando la codificación especificada en el constructor del stream. Cuando se llama al método write() los caracteres que vienen del stream son convertidos en su correspondiente byte codificado y es almacenado en un buffer interno. Cuando el buffer se llena, el contenido se escribe en el stream de bytes de salida. Ejemplo. GreekWriter El código para esta clase escribe un archivo de salida contiene algunas letras en alfabeto griego. Si intentas abrir el archivo Out.txt con un editor, veras algo parecido a basura. import java.io.*; class GreekWriter { public static void main(String[] args) { String str = "\u03B1\u03C1\u03B5\u03C4\u03B7"; try { Writer out = new OutputStreamWriter( 223 Tutorial para el examen de certificación: SCJP 1.2 } } new FileOutputStream("out.txt"), “8859_7"); // 8859_7 is the ISO code for ASCII plus greek, // although this // example also works on my machine if it is set to UTF8 out.write(str); out.close(); } catch (IOException e) { e.printStackTrace(); } Ejemplo. GreekReader import java.io.*; import java.awt.*; class GreekReader extends Frame{ /******************************************************* *Companion program to GreekWriter to illustrate *InputStreamReader and OutputStreamWriter as part *of the objectives for the Sun Certified Java Programmers *exam. Marcus Green 2000 *********************************************************/ String str; public static void main(String[] args) { GreekReader gr = new GreekReader(); gr.go(); gr.setWin(); } public void go(){ try { FileInputStream fis = new FileInputStream("out.txt"); InputStreamReader isr = new InputStreamReader(fis,"8859_7"); Reader in = new BufferedReader(isr); StringBuffer buf = new StringBuffer(); int ch; while ((ch = in.read()) > -1) { buf.append((char)ch); } in.close(); str = buf.toString(); } catch (IOException e) { e.printStackTrace(); } } public void paint(Graphics g) { //paint method automatically called by the system Insets insets = getInsets(); int x = insets.left, y = insets.top; //Add 30 to y or we will only see the 224 javaHispano. Tu comunidad, tu lenguaje. //downstrokes of the letters g.drawString(str, x, y +30); } public void setWin(){ //Nice big font so we can see the characters. Font font = new Font("Monospaced", Font.BOLD, 59); setFont(font); setSize(200,200); setVisible(true); //Show the frame show(); } } GreekReader en ejecución. Preguntas Pregunta 1) ¿Cuales de las siguientes sentencias son verdaderas? 1) La clase OutputStream debe tomar una codificación de caracteres como parámetro para su constructor. 2) La codificación predefinida para la clase OutputStreamWriter es ASCII 3) Las clases InputStreamWriter y OuputStreamWriter puede tomar como parámetro para su constructor una codificación de caracteres. 4) La clase InputStreanReader puede tomar un stream como uno de sus constructores. Pregunta 2) ¿Cuales de las siguiente sentencias son verdaderas? 225 Tutorial para el examen de certificación: SCJP 1.2 1) Java puede desplegar conjuntos de caracteres independientemente del sistema operativo subyacente. 2) La clase InputStreamReader puede tomar una instancia de otra clase InputStream como constructor. 3) Un InputStreamReader puede actuar como constructor de un OutputStreamReader para hacer conversiones entre juegos de caracteres. 4) Java usa el código ASCII para almacenar cadenas internamente. Pregunta 3) ¿Cuales de las siguientes son firmas validas para la clase InputStreamReder? 1) InputStreamReader(InputStream in, String encoding); 2) InputStreamReader(String encoding,InputStream in); 3) InputStreamReader(String encoding,File f); 4) InputStreamReader(InputStream in); Pregunta 4) ¿Cuales de los siguientes son métodos de la clase InputStreamReader? 1) read() 2) write() 3) getBuffer() 4) getString() Pregunta 5) ¿Cuales de las siguientes son sentencias son verdaderas? 1) Java utiliza UNICODE para almacenar internamente literales de cadena 2) Java utiliza ASCII para almacenar internamente literales de cadena 3) Java utiliza UTF-8 para almacenar internamente literales de cadena 4) Java utiliza la codificación nativa de la plataforma para almacenar internamente literales de cadena Respuestas Respuesta 1) 2) La codificación predefinida para la clase OutputStreamWriter es ASCII 4) La clase InputStreanReader puede tomar un stream como de sus constructores. Respuesta 2) 1) Java puede desplegar conjuntos de caracteres independientemente del sistema operativo subyacente. 2) La clase InputStreamReader puede tomar una instancia de otra clase InputStream como constructor. 226 javaHispano. Tu comunidad, tu lenguaje. Aunque Java puede guardar caracteres independientemente del sistema operativo subyacente, el conjunto de caracteres apropiado debe instalarse para poder desplegar esos caracteres. Generalmente los streams son unidos con otros streams, por ejemplo un InputStream toma como constructor a otro InputStream y un OutputStream toma como constructor a otro OutputStream. Java utiliza el sistema de codificación UTF para almacenar internamente cadenas. Respuesta 3) 1) InputStreamReader(InputStream in, String encoding); 4) InputStreamReader(InputStream in); Si no especificas una codificación la JVM asumirá como codificación la predeterminada de la plataforma. Respuesta 4) 1) read() Respuesta 5) 3) Java utiliza UTF-8 para almacenar internamente literales de cadena Otras Fuentes para este tema: La documentación de Sun sobre las clases InputStreamReader y OutputStreamWriter: http://java.sun.com/products/jdk/1.2/docs/api/java/io/OutputStreamWriter.html http://java.sun.com/products/jdk/1.2/docs/api/java/io/InputStreamReader.html JavaCaps: http://www.javacaps.com/sjpc_io_obj2.html Todo lo que puedes querer saber sobre UNICODE: http://www.unicode.org/ 227 Tutorial para el examen de certificación: SCJP 1.2 Objetivo 3 Distinguir entre las condiciones bajo las cuales se usaría conversión al código nativo de plataforma y las condiciones bajo las cuales se usaría una conversión específica. Este podría ser un objetivo de "bondad y disciplina". Con esto quiero decir que algunos puritanos podrían tomar la actitud de que siempre deberías especificar el código de conversión de caracteres porque nunca sabes donde, cuando y como se usará tu programa. Esto es porque muchos programadores asumieron que su código nunca tendría que vérselas con el problema del año 2000 por el que hubo tanto lío. Bien, es un buen pago para algunos programadores. Si tomas un punto de vista más benigno, este objetivo te pide identificar si tu código esta preparado para enfrentarse a un código de caracteres no estándar. Si tu código de caracteres por defecto no es ISO-LATIN-1 y consideras que el inglés es el lenguaje estándar para los negocios, o si necesitas manipular distintos códigos de caracteres, entonces aprovéchate de la posibilidad de hacer conversiones específicas. Si algo de esto no significa nada para ti, vuelve a leer la sección anterior sobre las clases de lectura y escritura de archivos. Otras Fuentes para este tema: La documentación de Sun sobre la internacionalización: http://java.sun.com/docs/books/tutorial/i18n/text/stream.html http://java.sun.com/products/jdk/1.1/docs/guide/intl/ http://java.sun.com/docs/books/tutorial/i18n/index.html 228 javaHispano. Tu comunidad, tu lenguaje. Objetivo 4 Seleccionar los argumentos validos para los constructores de una lista de clases contenidas en el paquete java.io. El énfasis en este objetivo es que conozcas los argumentos validos para los constructores de estas clases. Lo mas relevante que se puede mencionar es que la clase RandomFile no toma algún stream como argumento para alguno de sus constructores. Estas clases toman instancias de otros streams como argumentos. Así en el examen te pueden preguntar si alguna de ellas puede tomar una instancia de la clase File, un String con el nombre del archivo o la ruta del archivo, para ver si realmente conoces los argumentos válidos para sus constructores. Un constructor valido será algún tipo de stream más otros posibles parámetros. FileInputStream and OutputStream Las clases FileInputStream y FileOuputStream tomen como parámetro principal para sus constructores alguna referencia a un archivo. Esta puede ser un String que contenga el nombre del archivo, una instancia de la clase File o una instancia de la clase FileDescriptor. Estas clases a menudo se utilizan como el primer paso para ligar clases de streams. Típicamente una instancia de la clase FileInputStream recibe una referencia a un archivo-cualquiera de las que mencionamos anteriormente-, y se conecta con una instancia de la clase InputStreamReader para leer los caracteres que contiene dicho archivo. Este es un ejemplo de como se liga la clase FileInputStream con la clase InputStreamReader. Este programa mostrará su propio código fuente. import java.io.*; public class Fis{ public static void main(String argv[]){ try{ FileInputStream in = new FileInputStream("Fis.java"); InputStreamReader isr = new InputStreamReader(in); int ch=0; while((ch = in.read())> -1){ StringBuffer buf = new StringBuffer(); buf.append((char)ch); System.out.print(buf.toString()); } } catch (IOException e){System.out.println(e.getMessage());} } } Lo anterior es aconsejable cuando programas en el “mundo real” y utilizas la clase InputStreamReader, este tipo de situaciones te permite implementa fácilmente la 229 Tutorial para el examen de certificación: SCJP 1.2 internacionalización. Observa el ejemplo GreekReader.java mostrado anteriormente para ver un ejemplo de esto. BufferedInputStream y BufferedOutputStream Estas clases descienden directamente de las clases FilterInputStream y FilterOutputStream respectivamente. Permiten leer más información de manera inmediata colocándola en un buffer. Esto incrementa la eficiencia, ya que es mucho más rápido leer desde la memoria que desde el disco. Este tipo de lectura es particularmente útil si tienes que estar leyendo una gran cantidad de datos. Un ejemplo puede ser cuando estés procesando varias decenas de Megabytes de texto. Las clases BufferedInputStream y BufferedOutputStream toman una instancia de las clases InputStream y OutputStream respectivamente como parámetro para alguno de sus constructores, pero también pueden tomar un parámetro adicional que defina el tamaño del buffer. Así, puedes ajustar el tamaño del buffer que vas a utilizar. Este es un ejemplo de uso de las clases BufferedInputStream, observa como el código es similar con el ejemplo anterior únicamente se substituye la línea del InputStreamReader por BufferedInputStream. import java.io.*; public class BufIn{ public static void main(String argv[]){ try{ FileInputStream fin = new FileInputStream("BufIn.java"); BufferedInputStream bin = new BufferedInputStream(fin); int ch=0; while((ch=bin.read())> -1){ StringBuffer buf = new StringBuffer(); buf.append((char)ch); System.out.print(buf.toString()); } } } }catch(IOException e){System.out.println(e.getMessage());}; DataInputStream y DataOutputStream Las clases DataInputStream y DataOuputStream se usan para leer de una manera portable representaciones binarias de los tipos de datos primitivos de java. Esto te da acceso a varios métodos como readDouble o readInt que funcionarán de la misma manera sobre distintas plataformas. En el JDK 1.0 está era una de las principales maneras de acceder a texto UNICODE, pero desde la versión 1.1 del JDK ha sido substituido por las clases Reader . Esas clases toman una instancia de la clase Stream como parámetro de su constructor. 230 javaHispano. Tu comunidad, tu lenguaje. El siguiente ejemplo escribe un carácter en un archivo y después lo lee y muestra en la consola. //Write the file import java.io.*; public class Dos{ public static void main(String argv[]){ try{ FileOutputStream fos = new FileOutputStream("fos.dat"); DataOutputStream dos = new DataOutputStream(fos); dos.writeChar('J'); }catch(IOException e){System.out.println(e.getMessage());} } } //Read import public public the file java.io.*; class Dis{ static void main(String argv[]){ try{ FileInputStream fis= new FileInputStream("fos.dat"); DataInputStream dis = new DataInputStream(fis); System.out.println(dis.readChar()); }catch(IOException e){System.out.println(e.getMessage());} } } La clase File La clase File tiene tres versiones de constructores. Estos son: File(String path); File(String path, String name) File(File dir, String name); Los tres son muy similares y realizan la misma función. El primer constructor que toma un String recibe el nombre de un archivo, incluyendo su ruta, en forma de cadena. Esta ruta puede ser absoluta o relativa. El segundo toma por separado la ruta y el nombre del archivo y el tercer constructor es casi igual al primero excepto que el primer parámetro que indica el directorio del archivo es del tipo File en lugar del tipo String. RandomAccesFile Lo más importante que debes saber sobre los constructores de esta clase, es que toman 2 parámetros para sus constructores y que el segundo parámetro es un String que contiene el modo de apertura del archivo. Más adelante se explica con detalle. 231 Tutorial para el examen de certificación: SCJP 1.2 Preguntas Pregunta 1) ¿Cuáles de las siguientes clases con parámetros validos para los constructores de la clase FileInputStream? 1) File 2) String 3) FileDescriptor 4) RadomAccessFile Pregunta 2) ¿Cuáles de los siguientes constructores son validos para la clase BufferedInputStream? 1) BufferedInputStream(FileInputStream in, int size) 2) BufferedInputStream(FileInputStream in) 3) BufferedInputStream(FileOutputStream fos) 4) BufferedInputStream(RandomAccessFile ram) Pregunta 3) ¿Cuáles de los siguientes constructores son validos para la clase DataInputStream? 1) DataInputStream(FileInputStream in, int size) 2) DataInputStream(FileInputStream in) 3) DataInputStream(File f) 4) DataInputStream(String s) Pregunta 4) Dado el siguiente código: import java.io.*; public class Dos{ public static void main(String argv[]){ FileOutputStream fos = new FileOutputStream("fos.dat"); DataOutputStream dos = new DataOutputStream(fos); BufferedOutputStream bos = new BufferedOutputStream(dos); 232 javaHispano. Tu comunidad, tu lenguaje. } dos.write('8'); } ¿Cuál de las siguientes sentencias es verdadera? 1) El código no se compilará 2) No compilará debido a que la clase BufferedOutputStream no tiene un constructor que tome como parámetro una instancia de la clase DataOutputStream 3) Se compilara y escribirá el byte 8 en el archivo 4) Se compilara y escribirá la cadena “8” en el archivo Pregunta 5) ¿Cuáles de los siguientes son constructores validos? 1) File(String path); 2) File(String path, String name) 3) RandomAccessFile(File) 4) File(RandomAccesFile name) Pregunta 6) Dado el siguiente código: import java.io.*; public class Ppvg{ public static void main(String argv[]){ Ppvg p = new Ppvg(); p.go(); } public void go(){ try{ DataInputStream dis = new DataInputStream(System.in); dis.read(); }catch(Exception e){} System.out.println("Continuing"); } } ¿Cuál de las siguientes sentencias es verdadera? 1) El código se compilará y al ejecutarse se detendrá hasta que se pulse una tecla 2) El código no se compilará debido a que System.in es una clase estática 3) El código se compilará y se ejecutará sin generar ninguna salida 4) El código no se compilará debido a que System.in no es parámetro para un constructor valido para la clase DataInputStream Respuestas 233 Tutorial para el examen de certificación: SCJP 1.2 Respuesta 1) 1) File 2) String 3) FileDescriptor Respuesta 2) 1) BufferedInputStream(FileInputStream in, int size) 2) BufferedInputStream(FileInputStream in) Respuesta 3) 2) DataInputStream(FileInputStream in) Respuesta 4) 1) El código no se compilará El código no se compilará debido a que no se encuentra dentro de un bloque try/catch Respuesta 5) 1) File(String path); 2) File(String path, String name) Respuesta 6) 1) El código se compilará y al ejecutarse se detendrá hasta que se pulse una tecla Otras Fuentes para este tema La documentación de las API´s en SUN Buffered I/O http://java.sun.com/products/jdk/1.2/docs/api/java/io/BufferedInputStream.html http://java.sun.com/products/jdk/1.2/docs/api/java/io/BufferedOutputStream.html Data I/O http://java.sun.com/products/jdk/1.2/docs/api/java/io/DataInputStream.html http://java.sun.com/products/jdk/1.2/docs/api/java/io/DataOutputStream.html 234 javaHispano. Tu comunidad, tu lenguaje. Objetivo 5 Escribir el código adecuado para leer, escribir o actualizar ficheros usando objetos FileInputStream, FileOutputStream y RandomAccessFile. FileInputStream and FileOutputStream El siguiente ejemplo crea un fichero de texto llamado Out.txt y escribe el texto "Hello" en él. Si abres el fichero resultante verás que contiene el texto "H e l l o" (con un espacio en blanco entre cada letra). Sospecho que es porque un carácter en Java es de 16 bits y uno en ASCII son 8 bits, los 8 bits de más peso dan como resultado un carácter en blanco. Esto también muestra el objetivo previo de usar la clase DataOutputstream, descendiente de FilterOutputStream. Esto permite que sean escritos los caracteres legibles para el ser humano. Si este ejemplo solo usara métodos de FileOutputStream estarías limitado a escribir bytes o enteros. Si haces esto y muestras el resultado por la consola, solo verás caracteres extraños. import java.io.*; public class Fos{ String s = new String("Hello"); public static void main(String argv[]){ Fos f = new Fos(); f.amethod(); } public void amethod(){ try{ FileOutputStream fos = new FileOutputStream("Out.txt"); //DataOutputStream te permite escribir caracteres DataOutputStream dos = new DataOutputStream(fos); dos.writeChars(s); }catch(IOException ioe) {} } } El siguiente ejemplo leerá el fichero de texto producido en el ejemplo anterior y mostrará el texto en la consola. Date cuenta de que puesto que el programa anterior generó el fichero como UNICODE, si creas un fichero de texto con un editor de textos conteniendo una cadena de caracteres como "Hello", el siguiente código probablemente sólo mostrará caracteres extraños. import java.io.*; public class Fis{ 235 Tutorial para el examen de certificación: SCJP 1.2 public static void main(String argv[]){ Fis f = new Fis(); f.amethod(); } public void amethod(){ try{ FileInputStream fis = new FileInputStream("Out.txt"); DataInputStream dis = new DataInputStream(fis); while(true){ char c =dis.readChar(); System.out.println(c); } }catch(IOException ioe) {} } } RandomAccessFile La clase RandomAccessFile esta desarrollada libremente y no encaja fácilmente dentro de las clases de flujos de I/O. Si en el examen encuentras respuestas que mezclan instancias de RandomAccessFile con otros flujos de datos, casi seguramente son respuestas erróneas. El constructor de RandomAccessFile toma una instancia de la clase File o una cadena de caracteres, y un argumento indicando el modo de apertura. El argumento referente al modo de apertura puede ser "r" para solo lectura o "rw" para lectura y escritura. Memoriza estas dos opciones, que no te engañen en el examen con modos como "w", "ro" o "r+w"; A diferencia de la clase File, si intentas pasar el nombre de un archivo al constructor, RandomAccessFile intentará abrir el archivo. Si pasas únicamente el modo "r" como parámetro y el archivo no existe se lanzará una excepción. Y si pasas como parámetro el modo "rw" se intentará crear el archivo desde el sistema operativo. La clase RandomAccessFile no toma un stream como un parámetro para alguno de sus constructores.. Este ejemplo leerá el fichero Out.txt creado por el ejemplo anterior Fos.java. Por culpa del byte superior en blanco (byte, no bit), el resultado muestra un signo de interrogación con cada letra. 236 javaHispano. Tu comunidad, tu lenguaje. import java.io.*; public class Raf{ public static void main(String argv[]){ Raf r = new Raf(); r.amethod(); } } public void amethod(){ try{ RandomAccessFile raf = new RandomAccessFile("Out.txt","rw"); for(int i=0; i<10;i++){ raf.seek(i); char myc = raf.readChar(); //? mostrado para los bytes de mas peso, los superiores System.out.println(myc); } } catch(IOException ioe) {} } Pregunta 1) Asumiendo que se ha escrito código para el manejo de excepciones, ¿Cuáles de las siguientes sentencias creará una instancia de la clase RandomAccessFile? 1)RandomAccessFile 2)RandomAccessFile 3)RandomAccessFile 4)RandomAccessFile raf raf raf raf = = = = new new new new RandomAccessFile("myfile.txt","+rw"); RandomAccessFile( new DataInputStream()); RandomAccessFile("myfile.txt"); RandomAccessFile( new File("myfile.txt")); Pregunta 2) ¿Cuales de las siguientes sentencias son verdaderas? 1) La clase RandomAccessFile permite moverte hacia adelante y hacia atrás de un archivo sin tener que re-abrirlo. 2) Una instancia de la clase RandomAccessFile puede ser usada como parámetro para un constructor de la clase FileInputStream 237 Tutorial para el examen de certificación: SCJP 1.2 3) Los métodos de la clase RandomAccessFile no lanzan excepciones 4) Crear una instancia de la clase RandomAccessFile lanzará una excepción si el archivo no existe Pregunta 3) ¿Cuales de las siguientes sentencias son verdaderas? 1) La Clase FileInputStream puede tomar el nombre de un archivo o una instancia de la clase File como parámetros para alguno de sus constructores 2) La Clase FileInputStream lanzará una excepción si el nombre del archivo pasado como parámetro al constructor no existe. 3) Los métodos de la clase FileInputStream son apropiados especialmente para manipular archivos de texto. 4) El método delete de la clase FileInputStream elimina un archivo desde el sistema operativo. Pregunta 4) ¿Que pasará cuando intentes compilar y ejecutar el siguiente código? import java.io.*; public class Fos{ String s = new String("Hello"); public static void main(String argv[]){ Fos f = new Fos(); f.amethod(); } public void amethod(){ FileOutputStream fos = new FileOutputStream("Out.txt"); fos.write(10); } } 1) Error al compilar 2) Error al ejecutar 3) Se creará un archivo llamado Out.txt conteniendo el texto "10" 4) Se creará un archivo llamado Out.txt Pregunta 5) ¿Cuales de las siguientes sentencias son verdaderas? 1) El método seek de la clase FileInputStream define la posición actual del puntero del archivo. 2) El método read de la clase FileinputStream lee un número definido de bytes. 3) El método get de la clase FileInputStream lee un número definido de bytes. 238 javaHispano. Tu comunidad, tu lenguaje. 4) Una instancia de la clase FileOuputStream se puede cerrar utilizando en método close. Respuestas Respuesta 1) 1) RandomAccessFile raf = new RandomAccessFile("myfile.txt","+rw"); La clase RandomAccessFile es una anomalía en la arquitectura de I/O de Java. Desciende directamente de Object y no es parte de la arquitectura de flujos de datos. Respuesta 2) 1) La clase RandomAccessFile permite moverte hacia adelante y hacia atrás de un archivo sin tener que re-abrirlo. 4) Crear una instancia de la clase RandomAccessFile lanzará una excepción si el archivo no existe. Respuesta 3) 1) La Clase FileInputStream puede tomar el nombre de un archivo o una instancia de la clase File como parámetro para alguno de sus constructores 2) La Clase FileInputStream lanzará una excepción si el nombre del archivo pasado como parámetro al constructor no existe. Las clases FileInputStream y FileOutputStream están diseñadas para leer y escribir bytes más que texto. Respuesta 4) 1) Error al compilar Fos.java:10: Exception java.io.IOException must be caught, or it must be declare in the throws clause of this method. FileOutputStream fos = new FileOutputStream("Out.txt"); Respuesta 5) 2) El método read de la clase FileinputStream lee un número definido de bytes. 4) Una instancia de la clase FileOuputStream se puede cerrar utilizando en método close. 239 Tutorial para el examen de certificación: SCJP 1.2 Otras Fuentes para este tema Este tema esta cubierto por el tutorial de Sun en http://java.sun.com/docs/books/tutorial/essential/io/ El JLS sobre Java IO, un poco académico http://www.infospheres.caltech.edu/resources/langspec-1.0/javaio.doc.html Richard Baldwin sobre I/O http://home.att.net/~baldwin.rg/Intermediate/Java060.htm O’reilly ha publicado un libro específico sobre Java I/O. Probablemente entra en más detalle que el necesario para el examen de certificación pero navegar por los ejemplos online te puede ayudar. El libro tiene generalmente buenas críticas en www.amazon.com http://www.oreilly.com/catalog/javaio/ Joyothi tiene varias útiles tablas sobre las clases de I/O en http://www.geocities.com/SiliconValley/Network/3693/io.html 240