第15条:用前缀避免命名空间冲突
- 选择与你的公司,应用程序或二者皆有关联之名称作为类名的前缀,并在所有代码中使用这以前缀,包括纯C函数及全局变量。前缀应该以三个大写字母为前缀,用前缀还有一个好处,就是有些app在混淆代码的时候,很方便快捷的混淆自己的iOS代码。
- 若自己开发的应用程序需要为其他程序提供插件支持,那么自己的开发的应用程序中使用到的第三方插件也需要修改其前缀。
- 此条看到了一个符号表,此表的功能是显示iOS文件的文件名,函数名,行数等。由此找到腾讯的Bugly插件,此插件可以在iOSAPP发布后,如果APP发生crash,或是卡顿等问题,可以及时查看到问题,定位问题很准确,且还有解决方案。
参考地址Bugly
第16条:提供“全能初始化方法”
- 对于提供多个初始化方法的类,应该提供一个全能的初始化方法,其他初始化方法以此方法为基础进行扩展。
2 在全能初始化方法里进行存储数据等,如果有存储机制改变等原有只需修改全能初始化方法即可。 - 若全能初始化方法与超类不同,则需覆写超类中的
对应方法
。--强调对应的方法,不是乱调用方法。 - 如果超类的初始化方法不适用于子类,那么应该覆写这个超类方法,并在其中抛出异常。
第17条:实现description方法
- 实现decription方法返回一个有意义的字符串,用以描述该实例。
- 若想在调试时打印更详尽的对象描述,则应实现debugDescription方法。
第18条:尽量使用不可变对象
- 在公共接口中(
.h文件
)声明属性的时候,尽量使用不可变对象。 - 在公共接口中,集合使用不可变对象时,尽量使用
readonly
特征修饰,且提供相应的增加移除方法,并重写属性的get
方法。
情况一
.h
文件
@interface WCCPerson : NSObject
@property (nonatomic, copy) NSString *myName;
@property (nonatomic, copy, readonly)NSArray *friends;
- (instancetype)initWithName:(NSString *)myName;
@end
@interface NSArray (WCCPerson)
- (NSString *)toString;
@end
.m
文件
@interface WCCPerson ()
@property (nonatomic, copy, readwrite)NSArray *friends;//将对外的属性中的readonly,改成readwrite
@end
@implementation WCCPerson
- (instancetype)initWithName:(NSString *)myName{
if (self = [super init]) {
_myName = [myName copy];
_friends = @[@"json", @"mike"];
}
return self;
}
- (NSArray *)friends{
return _friends;
}
@end
@implementation NSArray (WCCPerson)
- (NSString *)toString{
NSMutableString *string = [NSMutableString string];
for (NSUInteger i = 0; i < [self count]; i++) {
[string appendFormat:@"array[%lu] value is %@, ", (unsigned long)i, self[i]];
}
return [string copy];
}
@end
如果按照上面的写法也是可以的,对象内部操作_friends
属性,对象内部对属性进行赋值,只允许外部获取。但这样有一些问题,如果用户使用如下方式,则该写法有些漏洞。
WCCPerson *person = [[WCCPerson alloc] initWithName:@"wencun"];
NSArray *array = @[@1, @2];
[person setValue:array forKey:@"friends"];
// person.friends =
NSLog(@"%@",person.friends.toString);
//输出
2017-08-18 11:46:09.011585+0800 WCCTestProj[11807:270808] array[0] value is 1, array[1] value is 2,
虽然这样并不合法,但用户一旦这样操作,那么将不可避免。
用户也可能直接用类型信息查询功能查出属性所对应的实例变量在内存布局中的偏移量,以此来认为设置这个实例变量的值。这种也不符合规范。
情况二
.h
文件
@interface WCCPerson : NSObject
@property (nonatomic, copy) NSString *myName;
@property (nonatomic, copy, readonly)NSArray *friends;
- (instancetype)initWithName:(NSString *)myName;
- (void)addFriends:(NSArray *)objects;
- (void)removeFriends:(NSArray *)objects;
@end
@interface NSArray (WCCPerson)
- (NSString *)toString;
@end
.m
文件
@interface WCCPerson (){
NSMutableArray *mutableFriends;
}
//@property (nonatomic, copy, readwrite)NSArray *friends;
@end
@implementation WCCPerson
- (instancetype)initWithName:(NSString *)myName{
if (self = [super init]) {
_myName = [myName copy];
mutableFriends = [NSMutableArray new];
}
return self;
}
- (void)addFriends:(NSArray *)objects{
[mutableFriends addObject:objects];
}
- (void)removeFriends:(NSArray *)objects{
[mutableFriends removeObject:objects];
}
- (NSArray *)friends{
return [mutableFriends copy];
}
@end
这种写法可以避免使用setValue:forKey:
设置后,获取到自己设置的值,因为如果不使用对象提供的方法addFriends:
,是获取不到用户自己设置的正确的值的,这种情况就迫使用户使用对象提供的方法来设置值。
也可以减少用户对属性的操作,起到封装属性的作用了。
第19条:使用清晰而协调的命名方式
第20条:为私有方法名加前缀
- 给私有方法的名称加上前缀,这样可以很容易地将其同公共方法区分开
- 不要单用一个下划线做私有方法的前缀,因为这种做法是预留给苹果公司的。
第21条:理解Object-C错误模型
- 只有发生了可使整个应用程序崩溃的严重错误时,才应使用异常。此时无须考虑恢复的问题,而且应用程序此时也应该退出。也就是说不用再编写“异常安全”代码了。
id someResource = nil;
if (1) {
@throw [NSException exceptionWithName:NSInvalidArgumentException reason:@"参数异常" userInfo:@{}];
}
[someResource release];
- 在错误不那么严重的情况下,可以指派“委托方法”(delegate method)来处理错误,也可以把错误信息放在NSError对象里,经由“输出参数”返回给调用者。
- 在设计API时,NSError的第一种常见的用法是通过委托协议来传递此错误的,有错误发生时,当前对象会把错误信息经由协议中的某个方法传递给委托对象(delegate)。如下,NSURLConnection在委托协议中NSURLConnectionDelegate中定义了如下方法。
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error;
-
NSError的另外一种常见的用法是:经由方法的“输出参数”返回给调用者,如下:
.h
文件
- (BOOL)doSomething:(NSError *__autoreleasing *)error;
.m
文件
- (BOOL)doSomething:(NSError *__autoreleasing *)error{
NSLog(@"我在这里了doSomething");
if (error) {
*error = [NSError errorWithDomain:NSURLErrorKey code:0 userInfo:@{@"error":@"你错啦....."}];
return NO;
}
return YES;
}
使用ARC时,编译器会把方法签名中的NSError **
转换成NSError *__autoreleasing *
,也就是说,指针所指的对象会在方法执行完毕后自动释放,这个对象必须自动释放,因为该方法的调用者,不能保证他自己会释放掉此方法创建的NSError,所以必须加入autorelase。
当error为nil时,如果不加判断会报错。
第22条:理解NSCopying协议
- 若想令自己所写的对象具有拷贝功能,则需要实现NSCopying协议。
- 如果自定义的对象分为可变版本和不可变版本,那么就要同时实现NSCopying与NSMutableCopying协议。
- 复制对象时需要决定采用深拷贝还是浅拷贝,一般情况下应该尽量执行浅拷贝。
- 如果你所写的对象需要深拷贝,那么可考虑新增一个专门执行深拷贝的方法。
文中一段说:深拷贝的意思是:在拷贝对象自身时,将其底层数据也一并复制过去。Foundation框架中的所有collection类在默认情况下都执行浅拷贝,也就是说,只拷贝容器对象本身,而不复制其中的数据。这样做的主要原因在于,容器内的对象未必都能拷贝,而且调用者也未必想在拷贝容器时一并拷贝其中的每个对象。
NSMutableArray *element = [[NSMutableArray alloc] initWithObjects:@"1", nil];
NSMutableArray *mutableArray = [[NSMutableArray alloc] initWithObjects:element, nil];
//此处复制过后,确实指针变成新的了,似乎是产生了一个新对象。
NSMutableArray *copyMutableArray = [mutableArray mutableCopy];
NSLog(@"element is %p,mutableArray is %p,copyMutableArray is %p",element, mutableArray, copyMutableArray);
NSLog(@"element is %@,mutableArray is %@,copyMutableArray is %@",element, mutableArray, copyMutableArray);
//此处复制后,三个变量的值都变了
[element addObject:@"2"];
NSLog(@"============");
NSLog(@"element is %p,mutableArray is %p,copyMutableArray is %p",element, mutableArray, copyMutableArray);
NSLog(@"element is %@,mutableArray is %@,copyMutableArray is %@",element, mutableArray, copyMutableArray);
输出结果
2017-08-21 11:49:39.730755+0800 WCCTestProj[1678:161529] element is 0x600000443f90,mutableArray is 0x6000004443b0,copyMutableArray is 0x600000443ff0
2017-08-21 11:49:39.730991+0800 WCCTestProj[1678:161529] element is (
1
),mutableArray is (
(
1
)
),copyMutableArray is (
(
1
)
)
2017-08-21 11:49:39.731094+0800 WCCTestProj[1678:161529] ============
2017-08-21 11:49:39.731186+0800 WCCTestProj[1678:161529] element is 0x600000443f90,mutableArray is 0x6000004443b0,copyMutableArray is 0x600000443ff0
2017-08-21 11:49:39.731402+0800 WCCTestProj[1678:161529] element is (
1,
2
),mutableArray is (
(
1,
2
)
),copyMutableArray is (
(
1,
2
)
)