Lua 之数据结构

数组

通过整数下标访问的table中的元素,即是数组,下标默认从1开始。

一个创建二维数组的例子:

mt = {}
for i = , do
mt[i] = {}
for j = , do
mt[i][j] =
end
end

链表

list = nil
list = {next=list, value="world"}
list = {next=list, value="hello"} local l = list while l do
print(l.value)
l = l.next
end

队列

queue = {}

function queue.new()
return {first=, last=-}
end function queue.push(q, val)
local last = q.last +
q.last = last
q[last] = val
end function queue.pop(q)
local first = q.first if first > q.last then
error("queue is empty")
end local val = q[first]
q[first] = nil
q.first = q.first +
return val
end q = queue.new()
for i=,, do
queue.push(q, i)
end for i=q.first,q.last do
print(queue.pop(q))
end

集合

类似于python的set结构,例如:

function set(list)
local s = {}
for _, n in ipairs(list) do
s[n] = true
end
return s
end reserved = set{"beijing", "xian"}
reserved.beijing -- true
reserved.shanghai -- nil

字符串缓冲

假如需要读取一个文件的内容,常用的方式如下:

local buff = ""

for line in io.lines() do
buff = buff .. line .. "\n"
end print(buff)

假设每行有20 bytes,已读了2500行,那么buff现在就是50000B,当Lua做字符串连接时,就新建一个50020B的新字符串,并从buff中复制了50000B到这个新字符串。这样,对于后面的每一行,Lua都需要移动更多的内存,显然,这样做效率是比较低的。

下面是一个改进的版本,利用table做缓冲,最后利用table.concat将所有行连接起来:

local t = {}

