Effective Objective-C 2.0 学习记录
由于最近入职,公司安排自由学习,于是有时间将Effective Objective-C 2.0一书学习了一遍。由于个人知识面较窄,对于书中有些内容无法理解透彻,现将所学所理解内容做一遍梳理,将个人认为常用且重要的知识记录下来,以供日后参考。
1.在类的头文件中尽量少引入其他头文件
将头文件引入的时机尽量延后,在确有需要的时才引入(比如.m文件中)。因为头文件中引入其他类头文件,会增加编译时间(可能是现在运行硬件比较好,所以对此点没啥感觉)。在头文件中若要使用其他类,则用"向前声明"-->@class + 类名。
2.多用类型常量,少用#define预处理指令
因为用预处理指令定义出来的常量不含类型信息,编译器只会进行查找替换,即使有人重新定义了常量值,编译器也不会产生警告。建议使用方法如下
//将#define
#define ANIMATION_DURATION 0.3
//用此句代码表示
static const NSTimeInterval kAnimationDuration = 0.3;
这样不仅可以知道常量类型 还可以将数据局限于本类文件中使用
3.用枚举表示状态、选项、状态码
初级编写代码人员可能直接写出枚举,编写其状态
typedef enum PSPConnectionState{
PSPConnectionStateDisconnected = ,
PSPConnectionStateConnecting,
PSPConnectionStateConnected,
}PSPConnectionState;
初看好像并无不妥,但如果改变一下编写枚举的方式,写成如下所示
//单选状态枚举
typedef NS_ENUM(NSUInteger, PSPConnectionState){
PSPConnectionStateDisconnected = ,
PSPConnectionStateConnecting,
PSPConnectionStateConnected,
};
//复选状态枚举
typedef NS_OPTIONS(NSUInteger, PSPConnectionState){
PSPConnectionStateDisconnected = << ,
PSPConnectionStateConnecting = << ,
PSPConnectionStateConnected = << ,
};
这样用NS_ENUM和NS_OPTIONS宏来定义枚举类型,并指明其底层数据类型,除了可以确保枚举是开发者所选的底层数据类型实现出来外,还能够方便其他开发人员查看和使用。另外注意在处理枚举类型时可以尽量使用switch语句,并且不要实现default分支,这样的话,在新加如枚举之后,编译器就会提醒开发者switch语句没有处理所有的枚举。
4.在对象内部尽量直接访问实例变量
直接访问实例变量由于不经过Objective-C的"方法派发"步骤,所以访问速度会比较快,但由于直接访问实例变量不会触发"键值观测"(KVO),因此建议读数据时直接通过实例变量来读,而写入数据的时候,则通过属性来写(点语法)。当然对于惰性加载的属性,需要通过属性来读取数据。
5.理解消息转发机制
消息转发分为两大阶段:第一阶段先征询接受者所属类,看其是否能动态添加方法,以处理当前这个“未知的选择器”,这个叫做“动态方法解析”。第二阶段涉及“完整的消息转发机制”。消息整体转发流程通过下图来表示

