io操作:

  io.input(filename):指定一个输入流,可以是标准输入stdin,也可以是一个文件路径,返回一个文件句柄;

  io.output(filename):指定一个输出流,可以是标准输出stdout,也可以是一个文件路径,返回一个文件句柄;

  io.lines():迭代器,从指定的输入流中读取一行;

  io.read():从指定的输入流中读取数据,如果不带参数默认读取一行,

    参数是"*all"则一次读取整个文件数据;

    参数是"*line",读取一行;

    参数是"*number",读取一个数字;

    参数是number,读取不超过number个字节的数据;

  io.write(buffer):将数据写入到指定的输出流中;

  io.open(filename, mode):打开一个文件,如果成功返回一个文件对象,如果失败返回nil;

    local file = io.open("1.txt", "rw")

    file:read()

    file:write("123")

    file:close()

    file:seek(where, offset): where的取值: "cur"相对于当前位置进行偏移,"end"相对于结束位置进行偏移,"set"相对于文件开始位置进行偏移。返回值都是当前位置。

  io.close(file):关闭一个打开的文件;

  实例:

  1. 使用io.lines和table.concat,concat比直接使用..连接符号的效率要高很多,特别针对大字符串:  

local input = io.input("1.txt")
local output = io.output("2.txt") t = {}
for line in io.lines() do
t[#t + ] = line
end
local buffer = table.concat(t, "\n")
io.write(buffer) io.close(input)
io.close(output)

  2. 直接使用read,一次读取所有数据:

local input = io.input("1.txt")
local output = io.output("2.txt") local buffer = io.read("*all")
io.write(buffer) io.close(input)
io.close(output)

数据文件:

  lua作为脚步语言的一大特点就是可以将数据存在脚本文件中,然后通过执行“脚本数据文件”,就可以实现导入数据的功能。而且lua的设计的主要用途之一就是针对数据进行描述,在这方面lua进行了大量的优化,因此使用lua脚本来描述数据,比通过文件io操作更加高效:

  实例:

  data.lua,数据描述脚本文件:

load_data({"xiaoming",
"",
""
}) load_data({"xiaoqiang",
"",
""
})

  data_load.lua,数据导入脚本文件,通过定义一个导入方法,就可以将数据导入到当前的运行环境中:

t = {}
function load_data(data)
t[#t + ] = data
end local f = assert(loadfile("./data.lua"));
f() for i, data in ipairs(t) do
print(i, data[])
end

序列化:

  将一个table中的数据,序列化成json数据:

t = { , , , [] = "df", ["s"] = "", k = "", f = {"a", "b", "c"}, j = {{"df", "fd", "d", x = , y = }, {, ,}} }

function serialize(tbl)
local buffer_list = {}
local arr_list = {}
local value_list = {}
local value_string = "" local quot = "\""
local colon = ":"
local comma = ","
local lbrace = "{" rbrace = "}"
local lsbracket = "[" rsbracket = "]" local hasArr = false
local hasValue = false buffer_list[#buffer_list + ] = "{"
repeat
for key, value in ipairs(tbl) do
if (not hasArr) then
hasArr = true
end local typestr = type(value)
if typestr == "string" then
arr_list[#arr_list + ] = quot..value..quot
elseif typestr == "table" then
local subbuf = serialize(value)
arr_list[#arr_list + ] = subbuf
else
arr_list[#arr_list + ] = value
end
end
until true
if (hasArr) then
local keystr = "\"arr\""
local arrstr = table.concat(arr_list, comma); value_list[#value_list + ] = keystr..":["..arrstr.."]"
end
repeat
for key, value in pairs(tbl) do
repeat
local keytype = type(key)
if keytype == "number" then
break
end if not hasValue then
hasValue = true
end local keystr = "\""..key.."\":"
local typestr = type(value)
if typestr == "string" then
value_list[#value_list + ] = keystr.."\""..value.."\""
elseif typestr == "table" then
local subbuf = serialize(value)
value_list[#value_list + ] = keystr..subbuf
else
value_list[#value_list + ] = keystr..value
end
until true
end
until true
if (hasArr or hasValue) then
value_string = table.concat(value_list, ",") buffer_list[#buffer_list + ] = value_string
end
buffer_list[#buffer_list + ] = "}" local buffer = table.concat(buffer_list)
print(buffer) return buffer
end serialize(t)

  打印结果:

{
"arr": [
1,
2,
3
],
"s": "212",
"j": {
"arr": [
{
"arr": [
"df",
"fd",
"d"
],
"y": 2,
"x": 1
},
{
"arr": [
1,
2,
3
]
}
]
},
"k": "212",
"f": {
"arr": [
"a",
"b",
"c"
]
}
}

元表:

  元表本身是一种table,它可以为其他table定义了一套预定义的操作集合,这类操作集合称为“元方法”。

  算术类的元方法:

    _add:加法;__mul:乘法;__sub:减法;__div:除法;_unm:相反数;__mod:取模;__pow:幂;__concat:连接符;

  关系符元方法:

    __eq:等于;__le:小于等于;__lt:小于;

    等于操作符必须是两个对象拥有相同的元方法时候才会被调用,否则会返回false;

  __index:当在一个对象中无法找到一个key字段时,会返回nil。但如果为其设置了元表,并定义__index元方法,就会返回调用__index的返回值。__index元方法通常是一个函数,但也可以是一个table,如果是一个table,就对该table应用相同的方式进行查找。先检查table中是否存在该字段,如果存在返回该字段的值,如果没有找到,检查table的元表中的__index元方法,以此类推;

  __newindex:当对对象中的某个字段进行赋值操作时,lua会去搜索元表中是否定义了__newindex元方法,如果存在就调用元方法,反之就会进行一次普通的赋值操作。如果_newindex元方法指向的是一个table,那么相同的操作会应用在table上;

  lua还提供了两个可以避开__index和__newindex元方法的函数:rawset(t, key, value)和rawget(t, key);

  实例:

local mt = {}
local Point = {} mt.default = { x = , y = , r = , g = , b = , a = }
mt.__add = function (t1, t2)
return Point.new(t1.x + t2.x, t1.y + t2.y)
end
mt.__concat = function (t1, t2)
local x = t1.x + t2.x
local y = t1.y + t2.y
return "x:"..x.." y:"..y
end
mt.__le = function (t1, t2)
return t1.x <= t2.x and t1.y <= t2.y
end
mt.__eq = function (t1, t2)
return t1.x == t2.x and t1.y == t2.y
end
mt.__tostring = function (t)
return "Point x:"..t.x..",y:"..t.y
end
mt.__index = mt.default // 直接用table来索引,性能更好
-- function (t, k)
-- return mt.default[k]
-- end
mt.__newindex = --mt.default // 用table作为元方法,会把新的字段赋值给table
function(t, k, v)
rawset(t, k, v) // 跳过元方法,否则会堆栈溢出
end
mt.__metatable = "Point metatable" Point.new = function (x, y)
assert(type(x) == "number" and type(y) == "number", "Point.new invaild arguments") local pt = {}
setmetatable(pt, mt)
pt.x = x
pt.y = y
return pt;
end local p1 = Point.new(, )
local p2 = Point.new(, )
local p3 = p1 + p2 // 调用__add元方法
local str = p1..p2 // 调用__concat元方法 print(p3) // Point x:7,y:46
print(p3.x, p3.y) // 7 46
print(str) // x:7 y:46
print(p1 == p2) // false 调用__eq元方法
print(p3.r) // 255 调用__index元方法,从mt.default表中去查找
print(rawget(p3, "r")) // nil 由于避开了__index元方法,所以返回nil
p3.c = 123 // 调用__newindex元方法进行赋值
print(p3.c) // 123
print(p1.c) // nil 因为c字段是赋值给p3的,p1无法访问
print(rawget(p3, "c")) // 123 说明c是直接复制给p3的,没有放在元表上
print(mt.default["c"]) // nil 说明c是直接复制给p3的,没有放在元表上

  table的代理:

  使用代理的table,可以跟踪对原始table的访问,并可以限制对原始table的控制。

  

local _t = {x = , y = , z = }

local mt = {
__index = function (t, k)
print("proxy __index", t, k)
return t._t[k];
end, __newindex = function (t, k, v)
print("proxy __newindex", t, k, v)
t._t[k] = v
end
} function create_proxy (tbl)
local proxy = {}
proxy._t = tbl;
setmetatable(proxy, mt)
return proxy;
end proxy_t = create_proxy(_t); proxy_t.z =
proxy_t.z1 =
print(proxy_t.x, proxy_t.y, proxy_t.z, proxy_t.z1)

  打印结果:

proxy __newindex    table: 0x7fb1d2408e10    z
proxy __newindex table: 0x7fb1d2408e10 z1
proxy __index table: 0x7fb1d2408e10 x
proxy __index table: 0x7fb1d2408e10 y
proxy __index table: 0x7fb1d2408e10 z
proxy __index table: 0x7fb1d2408e10 z1

模块:

  使用require来导入一个模块,可以是lua模块,也可以是c模块。

  require的搜索策略:替换环境变量配置中的?(unix通常是:./?.lua;/usr/local/lua/?.lua;/usr/local/lua/?/init.lua),如果模块名中含有.,将.替换成目录分隔符(unix是/,windows是\)。例如 require "a.b" 会搜索如下路径:

lua: /Users/avl-showell/Desktop/lua-5.3./test/test1.lua:: module 'a.b' not found:
no field package.preload['a.b']
no file '/usr/local/share/lua/5.2/a/b.lua'
no file '/usr/local/share/lua/5.2/a/b/init.lua'
no file '/usr/local/lib/lua/5.2/a/b.lua'
no file '/usr/local/lib/lua/5.2/a/b/init.lua'
no file './a/b.lua'
no file '/usr/local/lib/lua/5.2/a/b.so'
no file '/usr/local/lib/lua/5.2/loadall.so'
no file './a/b.so'
no file '/usr/local/lib/lua/5.2/a.so'
no file '/usr/local/lib/lua/5.2/loadall.so'
no file './a.so'
function require (name)
if not package.loaded[name] then
local loader = findloader(name)
if loader == nil then
error("unable to load module "..name)
end
package.loaded[name] = true
local res = loader(name)
if res ~= nil then
package.loaded[name] = res
end
end
return package.loaded[name]
end

  模块创建方式:

local M = {}
......
_G[modulename] = M;
package.loaded[modulename] = M
return M

lua 高级的更多相关文章

  1. Lua高级教程Metatables

    什么是Metatable metatable是Lua中的重要概念,每一个table都可以加上metatable,以改变相应的table的行为. Metatables举例 -- 声明一个正常的关系变量 ...

  2. [Lua]Lua高级教程Metatables

    什么是Metatable metatable是Lua中的重要概念,每一个table都可以加上metatable,以改变相应的table的行为. Metatables举例 -- 声明一个正常的关系变量 ...

  3. (原创)cocos lua 热更新从零开始(一)最简单demo

    开发环境:WIN7 + cocos2dx 3.10 lua版本 0.学习这篇内容的基础是你要会创建并运行一个cocos lua项目 1.热更新的思想所谓的热更新,就是在线更新代码和资源.热更新的过程首 ...

  4. java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱

    java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱 redis数据库 Redis企业集群高级应用精品教程[图灵学院] Redis权威指南 利用redis + lua解决抢红包高并 ...

  5. class() 高级用法 -- lua

    class() 高级用法 class() 除了定义纯 Lua 类之外,还可以从 C++ 对象继承类. 比如需要创建一个工具栏,并在添加按钮时自动排列已有的按钮,那么我们可以使用如下的代码: -- 从 ...

  6. Lua学习高级篇

    Lua学习高级篇 之前已经说了很多,我目前的观点还是那样,在嵌入式脚本中,Lua是最优秀.最高效的,如果您有不同的观点,欢迎指正并讨论,切勿吐槽.这个系列完全来自于<Programming in ...

  7. lua中文教程【高级知识】

    一.编译和运行和调试 1.lua和其他解释型语言一样,先转换成为中间码再执行 2.dofile和loadfile的区别:loadfile编译返回不执行,返回错误代码:dofile执行,返回错误信息 3 ...

  8. lua 脚本之高级篇 (面向对象的完全支持),有性能问题。

    ---------------------------------------------------------- --面向对象核心库 ------------------------------- ...

  9. c++调用lua

    我们主要解决如下几个问题: 转:http://www.cnblogs.com/zisou/p/cocos2dx-lua2.html http://www.cnblogs.com/sevenyuan/p ...

随机推荐

  1. Python学习笔记-字典

    字典是python中唯一内建的映射类型. 创建字典phonebook = {'Alice':'2341','Beth':'9102'} 可以使用dict通过其他映射或者键值对的序列建立字典.关键值参数 ...

  2. ubuntu fix the grub boot(need Internet)

    sudo add-apt-repository ppa:yannubuntu/boot-repair sudo apt-get update sudo apt-get install -y boot- ...

  3. Python开发入门与实战4-模板页面

    4.Django基于模板页面 在前一章中,HTML是直接被硬编码在 Python views.py代码中,如下: from django.http import HttpResponse import ...

  4. windows 版的julia repl 启动时间已经大大优化!

    julia 是一门语法类似python 偏向主要用于科学计算的语言,julia吸收了很多其它语言的优点,内置了大量函数,使用起来很方便. 之前windows下的 julia repl(交互解释器)启动 ...

  5. Ubuntu中安装eclipse ,双击eclipse出现invalid configuration location问题

    ubuntu invalid configuration location   标签: myeclipse for ubuntu   ubuntu myeclipse   ubuntu安装myecli ...

  6. 原生javascript加载运行

    原生javascript加载运行 (function(){ //TODO sometings }()); 在要运行相应代码的位置加入script标签,创建函数并自执行; 关于window.onload ...

  7. 实时刷新Winform中Label的Text

    最直白的例子: private void btnStart_Click(object sender, EventArgs e) { ; ) { labelTime.Text = i.ToString( ...

  8. 数据库最大连接池max pool size

    本文导读:Max Pool Size如果未设置则默认为100,理论最大值为32767.最大连接数是连接池能申请的最大连接数,如果数据库连接请求超过此数,后面的数据库连接请求将被加入到等待队列中,这会影 ...

  9. What does "Rxlch" mean in ENCODE?

    In ENCODE project, we could see some files are called "...rxlch...", which means "rev ...

  10. SVD

    SVD分解(奇异值分解),本应是本科生就掌握的方法,然而却经常被忽视.实际上,SVD分解不但很直观,而且极其有用.SVD分解提供了一种方法将一个矩阵拆分成简单的,并且有意义的几块.它的几何解释可以看做 ...