参考
iOS开发探索-Base64编码
iOS URL编码&base64编码
URL安全的Base64编码,解码
为什么需要对URL进行编码
网络标准RFC 1738中规定URL中只能包含英文字、阿拉伯数字以及一些特殊字符。具体包括:字母和数组[0-9a-zA-Z]、特殊符号-._~:?#[]@!$&'()*+,;=` 。URL中若含有这些规定符号以外的符号,URL都是不合法的,因此对于一些含有特殊符号或中文字符的URL,在请求前需要对URL进行编码。
常见的 URL编码方式
下面URL作为本例中编码演示的原始URL
NSString *urlString = @"http://lotheve.com/个人信息?name=URL编码&other=这个参数有很多符号!*'();:@&=+ $,/?#[]%";
URL编码 方式一
使用Foundation框架中的stringByAddingPercentEscapesUsingEncoding方法对url进行编码(该方法在iOS9后弃用)。编解码代码如下:
NSString *encodeUrlString = [urlString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; //url编码(utf8)
NSString *decodeUrlString = [encodeUrlString stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; //url解码
NSLog(@"编码后的URL:%@",encodeUrlString);
NSLog(@"解码后的URL:%@",decodeUrlString);
打印结果:
使用这种方式编码,从结果中可以看到除了中文被编码,!*'();:@&=+ $,/?#[]% 这些符号中只有空格#[]%被编码。事实上这种方式只能对以下14种字符编码(不含冒号):`#%^{}[]|"<>空格这种方式适合于编码路径或者参数中包含中文或其他非法字符的URL,不适合于编码参数中包含URL保留字符【如?/&=等】的URL,因为若参数中含有例如&的保留字符,用这种方法不会对&进行编码,URL在识别的时候会将&作为参数的分隔符,最终识别的参数会有误。不过可以先将需要转码的参数编码之后再设置给URL,这样就不会有问题了。
URL编码 方式二
使用Foundation框架中的stringByAddingPercentEncodingWithAllowedCharacters方法对url进行编码(适用于iOS9之后)。通过传入一个NSCharacterSet字符集对象决定对哪些字符进行编码,所有字符集对象中的字符都会被编码。系统提供了一些包装好的字符集对象,也可以自定义需要编码的字符集对象。
NSString *encodeUrlString = [urlString stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]]; //编码
NSString *decodeUrlString = [encodeUrlString stringByRemovingPercentEncoding]; //解码
NSLog(@"编码后的URL:%@",encodeUrlString);
NSLog(@"解码后的URL:%@",decodeUrlString);
打印结果:
各种系统包装好的编码字符集包含的字符范围:
URLFragmentAllowedCharacterSet “#%<>[]^`{|}
URLHostAllowedCharacterSet “#%/<>?@\^`{|}
URLPasswordAllowedCharacterSet “#%/:<>?@[]^`{|}
URLPathAllowedCharacterSet “#%;<>?[]^`{|}
URLQueryAllowedCharacterSet “#%<>[]^`{|}
URLUserAllowedCharacterSet “#%/:<>?@[]^`
URL编码 方式三
使用CoreFoundation框架中的函数对url进行编码(该方法在iOS9之后同样弃用)。
NSString *encodeUrlString = CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes(kCFAllocatorDefault, (CFStringRef)urlString, NULL, (CFStringRef)@"!*'();:@&=+ $,/?%#[]", kCFStringEncodingUTF8)); //编码
NSString *decodeUrlString = CFBridgingRelease(CFURLCreateStringByReplacingPercentEscapesUsingEncoding(kCFAllocatorDefault, (CFStringRef)encodeUrlString, (CFStringRef)@"!*'();:@&=+ $,/?%#[]", kCFStringEncodingUTF8)); //解码
NSLog(@"编码后的URL:%@",encodeUrlString);
NSLog(@"解码后的URL:%@",decodeUrlString);
打印结果:
编码函数中的第四个参数为需要编码的字符的集合字符串,url中所有在这些字符范围内的字符都会被编码,可以根据实际需求定要转码的字符范围。
URL编码一个容易混淆的地方
虽说是URL编码,但实际操作其实是对字符串进行编码,因此在编码过程中编码对象【字符串】并没有什么“URL光环”,编码时完全根据需要编码的字符集合对整个URL进行遍历编码,即便是对于协议名http,只要目标编码字符集合中含有“http”在,协议同样也会被编码。
Base64编、解码
NSString *origStr = @"这是一个要经过base64编码的原始字符串";
NSData *data = [origStr dataUsingEncoding:NSUTF8StringEncoding];
NSString *base64EncodeString = [data base64EncodedStringWithOptions:0]; //编码
NSData *decodeData = [[NSData alloc] initWithBase64EncodedString:base64EncodeString options:0]; //解码
NSString *decodeString = [[NSString alloc] initWithData:decodeData encoding:NSUTF8StringEncoding];
NSLog(@"原始字符串:%@",origStr);
NSLog(@"base64编码字符串:%@",base64EncodeString);
NSLog(@"base64解码后字符串:%@",decodeString);
什么是 URL安全的Base64编码
指的是将标准base64编码后的字符串用url传递的场景,由于标准base64编码结果中可能含有/+符号,因此应该先对base64编码结果中的/+分别替换成-_,不然拼接到url中会有问题(/+在url作为保留字符有其自身的作用,如同&=?)。
如何进行 URL安全的Base64编码
Base64可以将二进制转码成可见字符方便进行http传输,但是base64转码时会生成“+”,“/”,“=”这些被URL进行转码的特殊字符,导致两方面数据不一致。
我们可以在发送前将“+”,“/”,“=”替换成URL不会转码的字符,接收到数据后,再将这些字符替换回去,再进行解码。
OC实现:
#pragma - 将saveBase64编码中的"-","_"字符串转换成"+","/",字符串长度余4倍的位补"="
+(NSData*)safeUrlBase64Decode:(NSString*)safeUrlbase64Str
{
// '-' -> '+'
// '_' -> '/'
// 不足4倍长度,补'='
NSMutableString * base64Str = [[NSMutableString alloc]initWithString:safeUrlbase64Str];
base64Str = (NSMutableString * )[base64Str stringByReplacingOccurrencesOfString:@"-" withString:@"+"];
base64Str = (NSMutableString * )[base64Str stringByReplacingOccurrencesOfString:@"_" withString:@"/"];
NSInteger mod4 = base64Str.length % 4;
if(mod4 > 0)
[base64Str appendString:[@"====" substringToIndex:(4-mod4)]];
NSLog(@"Base64原文:%@", base64Str);
return [GTMBase64 decodeData:[base64Str dataUsingEncoding:NSUTF8StringEncoding]];
}
#pragma - 因为Base64编码中包含有+,/,=这些不安全的URL字符串,所以要进行换字符
+(NSString*)safeUrlBase64Encode:(NSData*)data
{
// '+' -> '-'
// '/' -> '_'
// '=' -> ''
NSString * base64Str = [GTMBase64 stringByEncodingData:data];
NSMutableString * safeBase64Str = [[NSMutableString alloc]initWithString:base64Str];
safeBase64Str = (NSMutableString * )[safeBase64Str stringByReplacingOccurrencesOfString:@"+" withString:@"-"];
safeBase64Str = (NSMutableString * )[safeBase64Str stringByReplacingOccurrencesOfString:@"/" withString:@"_"];
safeBase64Str = (NSMutableString * )[safeBase64Str stringByReplacingOccurrencesOfString:@"=" withString:@""];
NSLog(@"safeBase64编码:%@", safeBase64Str);
return safeBase64Str;
}