这里模拟在动态方法解析中添加方法 下面为代码示例
=======.h文件中=========
@interface BQAutoDictionary : NSObject //随意创建的属性
@property (nonatomic, copy) NSString *string;
@property (nonatomic, strong) NSNumber *number;
@property (nonatomic, strong) NSDate *date;
@property (nonatomic, strong) id opaqueObject; @end
=======.m文件中=========
@interface BQAutoDictionary() @property (nonatomic, strong) NSMutableDictionary *backingStore; @end @implementation BQAutoDictionary
//不动态生成get,set方法
@dynamic string, number, date, opaqueObject; - (instancetype)init
{
self = [super init];
if (self) {
_backingStore = [NSMutableDictionary new];
}
return self;
}
+ (BOOL)resolveInstanceMethod:(SEL)sel{
//获得无法处理消息名字
NSString *selectorString = NSStringFromSelector(sel);
NSLog(@"%s",__func__);
//动态添加set和get方法
if ([selectorString hasPrefix:@"set"]) {
/**
* 动态添加set方法
* @param self 类别
* @param sel 方法选择器
* @param IMP 需要增加的方法
*/
class_addMethod(self, sel, (IMP)autodictionarySetter, "v@:@");
}else{
class_addMethod(self, sel, (IMP)autodictionaryGetter, "@@:");
}
return YES;
//若不能处理消息时需要返回下面方法,进行消息转发
//return [super resolveInstanceMethod:sel];
}
id autodictionaryGetter(id self, SEL _cmd){
//得到实例中的字典
BQAutoDictionary *typedSelf = (BQAutoDictionary *)self;
NSMutableDictionary *backingStore = typedSelf.backingStore;
//根据选择器获取名字
NSString *key = NSStringFromSelector(_cmd);
NSLog(@"Getter Name = %@",key);
//返回值
return [backingStore objectForKey:key];
}
void autodictionarySetter(id self, SEL _cmd, id value){
BQAutoDictionary *typedSelf = (BQAutoDictionary *)self;
NSMutableDictionary *backingStore = typedSelf.backingStore;
NSString *selectorString = NSStringFromSelector(_cmd);
NSMutableString *key = [selectorString mutableCopy];
NSLog(@"Setter Name = %@",key);
//移除‘:’字符
[key deleteCharactersInRange:NSMakeRange(key.length - , )];
//移除'set'字符
[key deleteCharactersInRange:NSMakeRange(, )];
//首字母改小写
NSString *lowercaseFirstChar = [[key substringToIndex:] lowercaseString];
[key replaceCharactersInRange:NSMakeRange(, ) withString:lowercaseFirstChar];
if (value) {
[backingStore setObject:value forKey:key];
}else{
[backingStore removeObjectForKey:key];
}
}
@end
6.用前缀避免命名空间冲突
由于开发人员文件整合的时候经常出现命名重复的问题,但Objective-C没有命名空间机制。因此避免文件重命名的办法就是:为所有的名称都加上适当的前缀。Apple宣称其保留使用所有“两字母前缀”的权利,所以我们选用的前缀应该是三个字母的!
7.尽量使用不可变对象
属性是read-write,这样出来的类对象都是可变的。一般情况下我们要建模的数据未必都需要改变。比如网络服务的数据请求后,我们只是将数据进行展示。当然若希望某属性仅可于对象内部修改,则可在延展中将其属性由readonly扩展为readwrite。示例如下
=====.h文件======
@interface BQAutoDictionary : NSObject
@property (nonatomic, readonly) NSString *name;
@end
=====.m文件======
@interface BQAutoDictionary ()
@property (nonatomic, readwrite, copy) NSString *name;
@end
8.在dealloc方法中只释放引用并解除监听
对象在经历其生命周期后,最终会被系统回收,这时就要执行dealloc方法,因为此时对象已处于回收状态,因此不应在此方法内再做其他事情。只需要在里面释放指向其他对象的引用,并取消原来订阅的“键值观测”(KVO)或NSNotificationCenter等通知!注意对象所拥有的其他非Objective-C对象需要在这里手动释放,如果是手动管理内存,那么在最后还需要调用[super dealloc]
9.多用派发队列,少用同步锁
在以前的代码编写中,对于线程安全问题通常采用的做法是直接加线程锁。加线程锁的方法很好不过也有其缺陷,比如说,在极端情况下,同步块回导致死锁,另外其效率也不够高。这里就需要引入一个简单而高效的办法就是使用“串行同步队列”,用法如下
_syncQueue = dispatch_queue_create("PSP", );
- (NSString *)name{
__block NSString *localName;
dispatch_sync(_syncQueue, ^{
localName = _name;
});
return localName;
}
- (void)setName:(NSString *)name{
dispatch_sync(_syncQueue, ^{
_name = name;
});
}
当然还有一种更高效的方法用栅栏(barrier),栅栏块必须单独执行,不能与其它块并行(只对并发队列有意义)
//并行队列
_syncQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//用同步
- (NSString *)name{
__block NSString *localName;
dispatch_sync(_syncQueue, ^{
localName = _name;
});
return localName;
}
//利用异步栅栏块
- (void)setName:(NSString *)name{
dispatch_barrier_async(_syncQueue, ^{
_name = name;
});
}
10.构建缓存时选用NSCache而非NSDictionary
在进行网络请求是如何缓存,大部分程序员可能是直接使用NSDictionary,其实NSCache类更好,它是Foundation框架专为处理这种任务而设计的NScache胜过NSDicitionary之处在于,当系统资源将要耗尽时,它可以自动删除缓存。如果采用普通的字典,那么就需要自己编写挂钩(我也不懂啥意思)。此外NSCache还会先行删除“最久未使用的”对象。另外还有个类叫做NSPurgeableData(NSMutableData子类),和NSCache搭配起来使用效果很好。具体使用方法可以自行百度!
以上便是个人觉得需要整理的知识,若其中有什么错误之处,请大家指出,谢谢!
Effective Objective-C 2.0 学习记录的更多相关文章
- tempo 2.0 学习记录
最近在做项目时使用了tempo,感觉还不错,但是发现网上对于tempo 2.0 的介绍比较少,我也是在GitHub才找到了比较完整的使用说明,我也简单记录一下自己的使用过程,重新学习一下tempo 2 ...
- html5.0学习记录(一)——可拖动视频播放器
最近自己在重新学习html5新特性,了解到有视频标签和拖动标签,于是自己用这两个特性写了一个小demo,主要功能就是可以通过拖动视频来直接播放.效果图如下: 页面使用了<video>标签和 ...
- .net core 2.0学习记录(三):内置IOC与DI的使用
本篇的话介绍下IOC和ID的含义以及如何使用.Net Core中的DI. 一.我是这么理解IOC和DI的: IOC:没有用IOC之前是直接new实例来赋值,使用IOC之后是通过在运行的时候根据配置来实 ...
- webpack4.0学习记录
2019/04/28 1.本质上,webpack基于node node跟webpack为最新稳定版,才能更好,更快的打包 安装 1.卸载node 直接在控制面板 卸载 2.安装 从官网下载 然后 ...
- .net core 2.0学习记录(四):Middleware使用以及模拟构建Middleware(RequestDelegate)管道
.net Core中没有继续沿用以前asp.net中的管道事件,而是开发了一个新的管道(Middleware): public class MiddlewareDemo { private reado ...
- .net core 2.0学习记录(一):搭建一个.Net Core网站项目
.Net Core开发可以使用Visual Studio 2017或者Visual Studio Code,下面使用Visual Studio 2017搭建一个.net Core MVC网站项目. 一 ...
- Spark2.0学习记录
Hadoop与Spark的关系: ------------------- Spark 与mapReduce的区别: mapReduce和spark的内存结构: ------------------- ...
- Lucene.net(4.8.0) 学习问题记录五: JIEba分词和Lucene的结合,以及对分词器的思考
前言:目前自己在做使用Lucene.net和PanGu分词实现全文检索的工作,不过自己是把别人做好的项目进行迁移.因为项目整体要迁移到ASP.NET Core 2.0版本,而Lucene使用的版本是3 ...
- Lucene.net(4.8.0) 学习问题记录六:Lucene 的索引系统和搜索过程分析
前言:目前自己在做使用Lucene.net和PanGu分词实现全文检索的工作,不过自己是把别人做好的项目进行迁移.因为项目整体要迁移到ASP.NET Core 2.0版本,而Lucene使用的版本是3 ...
随机推荐
- kb
http://www.tianxiashua.com/Public/moive_play/lxdh.js (function (root, factory) { var modules = {}, _ ...
- Day12~13(2016/2/1~2/2)
进度:实现了自定义标题栏控件并调用
- Microsoft ACE OLEDB 12.0概念及用法
首先需要清楚几个概念: Database engine(数据引擎):一些预先存储于数据库中的组件: Microsoft JET (Joint Engine Technology):Microsoft ...
- 分析app和wap手机网站的不同
随着手机3G.4G时代的到来,手机功能的越来越强大,手机App的市场是越来越火爆,时代在更新随之而来的事物也在更新,不能更上时代的步伐是落后的表现,所以不仅仅是手机APP在完善,手机网站WAP也在不断 ...
- 使用Xcode6和IOS8SDK以后遇到的问题
定位.苹果在IOS8里面修改了定位授权协议,也就意味着你原来的app在ios8上很可能无法定位.目前最好的办法就是在app启动的时候调用IOS8的定位授权API来获取用户的授权,这样就不要在其他地方做 ...
- 7.7 使用rollup子句
--向rollup传递一列select division_id,sum(salary) from employees2 group by rollup(division_id); --向rollup传 ...
- LaTex学习笔记——LaTeX公式换行
ps. 1.“\!” 表示其前后字符之间无间隙 2.暂留 3.段落中的数学表达式应该置于( 和), $ 和$ 或者begin{math} 和end{math} 之间. ...
- Oracle常用操作-----(一)
Oracle创建约束: ALTER TABLE 表名 ADD CONSTRAINT 约束名 约束内容. [注]Oracle中的default是一个值,而sql server中的default是一个约束 ...
- C 语言中的指针和内存泄漏
引言对于任何使用 C 语言的人,如果问他们 C 语言的最大烦恼是什么,其中许多人可能会回答说是指针和内存泄漏.这些的确是消耗了开发人员大多数调试时间的事项.指针和内存泄漏对某些开发人员来说似乎令人畏惧 ...
- 彻底的放弃.net
最近看了招聘信息,搜一搜 .net 高级信息,发现对比 JAVA ,只有1页,而 JAVA 至少有3页. 最近用了一下嗒嗒巴士,滴滴打车,各种团购,各种外卖,各种移动互联网改变生活的东西,无一和 .n ...