for line in io.lines() do
t[#t+] = line
end local s = table.concat(t, '\n')

那么问题来了,concat的内部工作原理是什么呢,它在连接字符串的时候如何做到高效运行呢?

将每次需要连接的字符串压入堆栈,如果新加入的字符串比栈顶字符串更大,就将两者连接。然后,再将连接后的新字符串与更下面的字符串比较,如果是新建字符串更长的话,则再次连接它们。这样的连接一直向下延续应用,直到遇到一个更大的字符串或者达到了栈底。

function NewStack()
return {""}
end function push(stack, s)
table.insert(stack, s) for i = table.getn(stack) - , , - do
if string.len(stack[i]) > string.len(stack[i+]) then
break
end stack[i] = stack[i] .. table.remove(stack)
end
end local s = NewStack()
for line in io.lines() do
push(s, line .. "\n")
end

下面是一个广度优先遍历有向图的例子,其中raw描述了图的所有边,每个边有两个点——起点、终点。

用了一个table来表示点,它有两个字段,name和adj,name表示该点的名称,adj表示它的终点。

raw = {
{'A','B'},
{'A','F'},
{'B','C'},
{'B','I'},
{'B','G'},
{'F','G'},
{'F','E'},
{'C','I'},
{'C','D'},
{'I','D'},
{'G','H'},
{'G','D'},
{'H','E'},
{'H','D'},
{'E','D'},
} local function name2node(graph, name)
if not graph[name] then
graph[name] = {name=name, adj={}}
end
return graph[name]
end local function readgraph()
local graph = {}
for _,v in pairs(raw) do
local namefrom, nameto = v[], v[] local from = name2node(graph, namefrom)
local to = name2node(graph, nameto)
from.adj[to] = true
end
return graph
end function findpath(curr, to, path, visited)
path = path or {}
visited = visited or {} if visited[curr] then -- no path
return nil
end visited[curr] = true
path[#path+] = curr if curr == to then -- be found
return path
end for node in pairs(curr.adj) do -- recursive
local p = findpath(node, to, path, visited)
if p then return p end
end path[#path] = nil
end function printpath(path)
for i=, #path do
print(path[i].name)
end
end g = readgraph()
a = name2node(g, 'A')
b = name2node(g, 'D')
p = findpath(a, b)
if p then printpath(p) end

序列化

打印一个lua对象:

function serialize(o)
if type(o) == "number" then
io.write(o)
elseif type(o) == "string" then
io.write(string.format("%q", o))
elseif type(o) == "table" then
io.write("{\n")
for k,v in pairs(o) do
io.write(" ",k," = ")
serialize(v)
io.write(",\n")
end
io.write("}\n")
else
error("cannot serialize a " .. type(o))
end
end local l = {name='cq', , like={'movie', 'music'}} serialize(l)

输出内容:

{
= ,
name = "cq",
like = {
= "movie",
= "music",
}
,
}

Lua 之数据结构的更多相关文章

  1. Cocos2d-x 脚本语言Lua基本数据结构-表(table)

    Cocos2d-x 脚本语言Lua基本数据结构-表(table) table是Lua中唯一的数据结构.其它语言所提供的数据结构,如:arrays.records.lists.queues.sets等. ...

  2. cocos2dx解析lua table数据结构 简易版.

    之前一直用xml填配置, cocos2dx自带了xml解析接口, 非常方便. 但是, 接口好用也改变不了xml的结构字符太多, 书写麻烦, 乱七八糟的事实. 很早就想换lua, 无奈引擎没有现成接口, ...

  3. Lua 的数据结构

    1. Arrays: 注意 #(data), # 加上 table名字 == size of data = {}; , do --行 , do --列 data[(y-)*+x] = (y-)*+x; ...

  4. 项目用到了lua的哪些部分

    昨天有位同事跟我说,我们的手游客户端(cocos2d-x lua binding)代码没有看到lua的特殊技巧,用起来跟其他语言差不多.<Programming in lua>毕竟有将近4 ...

  5. [转]编写高性能的Lua代码

    昨天晚上闲来无事,看室友在电脑上挂机玩游戏,用的一个辅助脚本,以为是lua写的脚本在跑,实际调查发现是按键精灵的脚本. 于是在网上找相关Lua开发游戏脚本的案例,看到一个人的博客,内容很不错,学到了很 ...

  6. Lua学习----零碎知识点

    Jit(just in time) 动态即时编译,边运行时边编译---->lua (主要是面向进程) Aot(ahead of time) 静态提前编译,运行前编译---->C#(主要是面 ...

  7. Lua设计与实现--读书笔记

    目录 lua简介 一种通用的数据类型:lua_TValue 字符串 Table lua实现一个队列 lua简介 C++底层核心模块,暴露核心接口给lua脚本层,网络的收发都在c++层完成,本书简述lu ...

  8. 让你的Python代码更加pythonic

    http://wuzhiwei.net/be_pythonic/ 何为pythonic? pythonic如果翻译成中文的话就是很python.很+名词结构的用法在中国不少,比如:很娘,很国足,很CC ...

  9. lua解析脚本过程中的关键数据结构介绍

    在这一篇文章中我先来介绍一下lua解析一个脚本文件时要用到的一些关键的数据结构,为将来的一系列代码分析打下一个良好的基础.在整个过程中,比较重要的几个源码文件分别是:llex.h,lparse.h.l ...

随机推荐

  1. ubuntu15.10下搭建cordova+ionic开发环境

    安装jdk 在命令下输入java如果没有安装会提示该命令包含于openjdk软件包 sudo apt-get install openjdk然后按下tab会列出openjdk开头的软件包 我这里就选择 ...

  2. html页面中meta的作用

    meta是用来在HTML文档中模拟HTTP协议的响应头报文.meta 标签用于网页的<head>与</head>中,meta 标签的用处很多.meta 的属性有两种:name和 ...

  3. [学习笔记]lca-倍增

    The article is to Jimmy.(方老师讲过了还不会,想怎样???) 求一棵树上两个节点的最近公共祖先有两种算法: (离线); 倍增(在线). 这篇博客只介绍倍增的写法. 表示节点的祖 ...

  4. CruiseControl.NET/CCNET配置(SVN+MSBuild+BAT+FTP)

    CCNET目前最新版本为1.8.5,官方很久没更新过了,如果投入生成环境使用,建议全部转到Jenkins上. 可以直接在这里下载:http://www.cnblogs.com/EasonJim/p/5 ...

  5. Android Material Design 学习笔记 - Matrial Theme

    google在2014年 I/O大会上推出了一种新的设计设计语言—Material design,这种设计语言语言旨在为手机.平板电脑.台式机和“其他平台”提供更一致.更广泛的“外观和感觉”(附上官方 ...

  6. nginx中SSI问题的研究

    最近感觉挺爽的,这个项目团队没有一个专门做PHP的,我是第一个进来做PHP(当然还有前端)的,哈哈,我会设计修改出适合我们业务的PHP框架,哈哈,感觉会学到很多东西的样子,前几天在组内20几个前辈面前 ...

  7. Java反射API使用实例

    /**     * 访问Class对应的类所包含的注释:getAnnotation();getDelaredAnnotation();     * 访问Class对应的类所包含的内部类:getDecl ...

  8. JavaScript制作时钟特效

    需求说明:制作显示年.月.日.星期几并且显示上午(AM)和下午(PM)的 12进制的时钟,具体效果如下所示: 代码如下: <!DOCTYPE HTML PUBLIC "-//W3C// ...

  9. Distance Between Points

    I need some help. I have to create a function that will calculate the distance between points (x1,y1 ...

  10. nginx跨域设置

    nginx跨域问题例子:访问http://10.0.0.10/ 需要能实现跨域 操作:http://10.0.0.10/项目是部署在tomcat里面,tomcat跨域暂时还不会,按照网上的方法操作也没 ...