这里加密使用的是三方库:RNCryptor
,它是一个跨语言AES加密/解密库。
主要目标是Swift
和Objective-C
,但C
,C ++
,C#
,Erlang
,Go
,Haskell
,Java
,PHP
,Python
,Javascript
和Ruby
中都有实现。
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查看需要提供路径,而这也是关键。
思路:
- 将网络请求下来的数据流(NSData)直接进行加密,加密成功后存入沙盒目录中。
- 在查看PDF时,先对加密的PDF进行解密,再将解密的PDF存入沙盒目录中(区分加解密PDF文件)。
- 获取解密的PDF文件路径,查看PDF文件。
- 退出查看当前的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