Effective Objective-C 2.0学习记录(一)

news2025/1/10 11:32:52

48.多用块枚举,少用for循环

  • for循环
  • 快速枚举(快速遍历)
  • 基于块的遍历方式

在编程中经常需要用到列举collection(NSArray、NSDictionary、NSSet等)中的元素,当前的Objective-C语言有多种办法实现此功能,可以用标准的C语言循环,也可以用OC1.0中的NSEnumerator以及OC2.0中的快速遍历。语言中引入“块”这一特性后,又多出来几种新的遍历方式,而这几种方式往往会容易被开发者忽视。采取这几种新方式遍历collection是,可以传入块,而collection中的每个元素都可能会放在块里被运行一遍,这种做法往往会大幅简化代码过程。

for循环

遍历数组的第一种办法就是老式的for循环,其功能非常有限基本写法如下:

NSArray* oneArray = /*...*/;
    for (int i = 0; i < oneArray.count; i++) {
        id object = oneArray[i];
        //Do something with 'object'
    }
    
    NSDictionary* oneDictionary = /*...*/;
    NSArray* keys = /*...*/;
    for (int i = 0; i < keys.count; i++) {
        id key = keys[i];
        id value = oneDictionary[key];
        
        //Do something with 'key' and 'value'
    }
    
    
    NSSet* oneSet = /*...*/;
    NSArray* objects = [oneSet allObjects];
    for (int i = 0; i < objects.count; i++) {
        id object = objects[i];
        //Do something with 'object'
    }

根据定义,字典和set都是无序的,所以无法根据特定的整数下表来直接访问,于是就需要先获取字典的所有键或是set里的所有对象,这两种情况下,都可以在获取到的有序数组上遍历来访问原字典已经原set上的值。但是缺点也是显而易见的。创建数组有额外开销,其他遍历方式中并不需要像这样去创建中介数组。
值得一提的是,for循环可以实现反向遍历。令“i–”这种方式执行,会比其他的方式进行反向遍历简单很多。

快速枚举(快速遍历)

NSEnumerator是一个抽象基类,其中只定义了两个方法

- (NSArray*)allobjects;
- (id)nextObject;

在Objective-C 1.0,有了一种使用nectObject的方法来实现遍历的新方式,利用的是枚举。
例:

NSArray* oneArray = /*...*/;
NSEnumerator* enumerator = [oneArray objectEnumerator];
    for (obejct = [enumerator nextObject] != nil) {
        id object = oneArray[i];
        //Do something with 'object'
    }

在Objective-C 2.0中,引入了快速遍历,与NSEnumerator来遍历差不多,但是更简洁,它为for循环开创了 in 关键字。

//array
for (id object in oneArray) {
        //Do something with 'object'
    }
//dictionary 
    for (id key in oneDictionary) {
        id value = oneDictionary[key];
        //Do something with 'key' and 'value'
    } 
//set    
    for (id object in oneSet) {
        //Do something with 'object'
    }

由于NSEnumerator对象也实现了NSFastEnumeration协议,所以可以用来执行反向遍历。

如下

for (id object in [oneArray reverseObjectEnumerator]) {
        //Do something with 'object'
    }

基于块的遍历方式

在当前的OC语言中,最新引入的一种做法就是基于块来遍历。NSArray中定义了下面这个方法,它可以实现最基本的遍历功能:

- (void) enumerateObjectsUsingBlock:(void(^)(id object, NSUInteger idx, BOOL *stop))block

除此之外,还有一系列类似的遍历方法,它们可以接受各种选项,已控制遍历操作,稍后会讨论那些方法。
在遍历数组及set是,每次迭代都要执行block参数所传入的块,这个块有三个参数,分别是当前迭代所针对的对象、所针对的下标、以及指向布尔值的指针。。前两个参数的意义不言而喻,而通过第三个参数所提供的机制,可以终止遍历操作。

例如:

[oneArray enumerateObjectsUsingBlock:^(id object, NSUInteger idx, BOOL *stop) {
        //Do something with 'object'
//        if (shouldStop) {
//            *stop = YES;
//        }
	}];

[oneDictionary enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {
        //Do something with 'key' and 'value'
//        if (shouldStop) {
//            *stop = YES;
//        }
	}];
    
    [oneSet enumerateObjectsUsingBlock:^(id obj, BOOL *stop) {
        //Do something with 'object'
//        if (shouldStop) {
//            *stop = YES;
//        }
	}];

此方式大大胜过其他方式的地方在于:遍历时可以直接从块里获取更多信息。在遍历数组时,可以知道当前所针对的下标。遍历有序set时也一样。
而在遍历字典时,无须额外编码,即可同时知道键与值,因而省却了根据键获取值这一步骤。用这种方式遍历字典,可以同时得知键与值,这可能比其他方式快很多,因为在字典内部数据结构中,键与值本来就是存储在一起的。

