博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android文件的加密与解密
阅读量:6537 次
发布时间:2019-06-24

本文共 4581 字,大约阅读时间需要 15 分钟。

文件加解密的流程及原理

1、加密方法:存储文件时,从输入流中截取文件的字节数组,对字节数组进行加密,至于加密的方式和算法就可以视需求而定了,然后把加密后的字节数组写入到文件中,最后生成加密后的文件;

2、解密方法:同加密方法一样,只不过是对字节数据进行解密,最后生成明文文件;

3、加密算法:android系统本身引入了javax包的Cipher类,这个类里提供了各种各样的通用的加密方式,如AES对称加密等;该程序中有个CipherUtil工具类,里面有一些简单的使用Cipher进行AES加解密的方法;当然最好还是好好学习一下Cipher类的使用;

4、注意事项:

a、如何判断一个文件是加密后的文件,最简单的方法就是对加密后的文件统一增加一个后缀名,    然后在解密之后将这个后缀名去除,还原回原有文件格式;    如:密文文件的统一后缀名为“.cipher”,明文文件名为"测试.txt",加密后的密文文件应该为“测试.txt.cipher”;    b、加密文件时还有一个重要的注意事项,就是加密后的密文和明文的长度是否相同,    如果文件时一次读取出所有字节数组进行加密的话不用担心这个问题,    但是当对文件分次读取加密或分段加密的话,就不得不考虑这个问题了,最方便的方法就是保证明文和加密后的密文长度相同;    如果长度不同,由于是分段加密的,密文是由一段一段子密文拼接成的,    解密时会找不到每段子密文,因为不知道每段子密文的长度是多少;

主要代码

