iOS中关于图片、PDF等文件加密
2017-04-27 11:20:25 # Objective-C

这里加密使用的是三方库:RNCryptor,它是一个跨语言AES加密/解密库。

主要目标是SwiftObjective-C,但C,C ++,C#,Erlang,Go,Haskell,Java,PHP,Python,JavascriptRuby中都有实现。

RNCryptor地址:

RNCryptor :https://github.com/RNCryptor/RNCryptor

以及OC专用的地址:

RNCryptor-objc :https://github.com/RNCryptor/RNCryptor-objc

下面就用到的图片和PDF文件加密做一下简单的介绍。

1、图片加解密

这个没有什么思想可言,直接看下代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
#pragma  mark --Image加解密
-(void)imageEncryptionAndDecryption
{
NSData *data = [NSData dataWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"default.jpg" ofType:nil]];
NSError *error;

//加密
NSData *encryptedData = [RNEncryptor encryptData:data withSettings:kRNCryptorAES256Settings password:aPassword error:&error ];

if (!error) {
NSLog(@"^_^ 加密成功 ……——(^_^)\n");
// NSLog(@"encryptedData==%@",encryptedData);
}

//解密
NSData *decryptedData = [RNDecryptor decryptData:encryptedData withPassword:aPassword error:&error];
if (!error) {
NSLog(@"^_^ 解密成功 ……——(^_^)\n");
// NSLog(@"decryptedData==%@",decryptedData);

self.imageView.image = [UIImage imageWithData:decryptedData];
}
}

2、PDF加解密

考虑到PDF文件可能较大的原因,这里在加解密时使用了子线程,以避免加解密过程耗时。

另:PDF查看需要提供路径,而这也是关键。

思路:

  1. 将网络请求下来的数据流(NSData)直接进行加密,加密成功后存入沙盒目录中。
  2. 在查看PDF时,先对加密的PDF进行解密,再将解密的PDF存入沙盒目录中(区分加解密PDF文件)。
  3. 获取解密的PDF文件路径,查看PDF文件。
  4. 退出查看当前的PDF文件时,删除解密后的PDF文件缓存,保留加密的PDF缓存。

查看代码:

加密

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
#pragma mark --PDF加密
-(void)PDFEncryption
{
__block NSData *encryptedData;
__block NSError *error;
NSString *filePath = [[NSBundle mainBundle]pathForResource:@"11.pdf" ofType:nil];
NSString *fileEncryPath = [NSHomeDirectory()stringByAppendingPathComponent:@"/Documents/TKAMC.qgh"];
NSFileManager *fileManager = [NSFileManager defaultManager];
//判断是否已存在加密文件,若存在直接执行解密过程。
if ([fileManager fileExistsAtPath:fileEncryPath]) {

[self PDFDecryptedData:[NSData dataWithContentsOfFile:fileEncryPath]];
return;
}
//异步去加密,防止占用太多内存
dispatch_async(dispatch_get_global_queue(0, 0), ^{
NSData *data = [NSData dataWithContentsOfFile:filePath];
//加密
encryptedData = [RNEncryptor encryptData:data withSettings:kRNCryptorAES256Settings password:aPassword error:&error ];
if (!error) {
NSLog(@"^_^ PDF加密成功 ……——(^_^)\n");
// NSLog(@"encryptedData==%@",encryptedData);
}
//在主线程上写入文件
dispatch_sync(dispatch_get_main_queue(), ^{
BOOL yes = [encryptedData writeToFile:fileEncryPath atomically:NO];
if (yes) {
NSLog(@"加密文件写入成功");
}else{
NSLog(@"加密文件写入失败");
}
NSLog(@"写入PDF路径:%@",fileEncryPath);
[self PDFDecryptedData:encryptedData];
});
});
}

解密

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
#pragma mark ---PDF解密
-(void)PDFDecryptedData:(NSData *)encryptedData{

NSString *fileDecryPath = [NSHomeDirectory()stringByAppendingPathComponent:@"/Documents/TKAMC"];
dispatch_async(dispatch_get_global_queue(0, 0), ^{
// 解密
NSError *error;

if (encryptedData != nil || aPassword != nil) {
NSData *decryptedData = [RNDecryptor decryptData:encryptedData
withPassword:aPassword
error:&error];

dispatch_sync(dispatch_get_main_queue(), ^{
BOOL yes = [decryptedData writeToFile:fileDecryPath atomically:NO];
if (yes) {
NSLog(@"解密文件写入成功");
NSLog(@"写入解密PDF路径:%@",fileDecryPath);
self.filepath = fileDecryPath;
[self pushVC];
}else{
NSLog(@"解密文件写入失败");
}
});
}else{
NSLog(@"加密数据为空");
}
});
}

注:这里加解密时并没有具体实现网络请求模块,只是简单的对本地文件进行了实践,但大体实现过程已经实现。

退出PDF时,删除解密的PDF文件

1
2
3
4
5
6
7
8
9
10
11
12
#pragma mark - ReaderViewControllerDelegate methods
- (void)dismissReaderViewController:(ReaderViewController *)viewController
{

//MARK:退出查看PDF时删除解密存储文件。
NSFileManager *fileManager = [NSFileManager defaultManager];

[fileManager removeItemAtPath:self.filepath error:nil];

[self dismissViewControllerAnimated:YES completion:nil];

}

这里提供一个个人测试使用的一个Demo,仅供参考

FileEncryption_Demo :https://github.com/guohuaCabin/FileEncryption_Demo