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 ...
随机推荐
- 转载:GridControl使用技巧
一.如何解决单击记录整行选中的问题 View->OptionsBehavior->EditorShowMode 设置为:Click 二.如何新增一条记录 (1).gridView.AddN ...
- Mac更换Sublime Text程序图标
更换方法: 下载.icns格式一个图标.http://www.easyicon.net/language.en/iconsearch/sublime/ 终端执行:open /Applications/ ...
- connect to https://dl-ssl.google.com refused
这个可能是网络问题,国内连google服务器经常连不上. 尝试用下面办法试下: 1.上图SDK Manager 的 Tools ->Options打开SDK Manager的Settings, ...
- Linux下使用shell实现上传linux下某个目录下所有文件到ftp
首先我们需要搞清楚单个文件怎么上传,把这个单文件上传到ftp上的实现命名为一个:upload_to_ftp_command.sh 之后,需要弄清楚怎么实现遍历一个目录下的所有文件的,把这个遍历某个目录 ...
- javascript序列化json 第二篇
这一节主要是记录父子关系的JSON生成,废话不多说,直接上代码. public partial class _default : System.Web.UI.Page { public static ...
- GDB调试多线程
先介绍一下GDB多线程调试的基本命令. info threads 显示当前可调试的所有线程,每个线程会有一个GDB为其分配的ID,后面操作线程的时候会用到这个ID. 前面有*的是当前调试的线程. th ...
- VS 解决方案管理器和 编辑窗口同步 联动
对于题目的解释就是 当我点击一下解决方案管理器中的 某一个文档时, 编辑窗口会联动的 同步到对应的窗口。之前好像被我无意中关掉了,今天重新建立一个项目无意中发现怎么设置了 如果想点击右边的文档 ...
- boost源码剖析----boost::any
boost源码剖析----boost::any 有的时候我们需要有一个万能类型来进行一些操作,这时候boost::any就派上用场了. boost::Any testInt(10); int val ...
- 导入maven工程遇见的问题【问题】
原工程是一个基于websocket的maven工程(源工程:http://www.cnblogs.com/xdp-gacl/p/5193279.html),把工程导入eclipse后报错.
- Spark实战4:异常检测算法Scala语言
异常检测原理是根据训练数据的高斯分布,计算均值和方差,若测试数据样本点带入高斯公式计算的概率低于某个阈值(0.1),判定为异常点. 1 创建数据集转化工具类,把csv数据集转化为RDD数据结构 imp ...