Также стоит рассмотреть возможность употребления вместо класса RandomAccessFile механизма отображаемых в память файлов.
Каналы
В этой главе были коротко упомянуты классы каналов PipedlnputStream, Piped-OutputStream, PipedReader и PipedWriter. Это не значит, что они редко используются или не слишком полезны, просто их смысл и действие нельзя донести до понимания до тех пор, пока не объяснена многозадачность: каналы предназначены для связи между отдельными
Средства чтения и записи файлов
Очень часто в программировании производится такая цепочка действий: файл считывается в память, там он изменяется, а потом снова записывается на диск. Одна из проблем при работе с библиотекой ввода/вывода Java состоит в том, что для выполнения таких достаточно типичных операций вам придется написать некоторое количество кода — не существует вспомогательных функций, на которые можно переложить такую деятельность. Что еще хуже, с надстройками вообще трудно запомнить, как открываются файлы. Поэтому имеет смысл добавить в вашу библиотеку вспомогательные классы, которые легко сделают нужное за вас. В Java SE5 у PrintWriter появился вспомогательный конструктор, позволяющий легко открыть текстовый файл для чтения. Тем не менее существует много других типичных задач, часто выполняемых в повседневной работе, и было бы разумно избавиться от лишнего кода, связанного с их выполнением.
Ниже показан такой класс Text File с набором статических методов, построчно считывающих и записывающих текстовые файлы. Вдобавок можно создать экземпляр класса Text File, который будет хранить содержимое файла в списке ArrayList (и функциональность списка ArrayList станет доступной при работе, с содержимым файла):
//• net/mi ndvi ew/uti1/TextFi1e java
// Статические функции для построчного считывания и записи
// текстовых файлов,- а также манипуляции файлом как списком ArrayList
package net mindview util;
import java io.*.
import java util *.
public class TextFile extends ArrayList
StringBuilder sb = new StringBuilderO; try {
BufferedReader in= new BufferedReader(new FileReader(
new Fi1е(fi 1eName).getAbsoluteFi1e()));
try {
String s,
while((s = in readLineO) != null) { sb append(s); sb.append("\n");
}
} finally {
in.closeO;
}
} catch(IOException e) {
throw new RuntimeException(e);
}
return sb toStringO;
}
// Запись файла за один вызов метода: public static void write(String fileName, String text) { try {
PrintWriter out = new PrintWriter(
new FileCfileName).getAbsoluteFileO);
try {
out.print(text); } finally {
out. closeO;
}
} catch(IOException e) {
throw new RuntimeException(e);
}
}
// Чтение файла с разбиением по регулярному выражению: public TextFi1e(String fileName. String splitter) {
super(Arrays.asLi st(readCfi1eName).splitCsplitter))); // Вызов splitO часто оставляет пустой объект // String в начальной позиции: if(get(0) equalsC")) remove(O);
}
// Обычное построчное чтение: public TextFi1e(String fileName) { this(fileName, "\n");
}
public void writeCString fileName) { try {
PrintWriter out = new PrintWriterC
new FileCfil eName). getAbsol uteFi 1 eO);
try {
forCString item : this)
out.println(item);
} finally {
out.closeO;
}
} catch(IOException e) {
throw new RuntimeException(e);
}
}
// Простая проверка :
public static void main(String[] args) {
String file = read("TextFile.java"); writeCtest.txt", file);
TextFile text = new TextFileCtest.txt"); text.wri te("test2.txt");
// Разбиение на уникальный отсортированный список слов: TreeSet
new TextFileC"TextFile.java". "WW+")); // Вывод слов, начинающихся с прописной буквы-System.out.pri ntln(words.headSet("a"));
}
} /* Output:
[0. ArrayList. Arrays. Break. BufferedReader. BufferedWriter. Clean. Display. File.
FileReader, FileWriter, IOException. Normally. Output. PrintWriter. Read. Regular.
RuntimeException. Simple. Static. String. StringBuilder. System, TextFile, Tools,
TreeSet. W. Write]
*///:-
Метод read() присоединяет каждую строку к StringBuilder, а за ней присоединяется перевод строки, удаленный при чтении. Затем возвращается объект String, содержащий весь файл. Метод write() открывает файл и записывает в него текст.
Обратите внимание: к каждой операции открытия файла добавляется парный вызов close() в секции finally. Тем самым обеспечивается гарантированное закрытие файла после завершения работы.
Конструктор использует метод read() для превращения файла в String, после чего он вызывает метод String.split(), чтобы разбить результат на строки. В качестве разделителя используются символы новой строки (если вы будете часто использовать этот класс, то, возможно, захотите переписать этот конструктор, чтобы он работал эффективнее). К сожалению, аналогичного метода для соединения строк нет, так что для записи строк придется обойтись нестатическим методом write().