(一)覆盖主类的方法实现
XMPP的IPV6适配
苹果要适配IPV6(only)了,查了一下网上的方法,基本上是在GCDAsyncSocket中改动。这样有个缺点;库更新的时候之前的改动失效。而且一般不建议在第三库里面改。刚好GCDAsyncSocket的的原方法是直接调用另一个方法。
上代码:
#import "GCDAsyncSocket+ECBIPV6.h"
#import "ECBIPV6SocketTransfer.h"
@implementation GCDAsyncSocket (ECBIPV6)
-(BOOL)connectToHost:(NSString *)host onPort:(uint16_t)port error:(NSError *__autoreleasing *)errPtr
{
//地址转换,代码就不贴了
NSString *newhost=[ECBIPV6SocketTransfer convertHostToAddress:host];
return [self connectToHost:newhost onPort:port withTimeout:-1 error:errPtr];
}
@end
GCDAsyncSocket的原方法:
- (BOOL)connectToHost:(NSString*)host onPort:(uint16_t)port error:(NSError **)errPtr
{
return [self connectToHost:host onPort:port withTimeout:-1 error:errPtr];
}
category覆盖主类方法切记要注意的一点是,不管有没有导入头文件,都会覆盖主类的方法。具体原因这里不展开说,如要详细了解,请看雷纯锋的博客:Objective-C Category 的实现原理
(二)使用protocol调用类的隐藏方法
NSobject+YYModel的以其人之道还治其人之身
之前一直在用YYModel的+ (NSArray *)modelArrayWithClass:(Class)cls array:(NSArray *)arr方法,后来项目中导入了YYKit就把YYModel删了,直接用YYkit中的NSobject+YYModel。蛋疼的是头文件里只公开了+ (NSArray *)modelArrayWithClass:(Class)cls json:(id)json。所以用的时候把array转换成json再调用。但是方法实现里却是+ (NSArray *)modelArrayWithClass:(Class)cls json:(id)json将json转化成array再调用+ (NSArray *)modelArrayWithClass:(Class)cls array:(NSArray *)arr。。。。,是不是大写的懵逼。
当然最简单的方法是直接将+ (NSArray *)modelArrayWithClass:(Class)cls array:(NSArray *)arr放在头文件中,但是秉着不改第三方库的原则,我并没有这样做。
往深处说,其实OC中没有隐藏方法,使用蛮力一样能调用头文件没有公开的方法。但这样一点都不能体现文章的主题:妙用。。。
YYModel中有个适配属性键值得方法:+ (NSDictionary *)modelCustomPropertyMapper,这个方法并不要求写在头文件中。看了源代码才发现,作者在NSObject分类中定义了一个协议:
@protocol YYModel
@optional
+ (nullable NSDictionary*)modelCustomPropertyMapper;
看到这里,似乎明白什么了,我马上也在NSArray,NSDictionary的分类中定义了一个协议并遵循它
贴出NSArray+ECB.h中的部分代码
@protocol ECBYYModelArrayProtocol
@optional
+ (NSArray *)modelArrayWithClass:(Class)cls array:(NSArray *)arr;
@end
@interface NSArray (ECB)<ECBYYModelArrayProtocol>
+(NSArray*)yy_modelArrayWithClass:(Class)cls array:(NSArray*)listArray;
NSArray+ECB.m中的实现
+(NSArray*)yy_modelArrayWithClass:(Class)cls array:(NSArray*)listArray
{
return [NSArray modelArrayWithClass:cls array:listArray ];
}
ps:这条后来看下基本没什么用
直接在分类头文件中声明方法也行,不好意思
(三)使用运行时重新定义容器的debugDescription
当数组中元素是OC对象的时候,NSLog或断点Debug用po输出的时候,显示的只有类名和地址。
如果想看到每个元素的属性和属性值,可以在分类中实现debugDescription或Description
这里就要用到运行时遍历对象的属性
代码如下:
- (NSString *)debugDescription
{
__block NSString *debugstring = @"";
[self enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
unsigned int outCount, i;
objc_property_t *properties = class_copyPropertyList([obj class], &outCount);
NSMutableDictionary *dict =[NSMutableDictionary dictionary];
for (i = 0; i<outCount;i++){
objc_property_t property = properties[i];
const char* char_f =property_getName(property);
NSString *propertyName = [NSString stringWithUTF8String:char_f];
id propertyValue = [obj valueForKey:propertyName];
if (propertyValue) [dict setObject:propertyValue forKey:propertyName ];
}
free(properties);
debugstring =[debugstring stringByAppendingString:[NSString stringWithFormat:@"%lu,<%@,%p>,%@\n\n",idx,[obj class],&obj,dict]];
}];
return debugstring;
}
测试效果如下: