По сути, класс RandomAccessFile похож на пару совмещенных в одном классе потоков DatalnputStream и DataOutputStream, к которым на всем «протяжении» применимы: метод getFilePointer(), показывающий, где вы «находитесь» в данный момент; метод seek(), позволяющий перемещаться на заданную позицию файла; и метод length(), определяющий максимальный размер файла. Вдобавок, конструктор этого класса требует второй аргумент (схоже с методом fopen() в С), устанавливающий режим использования файла: только для чтения (строка «г») или для чтения и для записи (строка «rw»). Поддержки файлов только для записи нет, поэтому разумно предположить, что класс RandomAccessFile можно было бы унаследовать от DatalnputStream без потери функциональности.
Прямое позиционирование допустимо только для класса RandomAccessFile, и работает оно только в случае файлов. Класс BufferedlnputStream позволяет вам
пометить некоторую позицию потока методом mark(), а затем вернуться к ней методом reset(). Однако эта возможность ограничена (позиция запоминается в единственной внутренней переменной) и потому нечасто востребована.
Большая часть (если не вся) функциональности класса RandomAccessFile в JDK 1.4 также реализуется
Типичное использование потоков ввода/вывода
Хотя из классов библиотеки ввода/вывода, реализующих потоки, можно составить множество разнообразных конфигураций, обычно используется несколько наиболее употребимых. Следующие примеры можно рассматривать как простое руководство по созданию типичных сочетаний классов для организации ввода/вывода и координации их взаимодействия.
В этих примерах используется упрощенная обработка исключений с передачей их на консоль, но такой способ подойдет только для небольших программ и утилит. В реальном коде следует использовать более совершенные средства обработки ошибок.
Буферизованное чтение из файла
Чтобы открыть файл для посимвольного чтения, используется класс File-InputReader; имя файла задается в виде строки (String) или объекта File. Ускорить процесс чтения помогает буферизация ввода, для этого полученная ссылка передается в конструктор класса BufferedReader. Так как в интерфейсе класса-имеется метод readLine(), все необходимое для чтения имеется в вашем распоряжении. При достижении конца файла метод readLine() возвращает ссылку null.
//: iо/BufferedInputFile.java import java.io.*;
public class BufferedlnputFile {
// Исключения направляются на консоль-public static String
read(String filename) throws IOException { // Чтение входных данных по строкам BufferedReader in = new BufferedReader( new FileReader(filename));
String s;
StringBuilder sb = new StringBuilderO; while((s = in.readLine())!= null)
sb.append(s + "\n"); in.closeO; return sb.toStringO;
}
public static void main(String[] args) throws IOException {
System.out.pri nt(read("BufferedlnputFi1e.java"));
}
} /// ~
Объект StringBuilder sb служит для объединения всего прочитанного текста (включая переводы строк, поскольку метод readLine() их отбрасывает). В завершение файл закрывается методом close().
Чтение из памяти
В этой секции результат String файла BufferedInputFile.read() используется для создания StringReader. Затем символы последовательно читаются методом read(), и каждый следующий символ посылается на консоль.
II io/MemoryInput java import j^va 10 *;
public class Memorylnput {
public static void main(String[] args) throws IOException {
StringReader in = new StringReader(
BufferedlnputFile.read("MemoryInput java"));
int c.
while((c = in readO) != -1)
System out print((char)c),
}
Обратите внимание: метод read() возвращает следующий символ в формате int, и для правильного вывода его необходимо предварительно преобразовать в char.
Форматированное чтение из памяти
Для чтения «форматированных» данных применяется класс DatalnputStream, ориентированный на ввод/вывод байтов, а не символов. В данном случае необходимо использовать классы иерархии InputStream, а не их аналоги на основе класса Reader. Конечно, можно прочитать все, что угодно (например, файл), через InputStream, но здесь используется тип String.
//• io/FormattedMemoryInput.java import java.io.*,
public class FormattedMemorylnput {
public static void main(Stnng[] args) throws IOException { try {
DatalnputStream in = new DataInputStream( new ByteArrayInputStream(
BufferedlnputFile.read(
"FormattedMemorylnput java") getBytes(
))).
while(true)
System.out.pri nt((char)i n.readByte()): } catch(EOFException e) {
System.err.println(uEnd of stream");
} /// ~
Для преобразования строки в массив байтов, пригодный для помещения в поток ByteArraylnputStream, в классе String предусмотрен метод getBytes(). Полученный ByteArraylnputStream представляет собой поток InputStream, подходящий для передачи DatalnputStream.