链接原文:
有如下情况下可以用到内存文件映射技术解决问题:
1.不要复制文件中所有的数据,只需要修改文件中局部的数据。
2.并行\分段处理大文件。
如下代码示使用javaNIO局部修改文件中指定位置的部分数据:
Java代码
/**
* 修改文件中的某一部分的数据测试:将字定位置的字母改为大写
* @param fName :要修改的文件名字
* @param start:起始字节
* @param len:要修改多少个字节
* @return :是否修改成功
* @throws Exception:文件读写中可能出的错
* @author javaFound
*/
public static boolean changeFile(String fName,int start,int len) throws Exception{
//创建一个随机读写文件对象
java.io.RandomAccessFile raf=new java.io.RandomAccessFile(fName,"rw");
long totalLen=raf.length();
System.out.println("文件总长字节是: "+totalLen);
//打开一个文件通道
java.nio.channels.FileChannel channel=raf.getChannel();
//映射文件中的某一部分数据以读写模式到内存中
java.nio.MappedByteBuffer buffer= channel.map(FileChannel.MapMode.READ_WRITE, start, len);
//示例修改字节
for(int i=0;i
byte src= buffer.get(i);
buffer.put(i,(byte)(src-31));//修改Buffer中映射的字节的值
System.out.println("被改为大写的原始字节是:"+src);
}
buffer.force();//强制输出,在buffer中的改动生效到文件
buffer.clear();
channel.close();
raf.close();
return true;
}
//测试主方法
public static void main(String[] args) throws Exception{
changeFile("BigFileRW.java",3,5);
System.out.println(" change OK... ");
}
要想看到测试结果,需要在项目的的当前目录下创建一个名为BigFileRW.java文本文件,其中写上10上以上的字母。运行如上程序,文件中第3个字母起后面5个都变为大写了。
QuickSimpleEncryptor.java
package org.bruce.vertices.asist.security;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.RandomAccessFile;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import org.bruce.vertices.asist.utils.ByteArrayUtil;
/**
* @author Bruce Yang
* 最初的目的是用于快速修改媒体类大文件的文件头。
* 如将大文件的字节数据全部加载到内存中进行处理的话,效率就实在是太低了,
* 所以采用了新 io 的直接在物理文件中进行修改的 api,效率直接提升为旧 io 的 1024 倍~
*/
public class QuickSimpleEncryptor {
public static final int REVERSE_LENGTH = 1024;
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
File f = new File("/Users/user/Desktop/960*640.png");
quickSimpleEncrypt(f);
}
/**
* 2012.06.28.01.45, it's what i really want!!
* @param f
* @return
*/
public static boolean quickSimpleEncrypt(File f) {
try {
// 创建一个随机读写文件对象~
RandomAccessFile raf = new RandomAccessFile(f, "rw");
long totalLen = raf.length();
System.out.println("文件总长字节是: " + totalLen);
// 打开一个文件通道~
FileChannel channel = raf.getChannel();
// 映射文件中的某一部分数据以读写模式到内存中~
MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_WRITE, 0, REVERSE_LENGTH);
// 示例修改字节~
for(int i = 0; i < REVERSE_LENGTH; ++ i) {
byte rawByte = buffer.get(i);
// 修改 Buffer 中映射的字节的值~
buffer.put(i, (byte)~rawByte);
}
// 强制输出,在 buffer 中的改动生效到文件~!
buffer.force();
buffer.clear();
channel.close();
raf.close();
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
// ---------------------------------- 邪恶的分割线 -------------------------------------
/**
* 得不到我想要的效果,写入之后将源文件的数据全部给覆盖掉了~
* @param f
*/
public static void simpleEncryptWrong(File f) {
try {
FileInputStream fis = new FileInputStream(f);
byte[] rawBytes = new byte[REVERSE_LENGTH];
fis.read(rawBytes);
System.out.println(ByteArrayUtil.parseByte2HexStr(rawBytes));
fis.close();
FileOutputStream fos = new FileOutputStream(f);
fos.write(rawBytes, 0, rawBytes.length);
fos.close();
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 修改文件中的某一部分的数据测试: 将字定位置的字母改为大写~
* @param fName要修改的文件名字
* @param offset起始字节
* @param len要修改多少个字节
* @return是否修改成功
* @throws Exception文件读写中可能出的错
* @author javaFound
*/
public static boolean changeFile(String fName, int offset, int len) throws Exception{
// 创建一个随机读写文件对象~
RandomAccessFile raf = new RandomAccessFile(fName, "rw");
long totalLen = raf.length();
System.out.println("文件总长字节是: " + totalLen);
// 打开一个文件通道~
FileChannel channel = raf.getChannel();
// 映射文件中的某一部分数据以读写模式到内存中~
MappedByteBuffer buffer= channel.map(FileChannel.MapMode.READ_WRITE, offset, len);
// 示例修改字节~
for(int i = 0; i < len; ++ i) {
byte src = buffer.get(i);
// 修改 Buffer 中映射的字节的值~
buffer.put(i,(byte)(src-31));
System.out.println("被改为大写的原始字节是:"+src);
}
// 强制输出,在 buffer 中的改动生效到文件~!
buffer.force();
buffer.clear();
channel.close();
raf.close();
return true;
}
public static void changeFileTest() throws Exception {
changeFile("BigFileRW.java", 3, 5);
System.out.println(" change OK... ");
}
}