另外一个好处就是,能够修改块的方法签名。以免进行类型转换的操作。从效果上讲,相当于把本来需要执行的类型操作交给块方法签名来做。
比方说,要用“快速遍历法”来遍历字典。若已知字典中的对象必为字符串,则可以这样编码:

for (NSString* key in oneDictionary) {
        NSString* object = (NSString*)oneDictionary[key];
        //Do something with 'key' and 'object'
    }

如果改用基于块的方式来遍历,那么就可以在块方法签名中直接转换:

[oneDictionary enumerateKeysAndObjectsUsingBlock:^(NSString* key, NSString* obj, BOOL *stop) {
        //Do something with 'key' and 'obj'
 	}];

指定对象的精确类型后,编译器就可以检测出开发者是否调用了该对象所不具备的方法,并在发现这种问题时报错。
如果明确知道某collection中的对象是什么类型,那就应该使用这种方法指明其类型。

用此方式可以进行反向遍历。之前所实现的是这里面的另一个版本(第一个),这是第二个“withOptions”,开发者可向其传入"选项掩码"(option mask):

在这里插入图片描述
总体来看,块枚举法有着其他遍历方式都具备的优势,还能带来更多的好处,与快速遍历法相比,它需要更多的代码,但是却能提供遍历时所针对的下标,在遍历字典时也能提供键与值,而且选项还有开启并发迭代功能,所以多写这点代码还是值得的。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/99161.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【专栏】核心篇09| 怎么保证缓存与DB的数据一致性

计算机类PDF整理&#xff1a;【详细&#xff01;&#xff01;】计算机类PDF整理 Redis专栏合集 【专栏】01| Redis夜的第一章 【专栏】基础篇02| Redis 旁路缓存的价值 【专栏】基础篇03| Redis 花样的数据结构 【专栏】基础篇04| Redis 该怎么保证数据不丢失&#xff08;上…

Python -- 模块和包

目录 1.Python中的模块 1.1 import 1.3 from...import * 1.4 as别名 2.常见的系统模块和使用 2.1 OS模块 2.2 sys模块 2.3 math模块 2.4 random模块 2.5 datetime模块 2.6 time模块 2.7 calendar模块 2.8 hashlib模块 2.9 hmac模块 2.10 copy模块 3.pip命令的使…

【机器学习---01】机器学习

文章目录1. 什么是机器学习&#xff1f;2. 机器学习分类2.1 基本分类2.2 按模型分类2.3 其他分类(不重要)3. 机器学习三要素4. 监督学习的应用(分类、标注、回归问题)1. 什么是机器学习&#xff1f; 定义&#xff1a;给定训练集D&#xff0c;让计算机从一个函数集合F {f1(x)&…

虚拟机打不开,提示“此主机不支持虚拟化实际模式”的详细解决方法

虚拟机打不开&#xff0c;提示“此主机不支持虚拟机实际模式”的解决方法 一、第一种情况安装/启动虚拟机失败&#xff0c; 在VMWare软件中&#xff0c;安装/启动虚拟机时&#xff0c;如果出以类似以下的错误提示&#xff1a; 出现该提示是由于电脑不支持虚拟化技术或是相关功…

IDEA报错:类文件具有错误的版本 61.0,应为52.0

springboot项目启动报错&#xff1a; 类文件具有错误的版本 61.0,应为52.0 请删除该文件或确保该文件位于正确的类路径子目录中 查阅了网上的很多资料&#xff0c;普遍原因说是springboot版本过高&#xff0c;高于3.0 需要在pom文件中降低版本 也有说是idea的maven配置java版…

网购商城网站

开发工具(eclipse/idea/vscode等)&#xff1a; 数据库(sqlite/mysql/sqlserver等)&#xff1a; 功能模块(请用文字描述&#xff0c;至少200字)&#xff1a;

【Python机器学习】层次聚类AGNES、二分K-Means算法的讲解及实战演示(图文解释 附源码)

需要源码和数据集请点赞关注收藏后评论区留言私信~~~ 层次聚类 在聚类算法中&#xff0c;有一类研究执行过程的算法&#xff0c;它们以其他聚类算法为基础&#xff0c;通过不同的运用方式试图达到提高效率&#xff0c;避免局部最优等目的&#xff0c;这类算法主要有网格聚类和…

easypoi导入excel空指针异常

问题描述 前端页面停留在导入页面&#xff0c;通过后端返回的接口&#xff0c;确认后端已经抛出异常查看系统调用错误日志为 java.lang.NullPointerException: nullat org.apache.poi.xssf.usermodel.XSSFClientAnchor.setCol2(XSSFClientAnchor.java:231)at org.apache.poi.…

基于EKF的四旋翼无人机姿态估计matlab仿真

