Você está na página 1de 59

A L G O R I T M I C A

Streams en Java

I I U N M S M

Generalidades
1 2

Un Stream es simplemente una fuente o destino de bytes. Los streams ms comunes son los archivos. Tambin pueden ser caracteres de un string o bloque de memoria o bytes de un socket. La idea es que el programador los trate de la misma forma que archivos. En Java se usan diferentes mecanismos para dar formato a datos. Ofrece clases orientada a manejo de bajo nivel y otras de alto nivel. Da gran flexibilidad y hace uso de filtros en cascada para manipular los datos de entrada o salida. En general la flexibilidad en ocasiones resulta ms

a pa ra to sa qu e lo qu e de se ar a m os .

A L G O R I T M I C A I I U N M S M

Jerarqua para flujos de

caracteres Clase para lectura de Caracteres

Clase abstracta

Clase para escritura de caracteres

Clase abstracta

A L G O R I T M I C A I I U N M S M

PrintWriter

Jerarqua para flujos de bytes


Lectura de bytes
Clase abstracta
A L G O R I T M I C A I I

Escritura de bytes

U N M S M

Clase abstracta

Clases bases para Salida


1

La clase OutputStream es una clase abstracta que especifica un nmero de operaciones para escribir un byte, para escribir un arreglo de bytes, y para cerrar el stream. Clases derivadas de sta son FileOutputStream y ByteArrayOutputStream, las cuales son clases concretas (no abstractas). Sus operaciones de write envan bytes a archivo o memoria. Para dar formato a los datos (imprimir nmeros y strings) se dispone de las clases Writers. PrintWriter es una clase de esa jerarqua, la cual recibe en su constructor el objeto con el cual podr escribir (write) los bytes. Ej. FileOutputStream fout = new

F il e O u t p u t S t r e a m ( o u t

put.dat); PrintWriter pout = new PrintWiter(fout);

A L G O R I T M I C A I I U N M S M

Ejemplo: Copia de un archivo


import java.io.*; public class Copy { public static void main(String[] args) throws IOException { File inputFile = new File("farrago.txt"); File outputFile = new File("outagain.txt"); FileReader in = new FileReader(inputFile); FileWriter out = new FileWriter(outputFile); int c; while ((c = in.read()) != -1) out.write(c); in.close() ;

ou e() } }

A L G O R I T M I C A I I U N M S M

Clases bases para Salida (Buffer)


1 2

Adems de formato podemos disponer de un buffer para mejorar el desempeo de algunos dispositivos de entrada y salida. El incorporar un buffer y con ello crear la cadena es un caso de filtrado, se usa: PrintWriter pout = new PrintWriter( new BufferedOutputStream (new FileOutpurStream(Output.dat)));

r e c i b e

PrintWrite r
1

BufferedOutputStrea m

FileOutputStrea m

Los filtros se pueden construir usando cualquier combinacin de clases encadenando una despus de la otra. La primera clase es aquella que permite leer o escribir objetos, y la ltima clase de la cadena enva

l o s b y t e s . L a s c l a s e

s intermedias pueden hacer el trabajo que necesitemos (buffer, encriptacin etc.)

A L G O R I T M I C A I I U N M S M

Clases bases para Salida


1 2

Para escribir texto, podemos usar PrintWriter y sus operaciones print o println. PrintWriter out; Employee harry; .... // apertura del PrintWriter entre otras out.println(3.14) ; out.print(Harry : ); out.print(harry); Cuando se imprimen objetos, se invoca el mtodo toString del objeto. La clase Object implementa este mtodo, el cual imprime la clase del objeto y su direccin. Lo

r e c o m e n d a b l e e s r e d e

finir este mtodo para producir un string que tenga ms sentido para la clase especfica.

A L G O R I T M I C A I I U N M S M

Clases bases para Entrada y Salida de bytes (no


necesariamente legible por humanos)
1

Para salida binaria, usamos las clases DataInputStream y DataOutputStream. Ellas proveen las siguientes operaciones: DataOutputStr DataInputStream eam readInt writeInt readShort writeShort readLong writeLong readFloat writeFloat readDouble writeDouble readBoolean writeBoolean readChar writeChar writeChars Ejemplo: DataOutputStream out = new

D a t a O u t p u t S t r e a m ( n e w F il e O

utputStream(output.dat)); out.writeDouble(3.14); out.writeChars(Harry);

A L G O R I T M I C A I I U N M S M

Clases bases para Entrada y Salida de bytes (no


necesariamente legible por humanos)
1

Una propiedad importante de las operaciones previas es que son independiente del procesador (tamao de datos). Se usa el ordenamiento de la red big-endian. La comunicacin es segura pues la clase se encarga de hacer las conversiones si el procesador es little-endian. Ej. Para guardar un string, primero se puede guardar el tamao y luego los caracteres. String s; .... out.writeInt(s.lengt h()); out.writeChars(s); Para leer el string de vuelta: int len = in.readInt();

