java文件头加密_Java实现文件的加密与解密

最近在做一个项目,需要将资源文件(包括图片、动画等类型)进行简单的加密后再上传至云上的服务器,而在应用程序中对该资源使用前先将读取到的文件数据进行解密以得到真正的文件信息。此策略的原因与好处是将准备好的资源存储在云上,使用时通过网络进行读取即可,减少了应用程序本身的大小。这一点对于移动应用尤其重要,特别是在资源量较大且需要对其进行保护的时候。毕竟在目前所处的大环境下,要想复制一款软件不难,那真正宝贵的就是不容易找到的资源了。

先对文件与加密的相关知识做一个极为简单的科普(知道的可以跳过)。

文件与字串

A、文件的操作流程一般为:打开-->读取-->对内容进行变换-->写入-->关闭。

B、常规性文件操作会用到的类有五个:File,InputStream,OutputStream,FileInoutStream,FileOutputStream,均包含在java.io下面。注意,在使用前必须对类文件进行导入,方法为import java.io.File(实现时需要分号结尾)。

C、创建InputStream类和OutputStream类的对象时,new关键字后边的类分别是FileInputStream和FileOutputStream(而不是其自身),如InputStream fin = new FileInputStream(File objectFile)。可以看出构造参数是File类型对象,其创建方式为File file = new File(String fileName)。

D、当String类对象作为函数参数时,可以直接传入常量字符串,如“D:\source.jpg”。String类对象构造方法中比较简单也是最常用的一种是String string = “string content”,当然,最终执行的是String string = new String(“string content”)。其实String是非常重要的类(可以说无处不在),提供了一套完善、高效操作字串的方法,使得开发者受益匪浅。

E、常规性文件操作涉及到的方法有五个:exist(),read(),write(),flush(),close()。exist()判断文件是否存在,调用者为File类对象;read()读取输入流中的内容,调用者为InputStream类对象;write()、flush()、close()的作用分别为向输出流中写内容、强制发送缓冲区中数据、保存并关闭文件,调用者为OutputStream类对象,不过InputStream类对象在操作完成后也需要close()。

加密算法

SHA:接收一段明文,然后以一种不可逆的方式将它转换成一段(通常更小)密文,也可以简单的理解为取一串输入码(称为预映射或信息),并把它们转化为长度较短、位数固定的输出序列即散列值(也称为信息摘要或信息认证代码)的过程。

DES:把64位的明文输入块变为64位的密文输出块,它所使用的密钥也是64位,主要分为两步:

(1)初始置换,把输入的64位数据块按位重新组合,并把输出分为L0、R0两部分,每部分各长32位,其置换规则为将输入的第58位换到第一位,第50位换到第2位......依此类推,最后一位是原来的第7位。L0、R0则是换位输出后的两部分,L0是输出的左32位,R0是右32位。

(2)逆置换,经过16次迭代运算后,得到L16、R16,将此作为输入,进行逆置换,逆置换正好是初始置换的逆运算,由此即得到密文输出。

3-DES:使用3条56位的密钥对数据进行三次加密,是DES向AES过渡的加密算法(1999年,NIST将3-DES指定为过渡的加密标准)。

AES:使用128、192、和256位密钥,并且用128位分组加密和解密数据。

异或:与其说这是一种加密算法,倒不如称其为文件信息的简单变换,将每一个数据与某给定数据进行异或操作即可完成加密或解密,如dataEncrypt = dataSource^dataSecret。

OK,是时候回到文件加密与解密的具体实现这个主题上来了。后续的举例均采用图片(包括GIF动画)类型,而其他类型资源的实现原理相同,就不一一给出了。首先来看对一幅JPG类型图片进行异或加密的Java实现,由于是第一次给出代码,所以贴上了Java文件“FileEncAndDec.java”的所有内容。

1 import java.io.File;

2 import java.io.InputStream;

3 import java.io.OutputStream;

4 import java.io.FileInputStream;

5 import java.io.FileOutputStream;

6

