最近的项目遇到了网络请求,需要在请求完成后回调delegate的方法。然而回调时经常遇到这种情况:delegate已经被释放,这时调用其方法则会引起crash。
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时间,坏处是我们要手动声明一下此函数才可以在自己的代码里 使用,否则会出现一个warning,提示“Implicit declaration of function ‘object_getClass’ is invalid in C99”,不过手动声明一下只要加一行代码就可以,也不麻烦。
下面是一个实例:
// WebService.h #import <Foundation/Foundation.h> @protocol ServiceDelegate; @interface WebService : NSObject { id <ServiceDelegate> _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 <NSObject> - (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