android中对文件加密解密的实现

现在项目里面有一个需求,本项目里面下载的视频和文档都不允许通过其他的播放器播放,在培训机构里面这样的需求很多。防止有人交一份钱,把所有的课件就拷给了别人。这样的事情培训机构肯定是不愿意的。现在我项目里面也出了这么个需求。下面介绍一下我的实现。

文件加解密的流程及原理

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

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

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

4、注意事项:

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

主要代码

/**自定义实现简单的文件加密解密工具
 * 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);
  }
}

效果如图:

代码就是这么多,都有注释。以后再有这种需求可以直接用。以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android数据加密之Aes加密

    前言: 项目中除了登陆,支付等接口采用rsa非对称加密,之外的采用aes对称加密,今天我们来认识一下aes加密. 其他几种加密方式:  •Android数据加密之Rsa加密  •Android数据加密之Aes加密  •Android数据加密之Des加密  •Android数据加密之MD5加密  •Android数据加密之Base64编码算法  •Android数据加密之SHA安全散列算法 什么是aes加密? 高级加密标准(英语:Advanced Encryption Standard,缩写:AE

  • Android AES加密工具类分享

    1.AES加密工具类 java不支持PKCS7Padding,只支持PKCS5Padding.我们知道加密算法由算法+模式+填充组成,下一篇介绍iOS和Android通用的AES加密,本篇文章使用PKCS5Padding加密方式. package com.example.aesdemo; import java.io.UnsupportedEncodingException; import javax.crypto.Cipher; import javax.crypto.spec.SecretK

  • Android编程加密算法小结(AES、Base64、RAS加密算法)

    本文实例总结了Android编程加密算法.分享给大家供大家参考,具体如下: android常用加密算法之Base64加密算法: package com.long; /** * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in

  • Android获取apk签名指纹的md5值(防止重新被打包)的实现方法

    本文实例讲述了Android获取apk签名指纹的md5值以防止重新被打包的实现方法.分享给大家供大家参考,具体如下: 做个记录(这里只是Java层的签名校验,java层容易被破解,我建议apk加固下) 获取md5值来进行Apk签名校验, 可以防止apk重新被打包. 下面我说说怎么获取apk签名的md5值(有三种方法) 1.用代码获取签名指纹的md5值 /** * MD5加密 * @param byteStr 需要加密的内容 * @return 返回 byteStr的md5值 */ public

  • Android编程之MD5加密算法实例分析

    本文实例分析了Android编程之MD5加密算法.分享给大家供大家参考,具体如下: Android MD5加密算与J2SE平台一模一样,因为Android 平台支持 java.security.MessageDigest这个包.实际上与J2SE平台一模一样. 算法签名: 复制代码 代码如下: String getMD5(String val) throws NoSuchAlgorithmException 输入一个String(需要加密的文本),得到一个加密输出String(加密后的文本) pa

  • android中AES加解密的使用方法

    今天在android项目中使用AES对数据进行加解密,遇到了很多问题,网上也找了很多资料,也不行.不过最后还是让我给搞出来了,这里把这个记录下来,不要让别人走我的弯路,因为网上绝大多数的例子都是行不通的.好了,接下来开始讲解 1.Aes工具类 package com.example.cheng.aesencrypt; import android.text.TextUtils; import java.security.NoSuchAlgorithmException; import java.

  • Android实现短信加密功能(发送加密短信、解密本地短信)

    短信加密此类功能由于新手学习的需求量较小,所以在网上很少有一些简单的demo供新手参考.小编做到此处也是花了比较多的时间自我构思,具体的过程也是不过多描述了,讲一下demo的内容. demo功能: 1.可以发送短信并且加密(通过改变string中的char) 2.能够查看手机中的短信 3.能够给收到的加密短信解密. 涉及到的知识点: 1.intent bundle传递 2.ContentResolver获取手机短信 3.listveiw与simpleAdapter 4.发送短信以及为发送短信设置

  • Java加密解密工具(适用于JavaSE/JavaEE/Android)

    本文实例为大家分享了一个适用于JavaSE/JavaEE/Android的Java加密解密工具,供大家学习,具体内容如下 package longshu.utils.security; import java.lang.reflect.Method; import java.security.InvalidKeyException; import java.security.Key; import java.security.MessageDigest; import java.security

  • android md5加密与rsa加解密实现代码

    复制代码 代码如下: import java.io.UnsupportedEncodingException; import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; public class MD5 { /* * MD5加密 */ public static String getDigest(String str) { MessageDigest messageDigest = nul

  • Android 加密解密字符串详解

    加密和解密的字符串: 复制代码 代码如下: package eoe.demo; import java.security.SecureRandom; import javax.crypto.Cipher; import javax.crypto.KeyGenerator; import javax.crypto.SecretKey; import javax.crypto.spec.SecretKeySpec; /** * Usage: * <pre> * String crypto = Si

随机推荐