S t r i n g B u f f e r b = n e w S t r i n

gBuffer(len); for (int i=0; i <len; i++) b.append(in.readChar()); String s = b.toString();


A L G O R I T M I C A I I U N M S M

10

Ejemplos
1

Lectura de una pgina WEB


1

URL url = new URL(www.elo.utfsm.cl); InputStream is = url.openStream(); DataInputStream in = new DataInputStream(is); String s; while((s=in.readLine()) != null) { ..... // procesar s ... } ZipInputStream zin =new ZipInputStream(new FileInputStream(employee.zip)); DataInputStream

din ata utS am );

Lectura desde archivo comprimido


1

A L G O R I T M I C A I I U N M S M

Archivos
1 2

Se dispone de las clase FileInputStream y FileOutputStream. Hay que recordar cerrar el archivo. FileInputStream fin = new FileInputStream (input.dat); ... fin.close(); No hay operaciones muy tiles para trabajar con los archivos en forma directa. Debemos utilizar un adaptador. DataInputStream in = new DataInputStream(fin);
Si deseamos hacer ambos lectura y salida de un archivo, se debe usar la clase RandomAccessFile. Esta clase no hereda de InputStream ni OutputStream.

s e e k

c o m o

e n

C .

5 6

Como no hay mltiple herencia en Java, esta clase implementa las interfaces DataInput y DataOutput. Para abrir un archivo random: RandomAccessFile in = new RandomAccessFile(input.dat, r); RandomAccessFile inOut = new RandomAccessFile(input.dat, rw); Los archivos random disponen de funciones para hacer un

12

A L G O R I T M I C A I I U N M S M

String como un Stream


1

Se emplean las clases ByteArrayOutputStream y ByteArrayInputStream. Para escribir sobre un string usando print().
Date bday = new Date(1975, 6,16); ByteArrayOutputStream bout = new ByteArrayOutputStream(); PrintWriter out = new PrintWriter(bout); out.print(Birthday: ); out.println(bday); String b = out.toString();

A L G O R I T M I C A I I U N M S M

Lectura de Entradas de Texto


1

Para escribir o leer dados binarios usamos DataInputStream o DataOutputStream. Para escritura formateada, PrintWriter.

usamos

Para lectura de texto en alto nivel no tenemos ninguna clase. Debemos hacerlo a mano. Por ejemplo: Para lectura de lneas podemos usar el mtodo readLine de la clase BufferedReader.
1

BufferedReader in = new BufferedReader(new

Fil Re ad er e m loy ee tx ); St ng lin e; wh le ((l ne = in. ea dL ne ))!

=null) { hacemos algo }


A L G O R I T M I C A I I U N M S M

14

Lectura de Entradas de Texto (cont)


1 2

3 4

Para leer nmeros, debemos cargar un string y luego convertirlo. Es posible que debamos extraer el dato desde el string para ello utilizamos la clase utilitaria StringTokenizer. Caso simple: String s=in.readLine(); double x=Double.parseDouble(s); .

A L G O R I T M I C A I I U N M S M

Persistencia en Java

A L G O R I T M I C A I I U N M S M

16

Persistencia en Java
1

Un objeto se dice persistente cuando es almacenado en un archivo u otro medio permanente. Un programa puede grabar objetos persistentes y luego recuperarlos en un tiempo posterior. A diferencia de C++ que slo soporta persistencia a travs de bibliotecas propietarias por lo cual su portabilidad y generalidad es limitada, Java se provee un mecanismo de serializacin para almacenar objetos en disco.
La serializacin se obtiene llamando al mtodo writeObject de la clase ObjectOutputStream para grabar el objeto, para recuperarlo llamamos al mtodo readObject de la clase ObjectInputStream. La serializacin adems de persistencia, se puede usar para transferir objetos desde una mquina a otra a

t r a v s d e u n s o c k e t .

A L G O R I T M I C A I I U N M S M

17

Interfaz Serializable
1

Slo objetos que implementen la interfaz Serializable pueden ser escritos a stream. La clase de cada objeto es codificada incluyendo el nombre de la clase y la firma de la clase (su prototipo) los valores de los sus campos y arreglos, y la clausura de cualquier otro objeto referenciado desde el objeto inicial. Para hacer que un objeto sea serializable, slo debemos declarar que implementa la interfaz serializable. Nada ms. No hay ntodos que debamos definir. Por razones de seguridad las clases no son serializable por defecto. Hay que tener claro el orden y tipo de los objetos almacenados en disco para recuperarlos en el mismo orden.

A L G O R I T M I C A I I U N M S M

Ejemplo: Empleados serializables


1

