现在的位置: 首页 > 综合 > 正文

对文件进行快速简单加密(防小白)

2018年05月17日 ⁄ 综合 ⁄ 共 4148字 ⁄ 字号 评论关闭

参考链接:http://www.java3z.com/cwbwebhome/article/article5/51041.html

链接原文:

有如下情况下可以用到内存文件映射技术解决问题: 
1.不要复制文件中所有的数据,只需要修改文件中局部的数据。 
2.并行\分段处理大文件。 

  如下代码示使用javaNIO局部修改文件中指定位置的部分数据:

Java代码
  1. /**  
  2.      * 修改文件中的某一部分的数据测试:将字定位置的字母改为大写  
  3.      * @param fName  :要修改的文件名字  
  4.      * @param start:起始字节  
  5.      * @param len:要修改多少个字节  
  6.      * @return :是否修改成功  
  7.      * @throws Exception:文件读写中可能出的错  
  8. * @author  javaFound  
  9.      */  
  10.     public static boolean changeFile(String fName,int start,int len) throws Exception{
      
  11.       //创建一个随机读写文件对象   
  12.         java.io.RandomAccessFile raf=new java.io.RandomAccessFile(fName,"rw");   
  13.         long totalLen=raf.length();   
  14.         System.out.println("文件总长字节是: "+totalLen);   
  15.         //打开一个文件通道   
  16.         java.nio.channels.FileChannel channel=raf.getChannel();   
  17.         //映射文件中的某一部分数据以读写模式到内存中   
  18.         java.nio.MappedByteBuffer buffer=  channel.map(FileChannel.MapMode.READ_WRITE, start, len);   
  19.         //示例修改字节   
  20.         for(int i=0;i<len;i++){
      
  21.         byte src=   buffer.get(i);   
  22.         buffer.put(i,(byte)(src-31));//修改Buffer中映射的字节的值   
  23.         System.out.println("被改为大写的原始字节是:"+src);   
  24.        }   
  25.         buffer.force();//强制输出,在buffer中的改动生效到文件   
  26.         buffer.clear();   
  27.         channel.close();   
  28.         raf.close();   
  29.         return true;   
  30.     }   
  31.     //测试主方法   
  32.     public static void main(String[] args) throws Exception{
      
  33.         changeFile("BigFileRW.java",3,5);   
  34.         System.out.println(" change OK... ");   
  35.     }  

要想看到测试结果,需要在项目的的当前目录下创建一个名为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... ");   
    }
}

抱歉!评论已关闭.