因为有好几个童鞋问过这个事了,参考ZXPUnicode:
首先编码转换的代码可以是:
- (NSString *)stringByReplaceUnicode:(NSString *)unicodeString
{
NSMutableString *convertedString = [unicodeString mutableCopy];
[convertedString replaceOccurrencesOfString:@"\\U" withString:@"\\u" options:0 range:NSMakeRange(0, convertedString.length)];
CFStringRef transform = CFSTR("Any-Hex/Java");
CFStringTransform((__bridge CFMutableStringRef)convertedString, NULL, transform, YES);
return convertedString;
}
虽然这样可以随便做category
手动调用了,但还是需要引用头文件,不方便。所以需要做Method Swizzling
,替换掉系统的description相关方法,在自定义交换方法中添加如上述代码类似的编码转换操作,一劳永逸(。◕ฺ∀◕ฺ)。具体可以是对NSDictionary
、NSArray
以及NSSet
作一个category
,在类被初始加载调用load
方法时做Method Swizzling
:
+ (void)load
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
zx_swizzleSelector([self class], @selector(descriptionWithLocale:indent:), @selector(zx_descriptionWithLocale:indent:));
});
}
- (NSString *)zx_descriptionWithLocale:(id)locale indent:(NSUInteger)level
{
return [self stringByReplaceUnicode:[self zx_descriptionWithLocale:locale indent:level]];
}
swizzleSelector
中交换了两个方法的实现。所以方法中调用zx_descriptionWithLocale:indent:
其实是调用的原生的descriptionWithLocale:indent:
方法。在系统方法返回时,进行编码转换就OK啦(。◕ฺ∀◕ฺ)。
原生的descriptionWithLocale:indent:
方法是这样获取描述的: 如果元素是NSString
对象则直接返回它; 当元素响应descriptionWithLocale:indent:
方法时,调用方法获得该元素的字符串描述; 当元素响应descriptionWithLocale:
方法时,调用方法获得该元素的字符串描述; 如果上述条件都不符合,就会调用该元素的description
方法获取字符串描述。
原生方法执行遍历子元素的时候,还是会调用descriptionWithLocale:indent:
方法来获取子元素的描述,当原生方法被调用时,因方法实现的交换又会执行自定义的交换方法的代码,形成间接递归,上述条件符合时原生方法会返回正确描述开始回归,回归时依次进行编码转换。这样有个小小的问题,就是回归过程中已经被编码转换过的字符串有可能会被重复转换好多次。这里我们交换的是descriptionWithLocale:indent:
这一个原生方法。如果可以的话,可以只交换原生的description
方法,然后输出的时候调一下description
(例如你有个字典dict,这样打印:NSLog(@"%@", dic.description)
), 这么做只是将系统方法返回的最终描述进行了一次编码转换。ps: 如果你不嫌麻烦的话,可以这么用 ๑乛◡乛๑ 。。。。
补上Method Swizzling代码:
static inline void zx_swizzleSelector(Class theClass, SEL originalSelector, SEL swizzledSelector)
{
Method originalMethod = class_getInstanceMethod(theClass, originalSelector);
Method swizzledMethod = class_getInstanceMethod(theClass, swizzledSelector);
BOOL didAddMethod =
class_addMethod(theClass,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
class_replaceMethod(theClass,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
}