该文章出自:guohuaden.com
这里加密使用的是三方库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、图片加解密
这个没有什么思想可言,直接看下代码
#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缓存。
查看代码:
加密
#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];
});
});
}
解密
#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文件
#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/Wheat-Qin/FileEncryption_Demo