最近的项目遇到了网络请求,需要在请求完成后回调delegate的方法。然而回调时经常遇到这种情况:delegate已经被释放,这时调用其方法则会引起。
objc的runtime中有两种判断类型的方式比较靠谱,他们可以直接取得任意一个objc_object(和id是完全一样的数据类型)的类或者类名。其函数如下:
//Returns the class name of a given object.const char *object_getClassName(id obj); //Returns the class of an object.Class object_getClass(id object);
第一个函数可以返回任意一个id的类名,第二个函数可以返回任意一个id的Class。这两个函数各有优劣。使用第一个函数判断类型是否改变的优点是在 iphone开发环境下默认公开,可以随便调用,缺点是要使用几字节的内存空间用于存放字符串,而且做字符串比较要稍微多花费一些CPU时间。第二个函数 优点是可以将获取的Class指针做为int型保存起来,只需要4字节,且比较起来节约CPU时间,坏处是我们要手动声明一下此函数才可以在自己的代码里 使用,否则会出现一个,提示“Implicit declaration of function ‘object_getClass’ is invalid in C99”,不过手动声明一下只要加一行代码就可以,也不麻烦。
下面是一个实例:
// WebService.h#import@protocol ServiceDelegate;@interface WebService : NSObject { id _myDelegate; Class _originalClass;}@property (nonatomic, assign) id myDelegate;- (void)postDataWithURL:(NSString *)myURL postData:(NSDictionary *)dataDic setDelegate:(id)theDelegate;- (void)serviceFun:(NSDictionary *)paramDic;@end@protocol ServiceDelegate - (void)serviceCallBack:(id)resultObject serviceFlag:(NSInteger)flag;@end
// WebService.m#import "WebService.h"Class object_getClass(id object);@implementation WebService@synthesize myDelegate = _myDelegate;- (void)postDataWithURL:(NSString *)myURL postData:(NSDictionary *)dataDic setDelegate:(id)theDelegate{ self.myDelegate = theDelegate; _originalClass = object_getClass(theDelegate); [NSThread detachNewThreadSelector:@selector(serviceFun:) toTarget:self withObject:dataDic];}- (void)serviceFun:(NSDictionary *)paramDic{ Class currentClass = object_getClass(self.myDelegate); if (currentClass == _originalClass) { // 如果delegate没有被释放 }}@end