1. ".h"文件中使用"向前聲明"代替導(dǎo)入頭文件
向前聲明(forward declearing)
eg: @class EOCEmployer
不告訴編譯器EOCEmployer類的全部細(xì)節(jié),只告訴編譯器有這個類名,在具體調(diào)用到這個累的方法或者屬性的時候仍需要使用
#import "EOCEmployer.h"
優(yōu)點(diǎn):盡量延后頭文件的導(dǎo)入時間(有利于提高文件的編譯速度)
? ? ? ? ?避免循環(huán)引用
注:若必須在頭文件中引入其他頭文件的時候:
? ? ? 1.) 需要引入頭文件繼承自某一父類,則頭文件中需要導(dǎo)入該父類
? ? ? 2.) 需要遵循協(xié)議的時候
2. 關(guān)于協(xié)議的位置
1.) 最好把協(xié)議放在一個單獨(dú)的頭文件中,而不是集成在一個大的頭文件中(避免循環(huán)引用問題)--委托協(xié)議除外。
委托協(xié)議:定義一個接口,某對象若想接受另一個對象的委托,則需要遵循其protrol(委托),以便成為其(delegate)委托對象。
2.) 委托協(xié)議名通常是(相關(guān)類名 + Delegate)
eg:TestProtocalDelegate
注 : delegate屬性需要用weak來修飾(目的是避免循環(huán)引用)
3. delegate的OP用法
在委托模式和數(shù)據(jù)源的protocol中有一大部分方法是optional的:(這種方法因?yàn)榭梢圆粚?shí)現(xiàn),所以在調(diào)用的時候)
if?( [ _delegaterespondsToSelector : @selector( doMethodA ) ] ) {
[ _delegaterespondsToSelector : @selector( doMethodA ) ];
}
這樣做比較麻煩,如果我們在委托對應(yīng)的類中添加一個枚舉屬性用于標(biāo)識哪個可選方法可以相應(yīng)的話就可以:
在簽訂delegate的類(即事先delegate方法的Class中):
[self.testObjdealWithMethodType:(MethodA|MethodB)];
在創(chuàng)建delegate的類中對應(yīng)的testObjdealWithMethodType方法中:
- (void)dealWithMethodType:(MethodEnum)type{
self.type= type;
NSIntegernum = type;
/**為了避免提示case (MethodA | MethodB)的值在枚舉中不存在的警告*/
switch(num) {
caseMethodA:
[self.delegatedoMethodA];
break;
caseMethodB:
[self.delegatedoMethodB];
break;
case(MethodA|MethodB):
[self.delegatedoMethodA];
[self.delegatedoMethodB];
break;?}
}
4. 使用類型常量代替宏
類型常量的優(yōu)點(diǎn):
1.) 不會與系統(tǒng)的宏因?yàn)槊貜?fù)造成沖突
2.) 包含類型信息,能更好的描述含義
命名規(guī)范:
1.) 局限于某個.m文件的常量(k+常規(guī)命名)
2.) 全局常量:一般寫在.h文件中(類名前綴+常規(guī)命名)
對外公開常量命名規(guī)范:(比如說注冊的通知名)
eg:externNSString*constEOCStringConstent;//.h文件中聲明
NSString*constEOCStringConstent =@"EOCStringConstentName";//.m文件中聲明
常規(guī)命名習(xí)慣:
1.) 方法和變量名使用駝峰法命名:(清晰的方法名從左到右讀起來好像一篇文章)
eg : 初始化一個矩形的方法名:
- (instancetype)initWithWidth:(CGFloat)width Length:(CGFloat)length;
方法命名注意事項(xiàng):
1.) 如果方法的返回值是新創(chuàng)建的,那么方法的前綴應(yīng)該是返回值類型,除非前面還有修飾語 ?注:屬性的存取方法除外(get set 等)
eg:- (NSString*)stringByReplacingOccurrencesOfString:(NSString*)target withString:(NSString*)replacement
? ? ? - (NSString*)lowercaseStringWithLocale:(nullableNSLocale*)locale
2.) 應(yīng)該把表達(dá)參數(shù)類型的名詞放在參數(shù)前面
eg:- (ObjectType)objectAtIndex:(NSUInteger)index;
3.) 如果方法要在當(dāng)前的對象上執(zhí)行操作,那么就應(yīng)該包含動詞,若執(zhí)行動作的時候還需要操作對象,那應(yīng)該在動詞后跟一個或多個名詞。
- (void)replaceObjectAtIndex:(NSUInteger)index withObject:(ObjectType)anObject;
4.) 不要使用str這種簡稱,使用string這種全程
5.) Boolean屬性應(yīng)該加is前綴,如果某方法返回分屬性的Boolean值,那么應(yīng)該根據(jù)其功能選has/is作為前綴
eg:@property(nonatomic,assign,getter=isDeal)Booleandeal;
? ? ? - (BOOL)hasPrefix:(NSString*)str;
? ? ? - (BOOL)isEqualToString:(NSString*)aString;
6.) 將get前綴留給借由輸出的參數(shù)保存返回值的方法
eg:- (void)getObjects:(ObjectType__unsafe_unretained[])objects range:(NSRange)range;
5. 使用枚舉表示:狀態(tài)碼,選項(xiàng),狀態(tài)
寫在最后:我們在使用switch處理Enum語句的時候,不要再最后加default,這樣的話在Enum中添加新的選項(xiàng)的時候編譯器會識別并faculty警告,提示修改
6.屬性與成員變量
1.) 盡量使用屬性來代替成員變量
? ? ?對象布局在編譯器就已經(jīng)固定了,(eg: 只要碰到訪問_firstName變量的代碼,編譯器就會將其替換為(對應(yīng)的)偏移量(offset))這個偏移量是 硬編碼 ,表示該變量距離存放對象的內(nèi)存區(qū)域的起始地址有多遠(yuǎn)。
oc應(yīng)對某些代碼庫:使用了一份就得累的定義,但是其相連接的庫使用了新的類的定義:
? ? ?做法:吧實(shí)例變量當(dāng)做一種存儲偏移量的“特殊變量”交由 類對象 保管。偏移量會在運(yùn)行期查找,如果類的定義變了,那么存儲的偏移量也就變了。
? ? ?這樣甚至可以實(shí)現(xiàn)在運(yùn)行期添加新的實(shí)例變量。
2.) 屬性:一種標(biāo)注寫法。系統(tǒng)默認(rèn)生成get,set方法的實(shí)例變量??梢栽L問封存在對象中的數(shù)據(jù)(并且編譯器會自動生成一套存取方法)。
eg: @property (nonatomic,assign) Method? Enumtype;
? ? ? //等效
? ? @interfaceTestProtocalClass(){
MethodEnumtype1;
}
- (MethodEnum)type1{
returntype1;
}
- (void)setType1:(MethodEnum)type{
type1 = type;
}
注:
1. 使用點(diǎn)語法和直接調(diào)用存取方法沒有任何區(qū)別
2. 在類的實(shí)現(xiàn)中可以使用 @synthesizefirstName = myFirstName 實(shí)現(xiàn)指定實(shí)例變量的名字
3. 使用 @dynamictype; 可以實(shí)現(xiàn)創(chuàng)建屬性的時候不為其創(chuàng)建存取方法,這時候使用代碼訪問屬性不會報(bào)錯(但要自己實(shí)現(xiàn)標(biāo)準(zhǔn)存取方法)
3.) 在讀取實(shí)例變量的時候才用直接訪問的形式,設(shè)置實(shí)例變量通過屬性來做
1. 由于不經(jīng)過方法派發(fā),所以直接訪問實(shí)例變量的速度更快。這種情況下,編譯器所生成的代碼會直接訪問保存對象實(shí)例變量的那塊區(qū)域。
2. 直接訪問實(shí)例變量是不會調(diào)用其“設(shè)置方法”,這就繞過了為相關(guān)屬性設(shè)置的“內(nèi)存管理語義” ?(eg : 在ARC下直接訪問一個copy的屬性的時候不會拷貝該值,只會保留指向新值得指針,銷毀原來的指針--類同聲明為retain)
3. 直接訪問實(shí)例變量不會觸發(fā)KVO,通知(監(jiān)聽的是指針)
4. 通過屬性 (有自定義的set/get方法) 有助于排查與之相關(guān)的錯誤,因?yàn)榭梢栽趕et/get方法中設(shè)置斷點(diǎn),監(jiān)調(diào)該屬性的調(diào)用者和訪問時機(jī)。
注:
1. 在初始化方法中設(shè)置屬性的相關(guān)值的時候總應(yīng)該直接訪問實(shí)例變量 (因?yàn)?: 可能會出現(xiàn)子類重寫set/get方法導(dǎo)致的異常)
2. 當(dāng)使用懶加載的時候 (lazy initialization) 必須通過“get”方法訪問屬性,否則不會初始化 (懶加載不會調(diào)用) 。
要點(diǎn):
1. 在對象內(nèi)部讀取數(shù)據(jù)時應(yīng)該直接通過實(shí)例對象獲取,寫入數(shù)據(jù)的時候通過set屬性方法
2. 在初始化和 delloc 中總應(yīng)該直接通過實(shí)例變量讀寫數(shù)據(jù)。
3. 有時候會使用懶加載的方式配置數(shù)據(jù),此時需要通過屬性讀取數(shù)據(jù)。
8. 對象等同性
"==" :比較的是兩個對象的指針是否相等
"isEqual(id)" :比較的是兩個OC對象是否相等
注:重寫(override)isEqual方法
1. 重寫isEqual設(shè)計(jì)邏輯:(當(dāng)兩個對象中所有的屬性的值都相同的時候兩個對象才相同)
- (BOOL)isEqual:(id)object{
if(self== object) {
return YES;
}
if ( [self class] != [object class]) {
return NO;
} else {
/**
添加兩個值是否相等的判斷邏輯
*/
return nil;
}
}
2. 實(shí)現(xiàn)hash方法
//簡單的實(shí)現(xiàn) :實(shí)現(xiàn)hash的目的是當(dāng)isEqual實(shí)現(xiàn)是必須兩個對象的hash返回值相同才執(zhí)行
- (NSInteger)hash{
return1337;
}
//比較好的實(shí)現(xiàn)方法
- (NSUInteger)hash{
NSUIntegerfirstNameHash = [_firstName hash];
NSUIntegerlastNameHash = [_lastName hash];
NSUIntegerageHash = [_age hash];
return firstNameHash^lastNameHash^ageHash;
}
要點(diǎn)
1. set(集合)中直接存入是不可能存入兩個相等的對象的,但是通過修改已經(jīng)存在在集合中的某個對象,使之與集合中的另一個對象完全相等,這這個集合中就能夠出現(xiàn)兩個完全相等的對象,此時使用:(NSSet *copySet = [set copy])新的道德copySet中不存在重復(fù)的對象(沖的被去掉了一個)
2. 相同的對象必然有相同的hash,但是有相同的hash不一定代表是相同的對象(撞庫)
9. 使用類工廠的方法實(shí)現(xiàn)類族
注:NSArray類的背后是類族(所有類似Array有可變不可變屬性的都是類族)
? ? ? ?。。?!這也就導(dǎo)致 if([mayBeAnArray Class] == [NSArray class]) 永遠(yuǎn)不可能成立
NSArray的初始化方法返回的是隱藏在類族公共接口后面的某個內(nèi)部類型
1. 判斷類的實(shí)例是否屬于某個類(isMenberofClass)
2. 判斷某個類的實(shí)例是否屬于某個類族(isKindofClass)。