固定長のファイルを読み取る場合は、Neil Coffey のソリューションが適しています。ただし、可変長のファイル (データが入り続ける) の場合、InputStreamReader を介して FileInputStream または FileChannel 入力ストリームで BufferedReader を直接使用すると、いくつかの問題が発生します。たとえば、ケースを検討してください
1) あるオフセットから現在のファイル長までのデータを読み取りたい。したがって、(InputStreamReader を介して) FileInputStream/FileChannel で BR を使用し、その readLine メソッドを使用します。しかし、データの読み取りに忙しい間に、BF の readLine が予想よりも多くのデータを読み取る原因となるデータが追加されたとしましょう (以前のファイル長)
2) readLine は終了しましたが、現在のファイルの長さ/チャンネル位置を読み取ろうとすると、突然データが追加され、現在のファイルの長さ/チャンネル位置が増加しましたが、既にこれよりも少ないデータを読み取っています。
上記のどちらの場合でも、読み取った実際のデータを知ることは困難です (readLine を使用して読み取ったデータの長さをそのまま使用することはできません。これは、キャリッジ リターンなどの一部の文字をスキップするためです)。
そのため、バッファリングされたバイト単位でデータを読み取り、これに対して BufferedReader ラッパーを使用することをお勧めします。私はこのようないくつかのメソッドを書きました
/** Read data from offset to length bytes in RandomAccessFile using BufferedReader
* @param offset
* @param length
* @param accessFile
* @throws IOException
*/
public static void readBufferedLines(long offset, long length, RandomAccessFile accessFile) throws IOException{
if(accessFile == null) return;
int bufferSize = BYTE_BUFFER_SIZE;// constant say 4096
if(offset < length && offset >= 0){
int index = 1;
long curPosition = offset;
/*
* iterate (length-from)/BYTE_BUFFER_SIZE times to read into buffer no matter where new line occurs
*/
while((curPosition + (index * BYTE_BUFFER_SIZE)) < length){
accessFile.seek(offset); // seek to last parsed data rather than last data read in to buffer
byte[] buf = new byte[bufferSize];
int read = accessFile.read(buf, 0, bufferSize);
index++;// Increment whether or not read successful
if(read > 0){
int lastnewLine = getLastLine(read,buf);
if(lastnewLine <= 0){ // no new line found in the buffer reset buffer size and continue
bufferSize = bufferSize+read;
continue;
}
else{
bufferSize = BYTE_BUFFER_SIZE;
}
readLine(buf, 0, lastnewLine); // read the lines from buffer and parse the line
offset = offset+lastnewLine; // update the last data read
}
}
// Read last chunk. The last chunk size in worst case is the total file when no newline occurs
if(offset < length){
accessFile.seek(offset);
byte[] buf = new byte[(int) (length-offset)];
int read = accessFile.read(buf, 0, buf.length);
if(read > 0){
readLine(buf, 0, read);
offset = offset+read; // update the last data read
}
}
}
}
private static void readLine(byte[] buf, int from , int lastnewLine) throws IOException{
String readLine = "";
BufferedReader reader = new BufferedReader(new InputStreamReader(new ByteArrayInputStream(buf,from,lastnewLine) ));
while( (readLine = reader.readLine()) != null){
//do something with readLine
System.out.println(readLine);
}
reader.close();
}
private static int getLastLine(int read, byte[] buf) {
if(buf == null ) return -1;
if(read > buf.length) read = buf.length;
while( read > 0 && !(buf[read-1] == '\n' || buf[read-1] == '\r')) read--;
return read;
}
public static void main(String[] args) throws IOException {
RandomAccessFile accessFile = new RandomAccessFile("C:/sri/test.log", "r");
readBufferedLines(0, accessFile.length(), accessFile);
accessFile.close();
}