Class Employee implements Seralizable {....} Employee staff = new Employee[3]; .... out.writeObject(staff); Luego podemos recuperar el objeto haciendo: Employee[] newStaff=(Employee[])in.readObject(); Slo objetos pueden ser serializados con writeObject().

A L G O R I T M I C A I I U N M S M

1 9

Tratamiento de referencia a objetos


1

Mltiples referencias a un nico objeto son codificadas usando un mecanismo de referencias compartidas de modo que el grafo de objetos puede ser restaurado con la misma forma original. Los mtodos writeObject y readObject se encargan de crear y almacenar un nmero de serie para cada objeto. De este modo objetos ya almacenados no son grabados nuevamente. Supongamos que dada mnager tiene una secretaria. Dos mnager podran compartir la secretaria, en este caso tendriamos algo como:

staff

tres secretarias luego sino algo equivalente a esta vista en memoria. Name=

Euge nia
Si grabamos staff, no queremos

A L G O R I T M I C A I I U N M S

2 0

Tratamiento de referencia a objetos


Para ello Java utiliza el siguiente algoritmo para serializar (poner nmero de series). 2 A todos los objetos grabados a discos se les asigna un nmero de serie. 3 Antes de grabar un En disco: objeto a disco se ve si ya ha sido grabado. Serial number=1 4 Si ya ha sido grabado, type=Employee se graba lo mismo name=Eugenia que el objeto con nmero de series xxx Serial number=2 Sino, se almacena el type=Manager objeto. secretary=objeto 1
1

Serial number=3 type=Manager secretary=objeto 1

Ma sec

Employee name=Eugenia Manager secretary=

A L G O R I T M I C A I I U N M S M

21

Mezcla de objetos serializables y datos bsicos


1 2

Podemos hacerlo por medio de los mtodos writeInt, readInt, etc dado que ObjectOutputStream implementa la interfaz DataOutput. Anlogo para la entrada. De datos. Ejemplo: para escribir un objeto, 1 FileOutputStream ostream = new FileOutputStream("t.tmp"); 2 ObjectOutputStream p = new ObjectOutputStream(ostream); 3 p.writeInt(12345); 4 p.writeObject("Today"); 5 p.writeObject(new Date()); 6 p.flush(); 7 ostream.close(); La lectura se hace en forma anloga. 1 FileInputStream istream = new FileInputStream("t.tmp"); 2 ObjectInputStream p = new ObjectInputStream(istream); 3 int i = p.readInt(); 4 String today = (String)p.readObject(); 5 Date date = (Date)p.readObject(); 6 istream.close();

A L G O R I T M I C A I I U N M S M

Cuando hay objetos no serializables


1

P a r a q u e n o r e c l a m e e l

Clases que requieren manejos especiales durante el proceso de serializacin o deserializacin deben implementar los mtodos:
1

2 3

4 2

private void readObject(java.io.ObjectInputStream stream) throws IOException, ClassNotFoundException; private void writeObject(java.io.ObjectOutputStream stream) throws IOException

Se aplica en casos que tengamos objetos que no sean serializables (aquellos que tienen algn dato no serializable) Por ejemplo Point2D.Double no es serializable en Java.

compilador, definimos nuestro dato Point2D.Double como transiente (transient) y luego definimos los mtodos indicados.

A L G O R I T M I C A I I U N M S M

Cuando hay objetos no serializables


Ejemplo: public class LabelPoint {.... Private String label; private transient Point2D.Double point; } 1 Luego implementamos: private void writeObject(ObjectOutputStream out) throws IOException { out.defaultWriteObject(); out.writeDouble(point.ge tX()); out.writeDouble(point.ge tY()); } private void readObject(ObjectInutStream in) throws IOException { in.defaultReadObject() ; double

x= oub dou y= oub poi Poi ubl }

A L G O R I T M I C A I I U N M S M

24

Tpicos Usos de los Streams (lectura por lneas)


public static void main(String[] args) throws IOException { // 1a. Se lee un file lnea a lnea
BufferedReader in = new BufferedReader( new FileReader("IOStreamDemo.java"));

String s, s2 = new String(); while((s = in.readLine())!= null) s2 += s + "\n"; in.close(); // 1b. Se lee una lnea por teclado
BufferedReader stdin = new BufferedReader( new InputStreamReader(System.in));

S y st e m .o u t. p ri n t( " E n t e r a li n e: ")

; System.out.println(stdin.readLine( ));

2 5

Parsing de tipos bsicos


String linea; int a; BufferedReader stdin = new BufferedReader( new InputStreamReader(System. in)); System.out.print("Enter a line:"); linea = stdin.readLine();

a = I n t e g e r . p a

rseInt(linea); System.out.println(a); Tambin estn disponibles: parseDouble(), parseLong()


2 6

PDF to Wor

Você também pode gostar