I am mapping a file("sample.txt") to memory using FileChannel.map()
and then closing the channel using fc.close()
. After this when I write to the file using FileOutputStream, I am getting the following error:
java.io.FileNotFoundException: sample.txt (The requested operation cannot be per formed on a file with a user-mapped section open)
File f = new File("sample.txt");
RandomAccessFile raf = new RandomAccessFile(f,"rw");
FileChannel fc = raf.getChannel();
MappedByteBuffer mbf = fc.map(FileChannel.MapMode.READ_ONLY, 0, fc.size());
fc.close();
raf.close();
FileOutputStream fos = new FileOutputStream(f);
fos.write(str.getBytes());
fos.close();
I presume this may be due to file being still mapped to the memory even after I close the FileChannel
. Am I right?. If so, how can I "unmap" the file from memory?(I can't find any methods for this in the API).
Thanks.
Edit: Looks like it(adding an unmap method) was submitted as RFE to sun some time back: http://bugs.sun.com/view_bug.do?bug_id=4724038
To work around this bug in Java, I had to do the following, which will work ok for small to medium-sized files:
From the
MappedByteBuffer
javadoc:Try calling
System.gc()
? Even that's only a suggestion to the VM.I found out information about
unmap
, it is a method ofFileChannelImpl
and not accessible, so you can invoke it by java reflect like:sun.misc.Cleaner javadoc says:
Running System.gc() is acceptable solution if your buffers total size is small, but if I was mapping gigabytes of files I would try to implement like this:
But! Make sure you don't access that buffer after cleaning or you will end up with:
Good luck!
Following static method could be used:
But this is unsafe solution because of following:
1) Lead to failures if someone use MappedByteBuffer after unmap
2) It relies on MappedByteBuffer implementation details
The mapped memory is used until it is freed by the garbage collector.
From FileChannel docs
From MappedByteBuffer java doc
So I would suggest ensuring there are no remaining references to the mapped byte buffer and then requesting a garbage collection.