Streams en Java
1
UN
MS
M
AL
GO
RIT
MIC
A II
Generalidades 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 aparatosa que lo que desearamos. 2
UN
MS
M
AL
GO
RIT
MIC
A II
Jerarqua para flujos de caracteres
Clase para lectura de Caracteres
Clase para escritura de caracteres
3
Clase abstracta
Clase abstracta
PrintWriter
UN
MS
M
AL
GO
RIT
MIC
A II
Jerarqua para flujos de bytes
Lectura de bytes
Escritura de bytes
4
Clase abstracta
Clase abstracta
UN
MS
M
AL
GO
RIT
MIC
A II
Clases bases para Salida
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 FileOutputStream(output.dat); PrintWriter pout = new PrintWiter(fout);
5
UN
MS
M
AL
GO
RIT
MIC
A II
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(); out.close(); } }
6
UN
MS
M
AL
GO
RIT
MIC
A II
Clases bases para Salida (Buffer)
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)));
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 o recibe los bytes. Las clases intermedias pueden hacer el trabajo que necesitemos (buffer, encriptacin etc.) 7
PrintWriter BufferedOutputStream FileOutputStream
UN
MS
M
AL
GO
RIT
MIC
A II
Clases bases para Salida
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 recomendable es re-definir este mtodo para producir un string que tenga ms sentido para la clase especfica. 8
UN
MS
M
AL
GO
RIT
MIC
A II
Clases bases para Entrada y Salida de bytes (no necesariamente legible por humanos)
Para salida binaria, usamos las clases DataInputStream y DataOutputStream. Ellas proveen las siguientes operaciones:
DataInputStream DataOutputStream readInt writeInt readShort writeShort readLong writeLong readFloat writeFloat readDouble writeDouble readBoolean writeBoolean readChar writeChar writeChars
Ejemplo: DataOutputStream out = new DataOutputStream(new FileOutputStream(output.dat)); out.writeDouble(3.14); out.writeChars(Harry);
9
UN
MS
M
AL
GO
RIT
MIC
A II
Clases bases para Entrada y Salida de bytes (no necesariamente legible por humanos)
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.length()); out.writeChars(s);
Para leer el string de vuelta: int len = in.readInt(); StringBuffer b = new StringBuffer(len); for (int i=0; i
Ejemplos Lectura de una pgina WEB
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 ... }
Lectura desde archivo comprimido
ZipInputStream zin =new ZipInputStream(new FileInputStream(employee.zip)); DataInputStream din=DataInputStream(zin);
11
UN
MS
M
AL
GO
RIT
MIC
A II
Archivos 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.
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 seek como en C.
12
UN
MS
M
AL
GO
RIT
MIC
A II
String como un Stream
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();
13
UN
MS
M
AL
GO
RIT
MIC
A II
Lectura de Entradas de Texto
Para escribir o leer dados binarios usamos DataInputStream o DataOutputStream.
Para escritura formateada, usamos PrintWriter.
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. BufferedReader in = new BufferedReader(new
FileReader(employee.txt); String line; while ((line = in.readLine())!=null) { hacemos algo }
14
UN
MS
M
AL
GO
RIT
MIC
A II
Lectura de Entradas de Texto (cont) 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);
.
15
UN
MS
M
AL
GO
RIT
MIC
A II
Persistencia en Java
16
UN
MS
M
AL
GO
RIT
MIC
A I
I
Persistencia en Java
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 travs de un socket .
17
UN
MS
M
AL
GO
RIT
MIC
A I
I
Interfaz Serializable
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.
18
UN
MS
M
AL
GO
RIT
MIC
A I
I
Ejemplo: Empleados serializables
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().
19
UN
MS
M
AL
GO
RIT
MIC
A I
I
Tratamiento de referencia a objetos 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:
Si grabamos staff, no queremos tres secretarias luego sino algo equivalente a esta vista en memoria.
20
staff Employee Name=Eugenia
Manager
Secretary=
Manager
Secretary=
UN
MS
M
AL
GO
RIT
MIC
A I
I
Tratamiento de referencia a objetos Para ello Java utiliza el siguiente algoritmo para serializar (poner nmero de
series).
A todos los objetos grabados a discos se les asigna un nmero de serie.
Antes de grabar un objeto a disco se ve si ya ha sido grabado.
Si ya ha sido grabado, se graba lo mismo que el objeto con nmero de series xxx
Sino, se almacena el objeto.
21
Employee
name=Eugenia
Manager
secretary=
Manager
secretary=
En disco:
Serial number=1
type=Employee
name=Eugenia
Serial number=2
type=Manager
secretary=objeto 1
Serial number=3
type=Manager
secretary=objeto 1
UN
MS
M
AL
GO
RIT
MIC
A I
I
Mezcla de objetos serializables y datos bsicos
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,
FileOutputStream ostream = new FileOutputStream("t.tmp");
ObjectOutputStream p = new ObjectOutputStream(ostream);
p.writeInt(12345);
p.writeObject("Today");
p.writeObject(new Date());
p.flush();
ostream.close();
La lectura se hace en forma anloga.
FileInputStream istream = new FileInputStream("t.tmp");
ObjectInputStream p = new ObjectInputStream(istream);
int i = p.readInt();
String today = (String)p.readObject();
Date date = (Date)p.readObject();
istream.close(); 22
UN
MS
M
AL
GO
RIT
MIC
A I
I
Cuando hay objetos no serializables
Clases que requieren manejos especiales durante el proceso de serializacin o deserializacin deben implementar los mtodos:
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.
Para que no reclame el compilador, definimos nuestro dato Point2D.Double como transiente (transient) y luego definimos los mtodos indicados.
23
UN
MS
M
AL
GO
RIT
MIC
A I
I
Cuando hay objetos no serializables Ejemplo:
public class LabelPoint
{....
Private String label;
private transient Point2D.Double point;
}
Luego implementamos: private void writeObject(ObjectOutputStream out) throws IOException { out.defaultWriteObject(); out.writeDouble(point.getX()); out.writeDouble(point.getY()); } private void readObject(ObjectInutStream in) throws IOException { in.defaultReadObject(); double x=in.readDouble(); double y=in.writeDouble(); point =new Point2D.Double(x,y); }
24
UN
MS
M
AL
GO
RIT
MIC
A I
I
25
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));
System.out.print("Enter a line:"); System.out.println(stdin.readLine());
}
26
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 = Integer.parseInt(linea);
System.out.println(a);
Tambin estn disponibles: parseDouble(), parseLong()
27
Tpicos Usos de los Streams (escritura por lneas) // throws IOException
String []s = {hola, que, tal};
// Se inicializa s
PrintWriter out1 = new PrintWriter( new BufferedWriter( new FileWriter("IODemo.out")));
int lineCount = 1;
for (int i=0; i
28
Tpicos Usos de los Streams (escritura de tipos bsicos) // throws IOException
DataOutputStream out2 = new
DataOutputStream(new
BufferedOutputStream(new FileOutputStream("Data.txt")));
out2.writeDouble(3.14159);
out2.writeBytes("That was pi\n); out2.writeChars( "That was pi\n); out2.writeDouble(1.41413);
out2.writeUTF("Square root of 2");
out2.close();
29
Tpicos Usos de los Streams (lectura de tipos bsicos) // throws IOException
DataInputStream in5 = new DataInputStream( new
BufferedInputStream( new FileInputStream("Data.txt")));
System.out.println(in5.readDouble());
System.out.println(in5.readLine()); // deprecated
System.out.println(in5.readDouble());
System.out.println(in5.readUTF());
30
Tpicos Usos de los Streams (files de acceso aleatorio)
// throws IOException
// acceso de lectura/escritura
RandomAccessFile rf = new RandomAccessFile("rtest.dat", "rw");
for(int i = 0; i < 10; i++)
rf.writeDouble(i*1.00);
rf.close();
rf = new RandomAccessFile("rtest.dat", "rw");
rf.seek(5*8); // salta 5 doubles (8 bytes cada uno)
rf.writeDouble(47.00); // modifica el sexto double
rf.close();
// acceso de slo lectura
rf = new RandomAccessFile("rtest.dat", "r");
for(int i = 0; i < 10; i++)
System.out.println( "Value " + i + ": " + rf.readDouble()); rf.close();
31
Tpicos Usos de los Streams (Object Stream) Java permite guardar objetos en ficheros, pero esos objetos
deben implementar la interfaz Serializable: public class MySerializableClass implements Serializable { ... }
Ejemplo: // throws IOException FileOutputStream out = new FileOutputStream("theTime.dat"); ObjectOutputStream s = new ObjectOutputStream(out); s.writeObject("Today"); s.writeObject(new Date(1,1,2006)); s.close(); // throws IOException y ClassNotFoundException FileInputStream in = new FileInputStream("theTime.dat"); ObjectInputStream s = new ObjectInputStream(in); String today = (String)s.readObject(); Date date = (Date)s.readObject();
s.close();