项目中遇到了一个比较棘手的问题,是做一个网络自动登录功能的app。这个登录本来是用网页授权,然后js将密码是用md5加密,服务器又会生成一组随机数比如/032/222/332,这样需要向服务器post的数据格式就变成了 username=aaa&password=md5(/032yourpassword/222/332),格式,于是屁颠屁颠的用正则匹配(@2152393771)获得字符串,打印出来正确无误,然后就是md5加密,然后就。。。密码错。。。
原因在于/032这这格式是代表8进制,如果控制打印出/032说明反斜杠已经被转义了,我们获取的字符应该是"//032",对其进行编码肯定是不对的,js的md5对象应该是这个八进制代表的字符而不是其字面值。经过长时间的搜索,也没有简单的把“\\"转换成”\“的方法。这样我们必须自己转义。首先八进制字符规则是:/开头,最长3位,至少1位,字面值不应该大于8,不满足任意一项,都会结束一个字符的解释,开始新字符的解释。转换成10进制的最大值为377。
有了逻辑我们可以转义了:
-(NSString *) octalEscape:(NSString*)input{
NSMutableString* writer = [[NSMutableString alloc] initWithCapacity:[input length]*2];
int pos = 0;
int len = [input length];
while (pos<len) {
int consume = [self trans:input :writer :pos];
if (consume==0) {
char c = [input characterAtIndex:pos];
[writer appendFormat:@"%c",c];
pos++;
continue;
}
for (int pt = 0 ; pt < consume; pt++) {
pos+=1;
}
}
return writer;
}
-(int)trans:(NSString*)input :(NSMutableString*)writer :(int)index{
char inputchar = [input characterAtIndex:index];
if (inputchar=='\\' && index < ([input length]-1) && isdigit([input characterAtIndex:(index+1)])) {
int start = index+1;
int end = index +2;
while (end < [input length] && isdigit([input characterAtIndex:end])) {
end ++;
int temp =[[NSString stringWithFormat:@"%d",[[input substringWithRange:NSMakeRange(start, end-start)] intValue]] intValue];
if (temp>OCTAL_MAX) {
end--;
break;
}
}
int result = [[NSString stringWithFormat:@"%d",[[input substringWithRange:NSMakeRange(start, end-start)] intValue]] intValue];
NSString * sss =[NSString stringWithFormat:@"%d",result];
//8进制数转换成10进制数
long ttt = strtol(sss.UTF8String, NULL, 8);
[writer appendFormat:@"%C",ttt];
return 1+end -start;
}
return 0;
}
P.S. 在java中使用Apache 的common lang的stringutils类包可以很方便的转义。