在項目開發中,和服務端交互數據時,若服務端數據為空時,會出現 <null>,客戶端解析時會 Crash,為了增強程序的健壯性,減少 Crash 的發生,可以使用 NullSafe 這個類別。它對不識別的類型返回 nil,而不是拋出異常,它減少了例如因為 JSON 解析中 數組或字符串為 null 時導致的 Crash。這些異常對客戶端來說是不可預期的。
使用時只需要把 NullSafe.m 文件拖進工程就可以了,它在程序運行時自動加載,你不需要再導入其他頭文件了。
如果想要禁止 NullSafe 的話,需要設置:NULLSAFE_ENABLED=0,或者在 .pch 文件中添加:
#ifdef DEBUG #define NULLSAFE_ENABLED 0 #endif
#import <objc/runtime.h> #import <Foundation/Foundation.h> #ifndef NULLSAFE_ENABLED #define NULLSAFE_ENABLED 1 #endif #pragma GCC diagnostic ignored "-Wgnu-conditional-omitted-operand" @implementation NSNull (NullSafe) #if NULLSAFE_ENABLED - (NSMethodSignature *)methodSignatureForSelector:(SEL)selector { @synchronized([self class]) { //look up method signature NSMethodSignature *signature = [super methodSignatureForSelector:selector]; if (!signature) { //not supported by NSNull, search other classes static NSMutableSet *classList = nil; static NSMutableDictionary *signatureCache = nil; if (signatureCache == nil) { classList = [[NSMutableSet alloc] init]; signatureCache = [[NSMutableDictionary alloc] init]; //get class list int numClasses = objc_getClassList(NULL, 0); Class *classes = (Class *)malloc(sizeof(Class) * (unsigned long)numClasses); numClasses = objc_getClassList(classes, numClasses); //add to list for checking NSMutableSet *excluded = [NSMutableSet set]; for (int i = 0; i < numClasses; i++) { //determine if class has a superclass Class someClass = classes[i]; Class superclass = class_getSuperclass(someClass); while (superclass) { if (superclass == [NSObject class]) { [classList addObject:someClass]; break; } [excluded addObject:NSStringFromClass(superclass)]; superclass = class_getSuperclass(superclass); } } //remove all classes that have subclasses for (Class someClass in excluded) { [classList removeObject:someClass]; } //free class list free(classes); } //check implementation cache first NSString *selectorString = NSStringFromSelector(selector); signature = signatureCache[selectorString]; if (!signature) { //find implementation for (Class someClass in classList) { if ([someClass instancesRespondToSelector:selector]) { signature = [someClass instanceMethodSignatureForSelector:selector]; break; } } //cache for next time signatureCache[selectorString] = signature ?: [NSNull null]; } else if ([signature isKindOfClass:[NSNull class]]) { signature = nil; } } return signature; } } - (void)forwardInvocation:(NSInvocation *)invocation { invocation.target = nil; [invocation invoke]; } #endif @end