目录 1.算法描述 2.仿真效果预览 3.MATLAB核心程序 4.完整MATLAB 1.算法描述 卡尔曼滤波是一种高效率的递归滤波器(自回归滤波器)&#xff0c;它能够从一系列的不完全包含噪声的测量中&#xff0c;估计动态系统的状态。这种滤波方法以它的发明者鲁道夫E卡尔曼&#xff08;R…

Android koin

1.源码地址 1.源码地址 2.作用 1.让代码看起来更简洁 现在是这样创建对象的 2.解耦 我们有一个类&#xff0c;然后有100个地方使用它&#xff0c;这个时候如果我们要修改构造参数&#xff0c;加入一个参数&#xff0c;那么我们就要修改100个地方&#xff1b;如果过了一个…

怎样让chatGPT给你打工然后月入过千?

前言 chatGPT最近火出圈了&#xff0c;怎么薅一个文字模型给你打工呢&#xff1f; 这个UP给了个思路&#xff1a;哔哩哔哩 emmm有点尴尬&#xff0c;可能是热度比较高&#xff0c;b站的视频作者自己下架了。 总结一下&#xff1a; 薅的对象百度文库创作中心&#xff1a;地址…

设计模式之装饰器模式

decorator design pattern 装饰模式的概念、装饰模式的结构、装饰模式的优缺点、装饰模式的使用场景、装饰模式与代理模式的区别、装饰模式的实现示例、装饰模式的源码分析 1、装饰模式的概念 装饰模式&#xff0c;即在不改变现有对象结构的前提下&#xff0c;动态的给对象增加…

【云原生】Grafana 介绍与实战操作

文章目录一、概述二、Grafana 安装1&#xff09;下载安装2&#xff09;安装包信息3&#xff09;启动服务4&#xff09;Grafana 访问三、Grafana 功能介绍四、使用mysql存储1&#xff09;安装mysql2&#xff09;修改grafana配置1、创建grafana用户和grafana库2、修改grafana配置…

[附源码]Python计算机毕业设计Django学分制环境下本科生学业预警帮扶系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

Innodb存储引擎-索引和算法(B+树索引、Cardinality、联合索引、覆盖索引、MRR优化、ICP优化、哈希索引、全文索引)

文章目录索引和算法B树索引聚集索引辅助索引B 树索引的分裂B 树索引的管理Cardinality 值B 树索引的使用联合索引覆盖索引优化器选择不使用索引的情况索引提示Multi-Range Read 优化(MRR)Index Condition Pushdown优化(ICP)哈希索引全文索引倒排索引InnoDB全文检索的实现全文检…

第一个MyBatis查询

⭐️前言⭐️ 在连接程序与数据库的工具中&#xff0c;我们之前使用的是JDBC技术&#xff0c;但是JDBC的操作流程极为繁琐&#xff0c;因此才有了更优秀框架——MyBatis&#xff0c;下边我们一起来看这个优秀框架MyBatis的操作与使用。 &#x1f349;欢迎点赞 &#x1f44d; 收…

Innodb存储引擎-锁(数据库锁的查看、快照读当前读、MVCC、自增长与锁、外键与锁、行锁、并发事务的问题、阻塞、死锁、锁升级、锁的实现)

文章目录锁lock 与latch读锁/写锁/意向锁INNODB_TRX/INNODB_LOCKS/INNODB_LOCK_WAITS一致性非锁定读(快照读)一致性锁定读(当前读)MVCC版本链Read View流程自增长与锁外键和锁行锁类型记录锁(record lock)间隙锁(gap lock)下一键锁(next-key lock)并发事务带来的问题阻塞死锁锁…

数据挖掘Java——DBSCAN算法的实现

一、DBSCAN算法的前置知识 DBSCAN算法&#xff1a;如果一个点q的区域内包含多于MinPts个对象&#xff0c;则创建一个q作为核心对象的簇。然后&#xff0c;反复地寻找从这些核心对象直接密度可达的对象&#xff0c;把一些密度可达簇进行合并。当没有新的点可以被添加到任何簇时…

7.加载properties属性文件

一、加载properties属性文件 目的&#xff1a;将数据库的连接参数抽取到一个单独的文件中&#xff0c;与Spring配置文件解耦 1. 编写jdbc.properties属性文件 jdbc.drivercom.mysql.jdbc.Driver jdbc.urljdbc:mysql://127.0.0.1:3306/spring_db jdbc.usernameroot jdbc.passwo…

基于萤火虫算法改进的DELM预测-附代码

萤火虫算法改进的深度极限学习机DELM的回归预测 文章目录萤火虫算法改进的深度极限学习机DELM的回归预测1.ELM原理2.深度极限学习机&#xff08;DELM&#xff09;原理3.萤火虫算法4.萤火虫算法改进DELM5.实验结果6.参考文献7.Matlab代码1.ELM原理 ELM基础原理请参考&#xff1…