7 public class FileEncAndDec {

8 private static final int numOfEncAndDec = 0x99; //加密解密秘钥

9 private static int dataOfFile = 0; //文件字节内容

10 public static void main(String[] args) {

11

12 File srcFile = new File("桌面.jpg"); //初始文件

13 File encFile = new File("encFile.tif"); //加密文件

14 File decFile = new File("decFile.bmp"); //解密文件

15

16 try {

17 EncFile(srcFile, encFile); //加密操作

18 } catch (Exception e) {

19 e.printStackTrace();

20 }

21 }./*欢迎加入java交流Q君样:909038429一起吹水聊天

22

23 private static void EncFile(File srcFile, File encFile) throws Exception {

24 if(!srcFile.exists()){

25 System.out.println("source file not exixt");

26 return;

27 }

28

29 if(!encFile.exists()){

30 System.out.println("encrypt file created");

31 encFile.createNewFile();

32 }

33 InputStream fis = new FileInputStream(srcFile);

34 OutputStream fos = new FileOutputStream(encFile);

35

36 while ((dataOfFile = fis.read()) > -1) {

37 fos.write(dataOfFile^numOfEncAndDec);

38 }

39

40 fis.close();

41 fos.flush();

42 fos.close();

43 }

44 }

从代码可以看出,给定的加密秘钥(异或数据,可以在合法范围内随便定义)为十六进制数0x99。图片资源为以中文命名的“桌面.jpg”,加密文件为“encFile.png”,还有值为“decFile.bmp”的String类对象作为解密文件名称。

相对应地,解密的实现几乎和加密相同,只是输入与输出文件不同,看下面代码。

1 private static void DecFile(File encFile, File decFile) throws Exception {

2   if(!encFile.exists()){

3     System.out.println("encrypt file not exixt");

4     return;

5   }

6

7   if(!decFile.exists()){

8     System.out.println("decrypt file created");

9     decFile.createNewFile();

10   }

11

12   InputStream fis = new FileInputStream(encFile);

13   OutputStream fos = new FileOutputStream(decFile);

14

15   while ((dataOfFile = fis.read()) > -1) {

16     fos.write(dataOfFile^numOfEncAndDec);

17   }

18

19   fis.close();

20   fos.flush();

21   fos.close();

22 }./*欢迎加入java交流Q君样:909038429一起吹水聊天

下面给出初始、加密及解密后的图标截图:

542e9966dea0

1.3.png

和预想的一致,经测试发现以上方法对GIF动画(不是GIF图片,而是可以播放的动画资源)的加密与解密同样适用,代码和截图也就没有区别了,不过还是贴上来:

1 File srcFile = new File("srcFile.gif"); //初始文件

2 File encFile = new File("encFile.gif"); //加密文件

3 File decFile = new File("decFile.gif"); //解密文件

542e9966dea0

1.32.png

有两点需要注意:

1、在调用加密与解密方法时,必须加上异常处理块(try{...}catch{...},否则编译不通过)。

2、对用来加密或解密的源文件进行打开(读取)操作之前,最好判断其是否存在,免得造成意想不到的错误和时间的浪费。因为若文件不存在,后续的操作都是没有意义的。

今天就先写到这,总结一下吧。文件加密简单地说就是对数据进行变换,虽然一千种方法可能会有一千种一种结果,但是思想是通用的。关键是加密所采用的算法的难易,有时间会对文中提到的算法用Java进行实现。

542e9966dea0

image

最新2020整理收集的一些高频面试题(都整理成文档),有很多干货,包含mysql,netty,spring,线程,spring cloud、jvm、源码、算法等详细讲解,也有详细的学习规划图,面试题整理等,需要获取这些内容的朋友请加Q君样:909038429

/./*欢迎加入java交流Q君样:909038429一起吹水聊天

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
题目如下: 编写程序实现文件上传功能。 有能力的同学可以自行设计文件下载功能,但是本次作业不要求。 服务器端: 要求能运行如下程序: java FileServer port foldername 实际的运行命令可能为 java FileServer 1234 d:\\share 解释如下: port为服务器开放的网络连接端口 foldername为服务器上某个文件夹,该文件夹存放客户端上传的文件。 客户端: 要求能运行如下程序: java FileClient server_ip port afile java FileClient server_ip port afolder 实际的运行命令可能为 java FileClient 127.0.0.1 1234 d:\\abc\\a.docx java FileClient 127.0.0.1 1234 d:\\abc 上面的程序运行完毕后,在服务器的文件下将看到客户端上传的文件a.docx 能看到客户端上传的文件夹abc,并且abc文件夹下的所有文件和子文件夹都上传到了服务器。 假设客户端有文件夹d:\\abc,该文件夹有子文件若干,有子文件夹若干。 客户端能够将某个文件上传到服务器,也可以将某个文件夹连同文件夹下的所有子文件和子文件夹 全部上传到服务器,服务器端保存客户端的文件夹结构和文件信息。 文件传输过程要求实现内容加密加密算法任意,可以是DES,AES,RSA之类的算法都可以。 注意:是文件在传输过程中加密,到服务器保存的时候,要求和客户端的文件内容一样, 服务器端的文件并不需要加密。 要求支持多客户端并发上传文件,不考虑文件名冲突,假设多个客户端同时上传的文件没有重名现象。 必须使用多线程编程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值