【lua实战】lua中pairs和ipairs的区别

2024-08-26 14:44
文章标签 实战 区别 lua pairs ipairs

本文主要是介绍【lua实战】lua中pairs和ipairs的区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

很久以前,我在使用lua的过程中,对于pairs和ipairs的理解还处于表层,认为我了解的就是全部。

ipairs就是对表中元素进行顺序排序,pairs就是对表中元素进行随机排序。

比如如下例子:

local t = {20, "ss", print, 10}
print("------ipairs------")
for k,v in ipairs(t) doprint(k,v)
endprint("------pairs------")
for k,v in  pairs(t) doprint(k,v)
end

如果放以前,我认为的输出结果(错误的结果)会是:

--错误的结果
------ipairs------
1       20
2       ss
3       function: 0x423420
4       10
------pairs------
--输出和上面差不多,只是改变了顺序,比如:
3       function: 0x423420
4       10
2       ss
1       20

但实际上,输出结果(正确的结果)将是:

------ipairs------
1       20
2       ss
3       function: 0x423420
4       10
------pairs------
1       20
2       ss
3       function: 0x423420
4       10

得到这个结果,我对比了一下《Lua程序设计(第4版)》5.4遍历表一节51页的pairs和ipairs的两个例子。原来有一处是我不曾注意的。

用于pairs例子的表t={10, print, x=12,k=”hi”},而用于ipairs例子的表t={10, print, 12, “hi”}。

也就是说,用于pairs的表t,包含10和print这两个正常数组值,以及x=12,k=”hi”两个key-value键值对。

ipairs的表t,将后面的两个key-value键值对改成了数组。

因此,pairs表对key-value的输出结果是1-10,k-hi,2-function:0x420610,x-12。

ipairs表对key-value输出结果为1-10,2-function:0x420610,3-12,4-hi。

而我的例子中,使用的t = {20, “ss”, print, 10},里面都是数组,按照1,2,3,4的顺序,存放20,ss,print,10。

假如我将key-value键值对引入t,比如local t = {20, ss=”ss”, print, b=10},会是什么结果呢?

local t = {20, ss="ss", print, b=10}
print("------ipairs------")
for k,v in ipairs(t) doprint(k,v)
endprint("------pairs------")
for k,v in  pairs(t) doprint(k,v)
end

最终输出结果将是:

------ipairs------
1       20
2       function: 0x423420
------pairs------
1       20
2       function: 0x423420
ss      ss
b       10

这是为什么呢?

原来,表中的元素,分为数组和键值对两种形式。数组就是一个值。键值对,一定会有一个键对应一个值。那么分析一下t表{20, ss=”ss”, print, b=10},为什么ipairs只会输出两行呢?

ipairs只会遍历表中的数组部分,遍历之后,遇到键值对或nil,直接退出。

因此,ipairs会找到t表中的{20,print},先输出。然后因为没其它数组元素了,因此直接退出了。

而pairs无论表t中是什么元素,都会先对数组元素按照顺序排序,然后对键值对元素进行哈希值排序,顺序不固定,对于表t中所有元素都会遍历到。

再看一个有趣的例子,这是我在网上找到的:

local t = {[1]=1,2,[3]=3,4,[5]=5,[6]=6}
print("------ipairs------")
for k,v in ipairs(t1) doprint(k,v)
endprint("------pairs------")
for k,v in  pairs(t1) doprint(k,v)
end

输出会是什么呢?

正确答案是:

------ipairs------
1       2
2       4
3       3
------pairs------
1       2
2       4
3       3
5       5
6       6

ipairs遍历的值为什么是2,4,3呢?

首先,ipairs和pairs会先寻找表中数组元素,也就是{2,4},那么索引1对应2,索引2对应4。

我们再来看看表中的键值对部分。也就是{[1]=1,[3]=3,[5]=5,[6]=6},某种程度上,[1]相当于索引1号的元素,[3]相当于索引为3号的元素,依次类推。但它们又是键值对,和纯粹数组元素还不同。数组的优先级更大,会将相应位置的键值对的值替换掉,因此[1]=2,[2]=4。

然后再看剩下的部分[3]=3,没问题。4号索引呢?表t中没有4号元素的,因此从4号开始,数组序列断开了。ipairs也就不再继续了。而pairs还会继续遍历剩下的键值对,但顺序是无法保证的,这里键5对应5,键6对应6,输出完成。

这里例子中,我再多说一句:

对于ipairs,会从1~n寻找顺序序列数组元素,遇到空就切断不会继续了。因此先找{2,4},再找[3]=3,4号找不到,断开不再继续。

pairs,也会先从1~n寻找顺序序列数组元素,先找{2,4},找完了,剩下的按照哈希算法排序,顺序不固定。不过在这个例子里,总是会输出2,4,3,5,6,让我一度怀疑自己的判断,但当我增加表中元素,可以更方便看出这一特点。

