實踐出真知,寫代碼是成長必經(jīng)的過程,至死方休。初版——待補充
開發(fā)過程中代碼整理
一. UI部分
1.根據(jù)Xib文件名稱生成一個ViewController的類
- (instancetype)init {
self = [super init];
if (self) {
self = [[[NSBundle mainBundle] loadNibNamed:@"ViewController" owner:nil options:nil] firstObject];
}
return self;
}
2.ViewDidLoad減負(fù)
ViewDidLoad在ViewCotrollView被創(chuàng)建時執(zhí)行,一般我們會在這個方法中進行初始化操作。但是這里我建議不要把過多的初始化代碼直接寫在該方法中,而是分成幾個不同的初始化方法。如:initView()
、initData()
、initInfo()
、initSetting()
等等;另外提倡使用懶加載和預(yù)加載的方式初始化成員變量,具體使用方法,請點擊此處進行學(xué)習(xí)。
3.WebView的代理方法
UIWebView是用來加載網(wǎng)頁的一個控件,目前官方已經(jīng)建議使用WKWebView來替代UIWebView進行網(wǎng)頁的加載和管理,但是我今天只是描述一種概念,與實際使用哪個控件沒有太大關(guān)系。
如果你需要使用UIWebView加載網(wǎng)頁,請至少實現(xiàn)這個代理:
- (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error;
這個代理的作用是當(dāng)頁面加載錯誤的時候做出相應(yīng)的處理,因為這種情況是一定會出現(xiàn)的,比如網(wǎng)絡(luò)問題或者網(wǎng)頁服務(wù)端出現(xiàn)問題等等。如果不處理,頁面就可能變成空白或者未知頁面,這絕對不是我們預(yù)期的顯示結(jié)果。至于其他的代理,請根據(jù)業(yè)務(wù)需求自行決定是否實現(xiàn)。
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType;
- (void)webViewDidStartLoad:(UIWebView *)webView;
- (void)webViewDidFinishLoad:(UIWebView *)webView;
4.導(dǎo)航欄的初始化
下面是在AppDelegate.m文件中對導(dǎo)航欄的一些初始化操作,其中涉及主題顏色設(shè)置、背景色設(shè)置、標(biāo)題的屬性設(shè)置和導(dǎo)航欄樣式設(shè)置,沒有什么要講的,請自行領(lǐng)會;
[[UINavigationBar appearance] setTintColor:[UIColor whiteColor]];
[[UINavigationBar appearance] setBarTintColor:[UIColor colorWithRed:57.0/255.0 green:141.0/255.0 blue:238.0/255.0 alpha:0.5]];
[[UINavigationBar appearance] setTitleTextAttributes:[NSDictionary dictionaryWithObjectsAndKeys:[UIColor whiteColor], NSForegroundColorAttributeName, [UIFont systemFontOfSize:20], NSFontAttributeName, nil]];
[[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleLightContent];
5.UITableView中Cell的重用問題
我開發(fā)的App中有一個頁面是縱向列表嵌套橫向列表,這個頁面在使用真機調(diào)試的時候總是出現(xiàn)卡頓,而使用模擬器就不會
二. 數(shù)據(jù)部分
1.Model的構(gòu)建——KVC
關(guān)于KVC的使用和技巧,有一篇詳解KVC的文章,請點擊上方標(biāo)題或者點擊此處自行學(xué)習(xí),我就不班門弄斧了!
2.打印Model的信息
開發(fā)過程中我們經(jīng)常會使用調(diào)試功能,在控制臺使用po命令打印某個對象或者變量的信息,但是我們發(fā)現(xiàn)如果你嘗試打印一個對象的信息時會出現(xiàn)這樣的信息(以ResultModel為例):
<ResultModel: 0x60800019d5a0>
很顯然這不是我們想看到的信息,我們希望看到的是ResultModel的數(shù)據(jù)結(jié)構(gòu)。那么我們需要重構(gòu)ResultModel的description方法,這個方法是對象的描述方法,可以理解為是對象的字符串表現(xiàn)形式。下面是重構(gòu)代碼:
- (NSString *)description {
return [NSString stringWithFormat:@"\ncode:%lu\nerror:%d\ndata:%@\nmessage:%@\nstatusCode:%@\n", self.code, self.error, self.data, self.message, self.statusCode];`
}
OK,我們再試一下po命令打印ResultModel:
code:0
error:0
data:{
addTime = "2017-08-14 09:56:59";
avatar = "<null>";
blackList = 0;
code = ea376934311645b188e9e789d96d03ab;
flag = 1;
id = 27;
imei = "2ADEF2A7-3090-42D6-888D-0D9BE1D35D3F";
inviter = 1;
isBindBonus = 1;
mobile = 18003860428;
password = qqqqqq;
promoCode = 41GI5;
qq = 511359588;
registerFrom = 2;
trademanager = wbxxmya;
username = "\U795d\U5f6d\U8f89";
vip = 0;
}
message:請求成功
statusCode:SUCCESS
搞定!結(jié)構(gòu)清晰,數(shù)據(jù)一目了然。這才是我們想要的效果,這個方法適應(yīng)任何一個對象,也就是說你可以輕松的打印出字典、數(shù)組、自定義Model甚至ViewController的詳細(xì)信息。
3.截獲所有通過App發(fā)送的請求和返回數(shù)據(jù)
如果你的項目中使用了WebView加載H5頁面,如果你需要獲取H5頁面加載過程中發(fā)出的Ajax請求的數(shù)據(jù),那么你必須會這項技能:NSURLCache獲取所有通過App發(fā)出的請求信息及返回數(shù)據(jù)。
1、首先自定義一個類來繼承NSURLCache:
//EEURLCache.h文件
@interface EEURLCache : NSURLCache
@end
2、然后在代理文件AppDelegate中設(shè)置URLCache代理為自定義的EEURLCache類:
[NSURLCache setSharedURLCache:[[EEURLCache alloc] init]];
3、接著在EEURLCache類中實現(xiàn)截獲數(shù)據(jù)的方法:
- (void)storeCachedResponse:(NSCachedURLResponse *)cachedResponse forRequest:(NSURLRequest *)request {
//cachedResponse為請求結(jié)果,request則是請求的信息;
}
現(xiàn)在你已經(jīng)拿到了所有通過App發(fā)出的網(wǎng)絡(luò)請求,包括Ajax請求,下面你就可以使用這些數(shù)據(jù)了,因為這些數(shù)據(jù)是通過異步的方式拿到的,所以如果想在某個頁面使用,請使用通知的方式把數(shù)據(jù)發(fā)送出去,在使用的地方接收這些數(shù)據(jù)就OK了,關(guān)于通知的使用方法,請自行學(xué)習(xí)。
4.宏還是常量?
開發(fā)過程中常常會用到宏定義或者常量,這兩種方式都可以實現(xiàn)全局的使用,并且具備便于維護的特點。那么我們使用宏還是常量呢?我的建議是需要明確類型的,請一定要使用常量,因為宏事無類型的;如果需要很多步驟才能得到數(shù)據(jù)的,請使用宏;如果可以用方法代替的,請盡量使用工具類方法來實現(xiàn)。基本就是這樣,下面我給出一個簡單的常量定義和實現(xiàn)的方法:
//HGAppConst.h文件
#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
extern NSString *const UMENG_KEY;
extern NSString *const SERVICE_ID;
extern NSString *const BAICHUAN_KEY;
extern NSString *const RONGCLOUD_KEY;
extern NSString *const BAIDU_API_KEY;
extern NSString *const BAIDU_SECRE_KEY;
/*反饋會話ID*/
extern NSString *const FEEDBACK_SESSION_ID;
*************我是一條分割線**************
//HGAppConst.m文件
#import "HGAppConst.h"
NSString *const UMENG_KEY = @"56ab2a07e0f55ac43e001d33";
NSString *const SERVICE_ID = @"KEFU146192494770355";
NSString *const BAICHUAN_KEY = @"23337006";
NSString *const RONGCLOUD_KEY = @"cpj2xarljnt5n";
NSString *const BAIDU_API_KEY = @"3VrbGNWXPfX7Oq0BOQIp8WEO";
NSString *const BAIDU_SECRE_KEY = @"1a9669e9b341282002967c641f23c686";
NSString *const FEEDBACK_SESSION_ID = @"15440428";
5.多類型數(shù)據(jù)或者多選項時,請盡可能使用枚舉變量
一般在項目中不建議使用純數(shù)字的,因為純數(shù)字沒有明確的含義,除非像0、1這種有一定特殊意義的值,否則盡量使用變量、宏、常量或者枚舉變量來表示。而出現(xiàn)多類型數(shù)據(jù)或者多選項的情況時,請盡可能使用枚舉變量,因為枚舉變量可以統(tǒng)一管理同一類別(這里的類別指的是抽象出來的類型,不是數(shù)據(jù)類型,因為枚舉變量的值都是整型)的數(shù)據(jù),比如顏色、大小、字體類型、商品分類、請求類型等等都可以使用枚舉的方式進行定義。
typedef enum {
ValidateTypeUnsignedNumber, //驗證大于0數(shù)量
ValidateTypeSignedNumber, //驗證有符號數(shù)量
ValidateTypeIDCard, //驗證身份證號
ValidateTypePassword, //驗證密碼
ValidateTypeUnsignedMoney, //驗證大于0金額
ValidateTypeSignedMoney, //驗證有符號金額
} ValidateType;
6.關(guān)于數(shù)組
數(shù)組變量在使用時,雖然NSMutableArray比較方便,而且利于擴展,但是還是建議盡可能使用NSArray而不是NSMutableArray,因為NSMutableArray會比NSArray占用更多的內(nèi)存。另外如果必須使用NSMutableArray,請一定記得初始化,否則你會發(fā)現(xiàn)你定義的NSMutableArray會一直為空。
NSMutableArray *mutableArray = [NSMutableArray array];
7.你知道結(jié)構(gòu)體是什么嗎?
大家知道什么是結(jié)構(gòu)體嗎?如果你有C語言基礎(chǔ),那么肯定是知道的,我在這里只是提一下,因為大家對這個結(jié)構(gòu)體的概念基本都理解,如果還沒有相關(guān)的知識,請自行百度。在此我只是想跟大家說一下使用方面的技巧,我們都知道數(shù)據(jù)轉(zhuǎn)模型,就是通常我們所說的Model類,使用起來是挺方便的,只需要傳入一個字典,然后就可以解析出一個Model類型的數(shù)據(jù)類,可以使用點語法直接調(diào)用成員變量。但是這種方式在部分場景中是不建議使用的,比如:這個Model只在一個ViewController中使用到,或者Model的使用頻率不頻繁,且數(shù)據(jù)結(jié)構(gòu)簡單,還有就是這個數(shù)據(jù)只被用到一次,不值得耗費時間來構(gòu)建一個Model類并且配置好相應(yīng)的初始化和構(gòu)造解析方法。這個時候我們就需要使用結(jié)構(gòu)體來代替Model類,這樣既方便,又快捷,而且非常節(jié)省資源,也不需要考慮內(nèi)存泄漏問題,具體原理不在這篇文檔的討論范圍,以后會專門來講。
三. 項目結(jié)構(gòu)
1.CocoaPods的使用
第三方庫的使用可以大幅度提升開發(fā)效率和優(yōu)化使用體驗,但是自主維護這些第三方庫是超級麻煩的,所幸我們有CocoaPods,我么完全可以使用這個工具來管理我們大部分的第三方庫。不過網(wǎng)上對這個工具的使用教程很多,講的也很詳細(xì),我就不多加贅述了,請點擊此處去學(xué)習(xí)。
2.單例類的作用和實現(xiàn)
單例類在我們開發(fā)的過程中會經(jīng)常用到,單例類的具備這些優(yōu)點:
1、提供了對唯一實例的受控訪問。
2、由于在系統(tǒng)內(nèi)存中只存在一個對象,因此可以節(jié)約系統(tǒng)資源,對于一些需要頻繁創(chuàng)建和銷毀的對象單例模式無疑可以提高系統(tǒng)的性能。
3、允許可變數(shù)目的實例。
OK,現(xiàn)在我們以UserInfoModel為例來構(gòu)建一個單例類:
//UserInfoModel.h文件
@interface UserInfoModel : NSObject
......UserInfoModel的數(shù)據(jù)屬性
+ (UserInfoModel *)sharedInstance;
+ (instancetype)userInfoModelWithDict:(NSDictionary *)dict;
- (instancetype)initWithDict:(NSDictionary *)dict;
@end
//UserInfoModel.m文件
#import "HGUserInfoModel.h"
static UserInfoModel* sharedSingleton = nil;
@implementation UserInfoModel
+ (UserInfoModel *)sharedInstance {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
sharedSingleton = [[super allocWithZone:NULL] init];
});
return sharedSingleton;
}
+ (id)allocWithZone:(struct _NSZone *)zone {
return [self sharedInstance];
}
+ (instancetype)userInfoModelWithDict:(NSDictionary *)dict {
return [[self sharedInstance] initWithDict:dict];
}
- (instancetype)initWithDict:(NSDictionary *)dict {
if (self == [super init]) {
[self setValuesForKeysWithDictionary:dict];
}
return self;
}
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
NSLog(@"key is undefinedKey!")
}
@end
四. 其他
1.獲取URL Scheme信息
URL Scheme是iOS應(yīng)用間進行互相喚起的一種機制,以下代理代碼,在AppDelegate中實現(xiàn),即可打印出所有喚起App的URL Scheme信息,包含喚起來源App的BundleID和請求信息等。
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation {
[BaseHelper EELogWithtype:EELogTypeNormalInfomation withTag:@"Calling Application Bundle ID" andMessage:sourceApplication];
[BaseHelper EELogWithtype:EELogTypeNormalInfomation withTag:@"URL" andMessage:url.description];
[BaseHelper EELogWithtype:EELogTypeNormalInfomation withTag:@"URL scheme" andMessage:url.scheme];
[BaseHelper EELogWithtype:EELogTypeNormalInfomation withTag:@"URL query" andMessage:url.query];
[BaseHelper EELogWithtype:EELogTypeNormalInfomation withTag:@"URL host" andMessage:url.host];
return YES;
}
這個URL Scheme的應(yīng)用很廣泛,包括喚起指定頁面、傳入數(shù)據(jù)、應(yīng)用間通訊等等。但是我今天要介紹的是另一個高級用法。就是通過模擬其他App,來截獲URL Scheme的詳細(xì)信息。比如有我知道支付寶的所有URL Scheme,然后我建立一個測試項目,URL Scheme地址直接復(fù)制支付寶App的URL Scheme,然后我給自己的手機安裝個項目,通過某些應(yīng)用支付喚起支付寶會提示是否打開支付寶,神奇的事情就從這里開始,你點擊打開就會發(fā)現(xiàn)打開了剛才安裝的那個項目。這個時候,如果你實現(xiàn)了上面的代理方法,你就可以在獲取到URL Scheme所有的信息,包含URL地址,query和所有參數(shù)。然后你就可以進行分析,做你想做的事情,比如:模擬支付寶交易,修改數(shù)據(jù)請求,使用自己開發(fā)的App進行二次中轉(zhuǎn)等等,只有你想不到,沒有做不到的。
2.退出App了,該釋放的釋放,該清理的清理
App如果退出,不論是主動退出還是意外退出,都需要對需要釋放的資源和需要清空的數(shù)據(jù)做處理,這個并不困難,只要在AppDelegate.m文件中實現(xiàn)以下方法即可。
- (void)applicationWillTerminate:(UIApplication *)application {
//此處添加需要處理的事件即可;
}
3.一個三秒之內(nèi)不能關(guān)閉的彈窗,你需要嗎?
我們知道彈窗是經(jīng)常出現(xiàn)在移動端App中的,彈窗一般分為兩種:列表式彈窗和彈出式彈窗。彈出式彈窗又分兩種,一種是提醒,一種是選擇,而無論哪種都需要有至少一個UIAlertAction來接收點擊事件,從而關(guān)閉這個彈窗,并且執(zhí)行相應(yīng)的事件。現(xiàn)在我給大家搞一個三秒之內(nèi)不能關(guān)閉的彈窗,實現(xiàn)代碼如下:
UIAlertController *alert = [UIPartHelper addAlertWithTile:@"提醒" andMessage:@"禁止使用返利網(wǎng),淘寶客等一切返利行為,因為返利被投訴將處以淘客傭金1.5倍罰款并將淘客傭金退還商家!" haveActionTitle:nil];
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
[alert addAction:[UIAlertAction actionWithTitle:@"知道了" style:UIAlertActionStyleDefault handler:nil]];
});
其中的UIPartHelper是我定義的工具類,不用太過關(guān)注這個,重點在使用dispatch_after方法延時三秒給alert添加一個按鈕,在三秒之內(nèi)用戶根本找不到這個按鈕,也就無法關(guān)閉彈窗,至于有什么作用,用得到的人就拿走,用不到就當(dāng)沒看到就行了。我寫這個東西只是為了拓寬大家的思路,其實很多看似很復(fù)雜的功能都是可以通過這種發(fā)散思維的方式來實現(xiàn)的。