1、JKDataHelper 數組
在大團隊協同開發過程當中,因爲每一個團隊成員的水平不一,很難控制代碼的質量,保證代碼的健壯性,常常會發生因爲後臺返回異常數據形成app崩潰閃退的狀況,爲了不這樣狀況使用JKDataHelper這個用於處理常見數據容錯的工具,極大程度上下降了由於數據容錯不到位產生崩潰閃退的機率。服務器
在工做中,咱們常常會遇到,因爲服務器返回數據的結構內容發生非正常的改變,而形成app崩潰閃退的狀況,雖然多次強調,可是出現的頻率仍然很高。當時心想雖然很大程度是人員技術水平的緣由,可是若是可以經過技術手段,屏蔽掉這樣的問題。不管你是什麼樣水平的開發者,只要使用了一種工具,就能很大程度上避免相似狀況的發生,豈不更好。就這樣JKDataHelper便應運而生了。 app
對數組進行處理的函數函數
+ (NSArray *)safeArray:(id)array;
內部實現:工具
+ (NSArray *)safeArray:(id)array { if ([array isKindOfClass:[NSArray class]]) { return array; } return nil; }
在app解析後臺API返回的數據時,常常會發生咱們約定好的解析某一個字端後,返回的數據原本應該是數組的,可是異常狀況可能時NSString類型的,也可能時NSDictionary類型的,這個時候若是咱們把解析到的數據執行NSArray相關的方法操做就會crash,好比查找數組中的某一個索引下的元素。上面的這個方法很好的避免了這種狀況的發生。若是不是數組類型的話,直接爲nil,後續即便仍然按照NSArray執行相關的操做也不會產生crash。spa
+ (NSMutableArray *)safeMutableArray:(id)mutableArray + (NSDictionary *)safeDictionary:(id)dict + (NSMutableDictionary *)safeMutableDictionary:(id)dict + (NSString *)safeStr:(id)str + (id)safeObj:(id)obj
以上幾個方法的思路同上。3d
+ (NSString *)safeStr:(id)str defaultStr:(NSString *)defaultStr
這個方法主要是用在解析NSString類型時,若是不是NSString類型,那麼則輸出設定的默認值。日誌
爲了方便使用我用宏定義進行了封裝code
#define JKSafeArray(array) [JKDataHelper safeArray:array] #define JKSafeMutableArray(mutableArray) [JKDataHelper safeMutableArray:mutableArray] #define JKSafeDic(dict) [JKDataHelper safeDictionary:dict] #define JKSafeMutableDic(mutableDict) [JKDataHelper safeMutableDictionary:mutableDict] #define JKSafeStr(str) [JKDataHelper safeStr:str] #define JKSafeStr1(str, defaultStr) [JKDataHelper safeStr:str defaultStr:defaultStr] #define JKSafeObj(obj) [JKDataHelper safeObj:obj]
pod "JKDataHelper"orm
2、AvoidCrash
一、若集成了騰訊Bugly或者友盟等等異常蒐集的SDK,AvoidCrash會影響到它們的異常蒐集嗎?
/** 上報錯誤信息 */ + (void)reportErrorName:(NSString *)errorName errorReason:(NSString *)errorReason callStack:(NSArray *)aStackArray extraInfo:(NSDictionary *)info{ [Bugly reportExceptionWithCategory:3 errorName reason:errorReason callStack:aStackArray extraInfo:info terminateApp:NO]; }
三、在AppDelegate中初始化AvoidCrash而且監聽通知:AvoidCrashNotification
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { [AvoidCrash makeAllEffective]; //監聽通知:AvoidCrashNotification, 獲取AvoidCrash捕獲的崩潰日誌的詳細信息 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(dealwithCrashMessage:) name:AvoidCrashNotification object:nil]; return YES; } - (void)dealwithCrashMessage:(NSNotification *)note { //異常攔截而且經過bugly上報 NSDictionary *info = note.userInfo; NSString *errorReason = [NSString stringWithFormat:@"【ErrorReason】%@========【ErrorPlace】%@========【DefaultToDo】%@========【ErrorName】%@", info[@"errorReason"], info[@"errorPlace"], info[@"defaultToDo"], info[@"errorName"]]; NSArray *callStack = info[@"callStackSymbols"]; [BuglyManager reportErrorName:Bugly_ErrorName_AvoidCrash errorReason:errorReason callStack:callStack extraInfo:nil];
四、寫一個AvoidCrash能夠攔截的異常
NSArray *array = @[@"iOS"]; NSString *string = array[100];
五、在Xcode控制檯能夠看到下圖的輸出
若要捕獲 unrecognized selector sent to instance 類型的異常,
一、首先查看下AvoidCrash中初始化AvoidCrash的兩個方法
/** * * 開始生效.你能夠在AppDelegate的didFinishLaunchingWithOptions方法中調用becomeEffective方法 * 【默認不開啓 對」unrecognized selector sent to instance」防止崩潰的處理】 * */ + (void)becomeEffective; /** * 相比於becomeEffective,增長 * 對」unrecognized selector sent to instance」防止崩潰的處理 * * 可是必須配合setupClassStringsArr:使用 */ + (void)makeAllEffective;
二、若要捕獲 unrecognized selector sent to instance 類型的異常,請使用[AvoidCrash makeAllEffective] 而且配合下面的兩個方法使用。(這兩個方法能夠配合使用,能夠同時使用)
/** * 初始化一個須要防止」unrecognized selector sent to instance」的崩潰的類名數組 * ⚠️不可將@"NSObject"加入classStrings數組中 * ⚠️不可將UI前綴的字符串加入classStrings數組中 */ + (void)setupNoneSelClassStringsArr:(NSArray<NSString *> *)classStrings; /** * 初始化一個須要防止」unrecognized selector sent to instance」的崩潰的類名前綴的數組 * ⚠️不可將UI前綴的字符串(包括@"UI")加入classStringPrefixs數組中 * ⚠️不可將NS前綴的字符串(包括@"NS")加入classStringPrefixs數組中 */ + (void)setupNoneSelClassStringPrefixsArr:(NSArray<NSString *> *)classStringPrefixs;
三、具體的使用方法
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { [AvoidCrash makeAllEffective]; //================================================ // 一、unrecognized selector sent to instance(方式1) //================================================ //若出現unrecognized selector sent to instance而且控制檯輸出: //-[__NSCFConstantString initWithName:age:height:weight:]: unrecognized selector sent to instance //你能夠將@"__NSCFConstantString"添加到以下數組中,固然,你也能夠將它的父類添加到下面數組中 //好比,對於部分字符串,繼承關係以下 //__NSCFConstantString --> __NSCFString --> NSMutableString --> NSString //你能夠將上面四個類隨意一個添加到下面的數組中,建議直接填入 NSString //我所開發的項目中所防止unrecognized selector sent to instance的類有下面幾個,主要是防止後臺數據格式錯亂致使的崩潰。我的以爲若要防止後臺接口數據錯亂,用下面的幾個類便可。 NSArray *noneSelClassStrings = @[ @"NSNull", @"NSNumber", @"NSString", @"NSDictionary", @"NSArray" ]; [AvoidCrash setupNoneSelClassStringsArr:noneSelClassStrings]; //================================================ // 二、unrecognized selector sent to instance(方式2) //================================================ //若須要防止某個前綴的類的unrecognized selector sent to instance //好比你所開發項目中使用的類的前綴:CC、DD //你能夠調用以下方法 NSArray *noneSelClassPrefix = @[ @"CC", @"DD" ]; [AvoidCrash setupNoneSelClassStringPrefixsArr:noneSelClassPrefix]; //監聽通知:AvoidCrashNotification, 獲取AvoidCrash捕獲的崩潰日誌的詳細信息 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(dealwithCrashMessage:) name:AvoidCrashNotification object:nil]; return YES; }