参考链接:http://www.java3z.com/cwbwebhome/article/article5/51041.html
链接原文:
有如下情况下可以用到内存文件映射技术解决问题:
1.不要复制文件中所有的数据,只需要修改文件中局部的数据。
2.并行\分段处理大文件。
如下代码示使用javaNIO局部修改文件中指定位置的部分数据:
- /**
- * 修改文件中的某一部分的数据测试:将字定位置的字母改为大写
- * @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<len;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... "); } }