写高质量OC代码52建议总结:48.多用块枚举,少用for循环
2017-07-27 17:09
495 查看
以下是for循环针对NSArray,NSSet,NSDictionary的遍历。字典和set都是无序的,无法根据特定的整数下标直接访问其值。遍历set和dictionary都需要而外创建数组储存对象和key。这个中介数组增加了不必要的而外开支。
-(void)demoforArray{
NSArray *anArray = /***/;
for (int i = 0; i < anArray.count; i++) {
id object = anArray[i];
// do something
}
}
-(void)demoforSet{
NSSet *aSet = /***/;
NSArray *objects = [aSet allObjects];
for (int i = 0; i < objects.count; i++) {
id object = object[i];
// do something
}
}
-(void)demoforDic{
NSDictionary *aDictionary = /***/;
NSArray *keys = [aDictionary allKeys];
for (int i = 0; i < keys.count; i++) {
id key = keys[i];
id value = aDictionary[key];
// do something
}
} 以下是通过NSEnumerator遍历,nextObject方法可以返回枚举里的下一个对象,等到所有数据都已经遍历,该方法会返回nil。reverseObjectEnumerator返回数组反向排列。-
-(void)demoFastForArray{
NSArray *anArray = /***/;
for (id object in anArray) {
// do something
}
}
-(void)demoFastForSet{
NSSet *anSet = /***/;
for (id object in anSet) {
// do something
}
}
-(void)demoFastForDic{
NSDictionary *anDic = /***/;
for (id key in anDic) {
id value = anDic[key];
// do something
}
} 以下是基于块的便利方式
-(void)enumerateObjectsUsingBlock:(void(^)(id object, NSUInteger idx, BOOL *stop))block;
前两个参数,分别提供了所针对的对象和下标,第三个参数可以终止便利操作。
-(void)demoBlockForArray{
NSArray *anArray = /***/;
[anArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
// do something
if (....) {
*stop = YES;
}
}];
}
-(void)demoBlockForSet{
NSSet *aSet = /***/;
[aSet enumerateObjectsUsingBlock:^(id _Nonnull obj, BOOL * _Nonnull stop) {
// do something
if (...) {
*stop = YES;
}
}];
}
-(void)demoBlockForDic{
NSDictionary *dic = /***/;
[dic enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
// do something
if (shouldStop) {
*stop = YES;
}
4000
}];
}
反向遍历:NSEnumerationReverse,数组和字典都对应两个方法
- (void)enumerateObjectsWithOptions:(NSEnumerationOptions)opts usingBlock:(void (NS_NOESCAPE ^)(ObjectType obj, NSUInteger idx, BOOL *stop))block NS_AVAILABLE(10_6, 4_0);
- (void)enumerateKeysAndObjectsWithOptions:(NSEnumerationOptions)opts usingBlock:(void (NS_NOESCAPE ^)(KeyType key, ObjectType obj, BOOL *stop))block NS_AVAILABLE(10_6, 4_0);
typedef NS_OPTIONS(NSUInteger, NSEnumerationOptions) {
NSEnumerationConcurrent = (1UL << 0),
NSEnumerationReverse = (1UL << 1),
};
总结:
1.遍历collection有四种方式,最基本的是for循环,其次是NSEnumerator遍历法,快速遍历法,块枚举法。
2.s块枚举发本身就可以通过GCD来并发执行遍历操作。
-(void)demoforArray{
NSArray *anArray = /***/;
for (int i = 0; i < anArray.count; i++) {
id object = anArray[i];
// do something
}
}
-(void)demoforSet{
NSSet *aSet = /***/;
NSArray *objects = [aSet allObjects];
for (int i = 0; i < objects.count; i++) {
id object = object[i];
// do something
}
}
-(void)demoforDic{
NSDictionary *aDictionary = /***/;
NSArray *keys = [aDictionary allKeys];
for (int i = 0; i < keys.count; i++) {
id key = keys[i];
id value = aDictionary[key];
// do something
}
} 以下是通过NSEnumerator遍历,nextObject方法可以返回枚举里的下一个对象,等到所有数据都已经遍历,该方法会返回nil。reverseObjectEnumerator返回数组反向排列。-
-(void)demoNSEnumeratorForArray{ NSArray *anArray = /***/; NSEnumerator *enumerator = [anArray objectEnumerator]; id object; while ((object = [enumerator nextObject]) != nil) { // do something } } -(void)demoNSEnumeratorForSet{ NSSet *anSet = /***/; NSEnumerator *enumerator = [anSet objectEnumerator]; id object; while ((object = [enumerator nextObject]) != nil) { // do something } } -(void)demoNSEnumeratorForDic{ NSDictionary *anDic = /***/; NSEnumerator *enumerator = [anDic keyEnumerator]; id key; while ((key = [enumerator nextObject]) != nil) { // do something } }以下是快速遍历方法
-(void)demoFastForArray{
NSArray *anArray = /***/;
for (id object in anArray) {
// do something
}
}
-(void)demoFastForSet{
NSSet *anSet = /***/;
for (id object in anSet) {
// do something
}
}
-(void)demoFastForDic{
NSDictionary *anDic = /***/;
for (id key in anDic) {
id value = anDic[key];
// do something
}
} 以下是基于块的便利方式
-(void)enumerateObjectsUsingBlock:(void(^)(id object, NSUInteger idx, BOOL *stop))block;
前两个参数,分别提供了所针对的对象和下标,第三个参数可以终止便利操作。
-(void)demoBlockForArray{
NSArray *anArray = /***/;
[anArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
// do something
if (....) {
*stop = YES;
}
}];
}
-(void)demoBlockForSet{
NSSet *aSet = /***/;
[aSet enumerateObjectsUsingBlock:^(id _Nonnull obj, BOOL * _Nonnull stop) {
// do something
if (...) {
*stop = YES;
}
}];
}
-(void)demoBlockForDic{
NSDictionary *dic = /***/;
[dic enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
// do something
if (shouldStop) {
*stop = YES;
}
4000
}];
}
反向遍历:NSEnumerationReverse,数组和字典都对应两个方法
- (void)enumerateObjectsWithOptions:(NSEnumerationOptions)opts usingBlock:(void (NS_NOESCAPE ^)(ObjectType obj, NSUInteger idx, BOOL *stop))block NS_AVAILABLE(10_6, 4_0);
- (void)enumerateKeysAndObjectsWithOptions:(NSEnumerationOptions)opts usingBlock:(void (NS_NOESCAPE ^)(KeyType key, ObjectType obj, BOOL *stop))block NS_AVAILABLE(10_6, 4_0);
typedef NS_OPTIONS(NSUInteger, NSEnumerationOptions) {
NSEnumerationConcurrent = (1UL << 0),
NSEnumerationReverse = (1UL << 1),
};
总结:
1.遍历collection有四种方式,最基本的是for循环,其次是NSEnumerator遍历法,快速遍历法,块枚举法。
2.s块枚举发本身就可以通过GCD来并发执行遍历操作。
相关文章推荐
- 写高质量OC代码52建议总结:41.多用派发列队,少用同步锁
- 写高质量OC代码52建议总结:30.以ARC简化引用计数
- 编写高质量OC代码52建议总结:27.使用“class-continuation 分类” 隐藏实现细节
- 写高质量OC代码52建议总结:51.load和initialize
- 编写高质量OC代码52建议总结:10.关联对象
- 编写高质量OC代码52建议总结:23.通过委托与数据源协议进行对象间通信
- 写高质量OC代码52建议总结:39.用handler块降低代码分散程度
- 写高质量OC代码52建议总结:52.NSTimer会保留其目标对象
- 编写高质量OC代码52建议总结:9.以“族类模式“隐藏实现细节
- 编写高质量OC代码52建议总结:13.用“方法调配技术”调试“黑盒方法”
- 写高质量OC代码52建议总结:44.通过Dispatch Group机制,根据系统资源状况来执行任务
- 编写高质量OC代码52建议总结:22.理解NSCopying协议
- 写高质量OC代码52建议总结:46.不要使用dispatch_get_current_queue
- 写高质量OC代码52建议总结:32.编写“异常安全代码”时留意内存管理问题
- 写高质量OC代码52建议总结:47.熟悉系统架构
- 写高质量OC代码52建议总结:45.使用dispatch_once来执行只需要运行一次的线程安全代码
- 编写高质量OC代码52建议总结:11.理解objc_msgSend的作用(消息机制)
- 写高质量OC代码52建议总结:31.在dealloc方法中只释放引用并解除监听
- 写高质量OC代码52建议总结:33.以弱引用避免保留环
- 编写高质量OC代码52建议总结:21.理解Objective-C的错误模型