/**自定义实现简单的文件加密解密工具

  • Created by zhangshuo on 2016/6/28.

    */
    public class CustomFileCipherUtil {

    /**

    • 加密后的文件的后缀
      */
      public static final String CIPHER_TEXT_SUFFIX = ".cipher";

    /**

    • 加解密时以32K个字节为单位进行加解密计算
      /
      private static final int CIPHER_BUFFER_LENGHT = 32
      1024;

    /**

    • 加密,这里主要是演示加密的原理,没有用什么实际的加密算法
    • @param filePath 明文文件绝对路径
    • @return

      */
      public static boolean encrypt(String filePath, CipherListener listener) {
      try {
      long startTime = System.currentTimeMillis();
      File f = new File(filePath);
      RandomAccessFile raf = new RandomAccessFile(f, "rw");
      long totalLenght = raf.length();
      FileChannel channel = raf.getChannel();

      long multiples = totalLenght / CIPHER_BUFFER_LENGHT;long remainder = totalLenght % CIPHER_BUFFER_LENGHT;MappedByteBuffer buffer = null;byte tmp;byte rawByte;//先对整除部分加密for(int i = 0; i < multiples; i++){    buffer = channel.map(            FileChannel.MapMode.READ_WRITE, i * CIPHER_BUFFER_LENGHT, (i + 1) * CIPHER_BUFFER_LENGHT);    //此处的加密方法很简单,只是简单的异或计算    for (int j = 0; j < CIPHER_BUFFER_LENGHT; ++j) {        rawByte = buffer.get(j);        tmp = (byte) (rawByte ^ j);        buffer.put(j, tmp);        if(null != listener){            listener.onProgress(i * CIPHER_BUFFER_LENGHT + j, totalLenght);        }    }    buffer.force();    buffer.clear();}//对余数部分加密buffer = channel.map(        FileChannel.MapMode.READ_WRITE, multiples * CIPHER_BUFFER_LENGHT, multiples * CIPHER_BUFFER_LENGHT + remainder);for (int j = 0; j < remainder; ++j) {    rawByte = buffer.get(j);    tmp = (byte) (rawByte ^ j);    buffer.put(j, tmp);    if(null != listener){        listener.onProgress(multiples * CIPHER_BUFFER_LENGHT + j, totalLenght);    }}buffer.force();buffer.clear();channel.close();raf.close();//对加密后的文件重命名,增加.cipher后缀

      // f.renameTo(new File(f.getPath() + CIPHER_TEXT_SUFFIX));

      Log.d("加密用时:", (System.currentTimeMillis() - startTime) /1000 + "s");
      return true;
      } catch (Exception e) {
      e.printStackTrace();
      return false;
      }
      }

    /**

    • 解密,这里主要是演示加密的原理,没有用什么实际的加密算法
    • @param filePath 密文文件绝对路径,文件需要以.cipher结尾才会认为其实可解密密文
    • @return

      */
      public static boolean decrypt(String filePath, CipherListener listener) {
      try {
      long startTime = System.currentTimeMillis();
      File f = new File(filePath);
      // if(!f.getPath().toLowerCase().endsWith(CIPHER_TEXT_SUFFIX)){
      // //后缀不同,认为是不可解密的密文
      // return false;
      // }

      RandomAccessFile raf = new RandomAccessFile(f, "rw");long totalLenght = raf.length();FileChannel channel = raf.getChannel();long multiples = totalLenght / CIPHER_BUFFER_LENGHT;long remainder = totalLenght % CIPHER_BUFFER_LENGHT;MappedByteBuffer buffer = null;byte tmp;byte rawByte;//先对整除部分解密for(int i = 0; i < multiples; i++){    buffer = channel.map(            FileChannel.MapMode.READ_WRITE, i * CIPHER_BUFFER_LENGHT, (i + 1) * CIPHER_BUFFER_LENGHT);    //此处的解密方法很简单,只是简单的异或计算    for (int j = 0; j < CIPHER_BUFFER_LENGHT; ++j) {        rawByte = buffer.get(j);        tmp = (byte) (rawByte ^ j);        buffer.put(j, tmp);        if(null != listener){            listener.onProgress(i * CIPHER_BUFFER_LENGHT + j, totalLenght);        }    }    buffer.force();    buffer.clear();}//对余数部分解密buffer = channel.map(        FileChannel.MapMode.READ_WRITE, multiples * CIPHER_BUFFER_LENGHT, multiples * CIPHER_BUFFER_LENGHT + remainder);for (int j = 0; j < remainder; ++j) {    rawByte = buffer.get(j);    tmp = (byte) (rawByte ^ j);    buffer.put(j, tmp);    if(null != listener){        listener.onProgress(multiples * CIPHER_BUFFER_LENGHT + j, totalLenght);    }}buffer.force();buffer.clear();channel.close();raf.close();//对加密后的文件重命名,增加.cipher后缀

      // f.renameTo(new File(f.getPath().substring(f.getPath().toLowerCase().indexOf(CIPHER_TEXT_SUFFIX))));

      Log.d("解密用时:", (System.currentTimeMillis() - startTime) / 1000 + "s");return true;

      } catch (Exception e) {

      e.printStackTrace();
      return false;
      }
      }

    /**

    • 用于加解密进度的监听器
      */
      public interface CipherListener{
      void onProgress(long current, long total);
      }
      }

截图

image1

Demo下载地址:

转载于:https://blog.51cto.com/10579773/2151501

你可能感兴趣的文章
ios开发应用内实现多语言自由切换
查看>>
转:iOS基于MVC的项目重构总结
查看>>
Tire树
查看>>
Multi-voltage和power gating的实现
查看>>
JavaScript面向对象 ~ 原型和继承(1)
查看>>
ubuntu下安装nginx时依赖库zlib,pcre,openssl安装方法
查看>>
spring cloud微服务分布式云架构--hystrix的使用
查看>>
linux tail
查看>>
解决Mac启动Eclipse Memory Analyzer报错问题
查看>>
jquery的$().each,$.each的区别
查看>>
mysql 缓存开启及测试
查看>>
自己写的进度条###
查看>>
windows磁盘扩容(动态磁盘)
查看>>
在jsp页面中添加富文本编译器(ueditor)+ 图片上传功能
查看>>
fedora12下安装oracle11客户端
查看>>
实现批量添加20个用户,用户名为user1-50,密码为user后面跟5个随机字符
查看>>
LVM磁盘管理
查看>>
Net命令详解
查看>>
CentOS linux 高可用集群之heartbeat
查看>>
用bat更改hosts文件批处理
查看>>