以下结果是把表t改成{[1]=1,2,[3]=3,4,[5]=5,[6]=6,[7]=7,[8]=8,[9]=9,[10]=10}之后的输出结果,可以看到从第8行开始,pairs的顺序就随机了。

------ipairs------
1       2
2       4
3       3
------pairs------
1       2
2       4
7       7
8       8
9       9
3       3
5       5
6       6
10      10

最后划重点:

ipairs:只遍历表中数组元素。遇到不连续的数组元素或nil会直接断开遍历。

pairs:遍历表中所有元素,包括表中数组元素和键值对元素。先根据数组元素从小到大进行排序,然后对键值对元素进行哈希算法后,进行排序。

这篇关于【lua实战】lua中pairs和ipairs的区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1108860

相关文章

MySQL 筛选条件放 ON后 vs 放 WHERE 后的区别解析

《MySQL筛选条件放ON后vs放WHERE后的区别解析》文章解释了在MySQL中,将筛选条件放在ON和WHERE中的区别,文章通过几个场景说明了ON和WHERE的区别,并总结了ON用于关... 今天我们来讲讲数据库筛选条件放 ON 后和放 WHERE 后的区别。ON 决定如何 "连接" 表,WHERE

SpringBoot整合 Quartz实现定时推送实战指南

《SpringBoot整合Quartz实现定时推送实战指南》文章介绍了SpringBoot中使用Quartz动态定时任务和任务持久化实现多条不确定结束时间并提前N分钟推送的方案,本文结合实例代码给大... 目录前言一、Quartz 是什么?1、核心定位:解决什么问题?2、Quartz 核心组件二、使用步骤1

Mybatis的mapper文件中#和$的区别示例解析

《Mybatis的mapper文件中#和$的区别示例解析》MyBatis的mapper文件中,#{}和${}是两种参数占位符,核心差异在于参数解析方式、SQL注入风险、适用场景,以下从底层原理、使用场... 目录MyBATis 中 mapper 文件里 #{} 与 ${} 的核心区别一、核心区别对比表二、底

SpringBoot整合AOP及使用案例实战

《SpringBoot整合AOP及使用案例实战》本文详细介绍了SpringAOP中的切入点表达式,重点讲解了execution表达式的语法和用法,通过案例实战,展示了AOP的基本使用、结合自定义注解以... 目录一、 引入依赖二、切入点表达式详解三、案例实战1. AOP基本使用2. AOP结合自定义注解3.

Spring Boot Interceptor的原理、配置、顺序控制及与Filter的关键区别对比分析

《SpringBootInterceptor的原理、配置、顺序控制及与Filter的关键区别对比分析》本文主要介绍了SpringBoot中的拦截器(Interceptor)及其与过滤器(Filt... 目录前言一、核心功能二、拦截器的实现2.1 定义自定义拦截器2.2 注册拦截器三、多拦截器的执行顺序四、过

C# Semaphore与SemaphoreSlim区别小结

《C#Semaphore与SemaphoreSlim区别小结》本文主要介绍了C#Semaphore与SemaphoreSlim区别小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的... 目录一、核心区别概览二、详细对比说明1.跨进程支持2.异步支持(关键区别!)3.性能差异4.API 差

Java中自旋锁与CAS机制的深层关系与区别

《Java中自旋锁与CAS机制的深层关系与区别》CAS算法即比较并替换,是一种实现并发编程时常用到的算法,Java并发包中的很多类都使用了CAS算法,:本文主要介绍Java中自旋锁与CAS机制深层... 目录1. 引言2. 比较并交换 (Compare-and-Swap, CAS) 核心原理2.1 CAS

Java 队列Queue从原理到实战指南

《Java队列Queue从原理到实战指南》本文介绍了Java中队列(Queue)的底层实现、常见方法及其区别,通过LinkedList和ArrayDeque的实现,以及循环队列的概念,展示了如何高效... 目录一、队列的认识队列的底层与集合框架常见的队列方法插入元素方法对比(add和offer)移除元素方法

Spring Boot基于 JWT 优化 Spring Security 无状态登录实战指南

《SpringBoot基于JWT优化SpringSecurity无状态登录实战指南》本文介绍如何使用JWT优化SpringSecurity实现无状态登录,提高接口安全性,并通过实际操作步骤... 目录Spring Boot 实战:基于 JWT 优化 Spring Security 无状态登录一、先搞懂:为什

C++11中的包装器实战案例

《C++11中的包装器实战案例》本文给大家介绍C++11中的包装器实战案例,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录引言1.std::function1.1.什么是std::function1.2.核心用法1.2.1.包装普通函数1.2.