Lua学习笔记(五):函数

  函数有两种用途:

  1).完成指定的任务,这种情况下函数作为调用语句使用;

  2).计算并返回值,这种情况下函数作为赋值语句表达式使用。

 1 function func_name (arguments-list)
 2     statements-list;
 3 end;
 4
 5 --调用函数的时候,如果参数列表为空,必须使用()表明是函数调用
 6 print(8*9, 9/8)
 7 a = math.sin(3) + math.cos(10)
 8 print(os.date())
 9
10 --上述规则有一个例外,当函数只有一个参数并且这个参数是字符串或者表构造的时候,()可有可无
11 print "Hello World"        <-->        print("Hello World")
12 dofile ‘a.lua‘            <-->        dofile (‘a.lua‘)
13 print [[a multi-line        <-->        print([[a multi-line
14             message]]                                message]])
15 f{x=10, y=20}                <-->        f({x=10, y=20})
16 type{}                        <-->        type({})

  Lua也提供了面向对象方式调用函数的语法,比如:o:foo(x)与o.foo(o,x)是等价的。

  Lua使用的函数,既可是Lua编写的,也可以是其他语言编写的,对于Lua程序员,用什么语言实现的函数使用起来都一样。

  Lua函数实参和形参的匹配与赋值语句类似,多余部分被忽略,缺少部分用nil补足。

1 function f(a, b) return a or b end
2
3 CALL                PARAMETERS
4
5 f(3)                a=3, b=nil
6 f(3, 4)            a=3, b=4
7 f(3, 4, 5)            a=3, b=4    (5 is discarded)

  1、多返回值

  Lua函数可以返回多个结果值,比如:string.find,其返回匹配串“开始和结束的下标”(如果不存在匹配串返回nil)。

 1 s, e = string.find("hello Lua users", "Lua")
 2 print(s, e)        --> 7    9
 3
 4 --Lua函数中,return后列出要返回的值得列表即可返回多值,如:
 5 function maximum (a)
 6     local mi = 1                -- maximum index
 7     local m = a[mi]            -- maximum value
 8     for i,val in ipairs(a) do
 9         if val > m then
10             mi = i
11             m = val
12         end
13     end
14     return m, mi
15 end
16
17 print(maximum({8,10,23,12,5}))        --> 23   3

  Lua总是调整函数返回值的个数以适用调用环境,当作为独立的语句调用函数时,所有返回值将被忽略。假设有如下三个函数:

1 function foo0 () end                        -- returns no results
2 function foo1 () return ‘a‘ end            -- returns 1 result
3 function foo2 () return ‘a‘,‘b‘ end    -- returns 2 results

  第一,当作为表达式调用函数时,有以下几种情况:

  1).当调用作为表达式最后一个参数或者仅有一个参数时,根据变量个数函数尽可能多地返回多个值,不足补nil,超出舍去。

  2).其他情况下,函数调用仅返回第一个值(如果没有返回值为nil)

 1 x,y = foo2()                -- x=‘a‘, y=‘b‘
 2 x = foo2()                    -- x=‘a‘, ‘b‘ is discarded
 3 x,y,z = 10,foo2()            -- x=10, y=‘a‘, z=‘b‘
 4
 5 x,y = foo0()                -- x=nil, y=nil
 6 x,y = foo1()                -- x=‘a‘, y=nil
 7 x,y,z = foo2()            -- x=‘a‘, y=‘b‘, z=nil
 8
 9 x,y = foo2(), 20            -- x=‘a‘, y=20
10 x,y = foo0(), 20, 30        -- x=‘nil‘, y=20, 30 is discarded

  第二,函数调用作为函数参数被调用时,和多值赋值是相同。

1 print(foo0())                -->
2 print(foo1())                --> a
3 print(foo2())                --> a   b
4 print(foo2(), 1)            --> a   1
5 print(foo2() .. "x")        --> ax

  第三,函数调用在表构造函数中初始化时,和多值赋值时相同。

1 a = {foo0()}                -- a = {}    (an empty table)
2 a = {foo1()}                -- a = {‘a‘}
3 a = {foo2()}                -- a = {‘a‘, ‘b‘}
4
5 a = {foo0(), foo2(), 4}    -- a[1] = nil, a[2] = ‘a‘, a[3] = 4

  另外,return f()这种形式,则返回"f()"的返回值:

 1 function foo (i)
 2     if i == 0 then return foo0()
 3     elseif i == 1 then return foo1()
 4     elseif i == 2 then return foo2()
 5     end
 6 end
 7
 8 print(foo(1))            --> a
 9 print(foo(2))            --> a  b
10 print(foo(0))            -- (no results)
11 print(foo(3))            -- (no results)

  可以使用圆括号强制使调用返回一个值。

1 print((foo0()))        --> nil
2 print((foo1()))        --> a
3 print((foo2()))        --> a

  一个return语句如果使用圆括号将返回值括起来也将导致返回一个值。

  函数多值返回的特殊函数unpack,接受一个数组作为输入参数,返回数组的所有元素。unpack被用来实现范型调用机制,在C语言中可以使用函数指针调用可变的函数,可以声明参数可变的函数,但不能两者同时可变。在Lua中如果你想调用可变参数的可变函数只需要这样:

1 f(unpack(a))

  unpack返回a所有的元素作为f()的参数

1 f = string.find
2 a = {"hello", "ll"}
3 print(f(unpack(a)))        --> 3  4

  预定义的unpack函数是用C语言实现的,我们也可以用Lua来完成:

1 function unpack(t, i)
2     i = i or 1
3     if t[i] then
4         return t[i], unpack(t, i + 1)
5     end
6 end

  2、可变参数

  Lua函数可以接受可变数目的参数,和C语言类似在函数参数列表中使用三点(...)表示函数有可变的参数。Lua将函数的参数放在一个叫arg的表中,除了参数以外,arg表中还有一个域n表示参数的个数。

  例如,我们可以重写print函数:

1 printResult = ""
2
3 function print(...)
4     for i,v in ipairs(arg) do
5         printResult = printResult .. tostring(v) .. "\t"
6     end
7     printResult = printResult .. "\n"
8 end

  有时候我们可能需要几个固定参数加上可变参数

1 function g (a, b, ...) end
2
3 CALL                PARAMETERS
4
5 g(3)                a=3, b=nil, arg={n=0}
6 g(3, 4)            a=3, b=4, arg={n=0}
7 g(3, 4, 5, 8)        a=3, b=4, arg={5, 8; n=2}

  如上面所示,Lua会将前面的实参传给函数的固定参数,后面的实参放在arg表中。

  举个具体的例子,如果我们只想要string.find返回的第二个值。一个典型的方法是使用哑元(dummy variable, 下划线):

 1 local _, x = string.find(s, p)
 2 -- now use `x‘
 3 ...
 4
 5 --还可以利用可变参数声明一个select函数
 6 function select (n, ...)
 7     return arg[n]
 8 end
 9
10 print(string.find("hello hello", " hel"))    --> 6  9
11 print(select(1, string.find("hello hello", " hel"))) --> 6
12 print(select(2, string.find("hello hello", " hel"))) --> 9

  有时候需要将函数的可变参数传递给另外的函数调用,可以使用前面我们说过的unpack(arg)返回arg表所有的可变参数,Lua提供了一个文本格式化的函数string.format:

1 function fwrite(fmt, ...)
2     return io.write(string.format(fmt, unpack(arg)))
3 end
4
5 --这个例子将文本格式化操作和写操作组合为一个函数

  3、命名参数

  Lua的函数参数是和位置相关的,调用时实参会按顺序依次传给形参。有时候用名字指定参数是很有用的,比如rename函数用来给一个文件重命名,有时候我们记不清命名前后两个参数的顺序了:

1 -- invalid code
2 rename(old="temp.lua", new="temp1.lua")

  上面这段代码是无效的,Lua可以通过将所有的参数放在一个表中,把表作为函数的唯一参数来实现上面这段伪代码的功能。因为Lua语法支持函数调用时实参可以是表的构造。

1 rename{old="temp.lua", new="temp1.lua"}

  根据这个想法我们重定义了rename:

1 function rename (arg)
2     return os.rename(arg.old, arg.new)
3 end

  当函数的参数很多的时候,这种函数参数的传递方式很方便的。例如GUI库中创建窗体的函数有很多参数并且大部分参数是可选的,可以用下面这种方式:

 1 w = Window {
 2     x=0, y=0, width=300, height=200,
 3     title = "Lua", background="blue",
 4     border = true
 5 }
 6
 7 function Window (options)
 8     -- check mandatory options
 9     if type(options.title) ~= "string" then
10         error("no title")
11     elseif type(options.width) ~= "number" then
12         error("no width")
13     elseif type(options.height) ~= "number" then
14         error("no height")
15     end
16
17     -- everything else is optional
18     _Window(options.title,
19         options.x or 0,            -- default value
20         options.y or 0,            -- default value
21         options.width, options.height,
22         options.background or "white",    -- default
23         options.border            -- default is false (nil)
24     )
25 end

Lua学习笔记(五):函数

时间: 2024-10-11 07:44:41

Lua学习笔记(五):函数的相关文章

lua学习笔记之函数

Lua学习笔记之函数 1.  函数的作用 函数主要完成指定的任务,这样的情况下函数作为调用语句使用,函数可以计算并返回值,这样的情况下函数作为赋值语句的表达式使用. 语法: funcationfunc_name(arguments-list) Statements-list end 调用函数的时候,如果参数列表为空,必须使用()表示是函数调用. Print(8*9,9/8) a = math.sin(3) +math.cos(10) print(os.date()) 上述规则有一个例外,当函数只

Lua学习笔记3. 函数可变参数和运算符、转义字符串、数组

1. Lua函数可以接受变长数目的参数,和C语言类似,在函数的参数列表中使用(...)表示函数可以接受变长参数 lua函数将参数存放在一个table中,例如arg,那么#arg可以获得参数的个数 function func_no_p(...) local arg={...} for k,v in pairs(arg} do print(v ..",") end print("输入参数个数:".. #arg) end func_no_p(1,2,34,1,"

Lua学习笔记(六):函数-续

Lua中的函数是带有词法定界(lexical scoping)的第一类值(first-class values).第一类值指:在Lua中函数和其他值(数值.字符串)一样,函数可以被存放在变量中,也可以存放在表中,可以作为函数的参数,还可以作为函数的返回值.词法定界指:嵌套的函数可以访问他外部函数中的变量.这一特性给Lua提供了强大的编程能力. Lua中关于函数稍微难以理解的是函数也可以没有名字,匿名的.当我们提到函数名(比如print),实际上是说一个指向函数的变量,像持有其他类型的变量一样:

Lua学习笔记8:文件读写

lua中文件读写经常在游戏配置中用到,比如客户端的音效音乐开关等. Lua官方API文档:点这里 I/O库为文件操作提供4个主要函数:io.open(),io.read(),io.write和io.close(). io.open(文件路径,打开方式):以指定方式打开一个文件,打开成功返回一个文件句柄,失败返回nil和错误描述. 可以传入以下六种打开方式: "r":读模式(默认): "w":写模式: "a":附加模式: "r+"

Caliburn.Micro学习笔记(五)----协同IResult

Caliburn.Micro学习笔记(五)----协同IResult 今天说一下协同IResult 看一下IResult接口 /// <summary> /// Allows custom code to execute after the return of a action. /// </summary> public interface IResult { /// <summary> /// Executes the result using the specif

Lua学习笔记(七):迭代器与泛型for

1.迭代器与闭包 迭代器是一种支持指针类型的结构,它可以遍历集合的每一个元素.在Lua中我们常常使用函数来描述迭代器,每次调用该函数就返回集合的下一个元素. 迭代器需要保留上一次成功调用的状态和下一次成功调用的状态,也就是他知道来自于哪里和将要前往哪里.闭包提供的机制可以很容易实现这个任务.记住:闭包是一个内部函数,它可以访问一个或者多个外部函数的外部局部变量.每次闭包的成功调用后这些外部局部变量都保存他们的值(状态).当然如果要创建一个闭包必须要创建其外部局部变量.所以一个典型的闭包的结构包含

angular学习笔记(五)-阶乘计算实例(1)

<!DOCTYPE html> <html ng-app> <head> <title>2.3.2计算阶乘实例1</title> <meta charset="utf-8"> <script src="../angular.js"></script> <script src="script.js"></script> </

Linux System Programming 学习笔记(五) 进程管理

1. 进程是unix系统中两个最重要的基础抽象之一(另一个是文件) A process is a running program A thread is the unit of activity inside of a process the virtualization of memory is associated with the process, the threads all share the same memory address space 2. pid The idle pro

java之jvm学习笔记五(实践写自己的类装载器)

java之jvm学习笔记五(实践写自己的类装载器) 课程源码:http://download.csdn.net/detail/yfqnihao/4866501 前面第三和第四节我们一直在强调一句话,类装载器和安全管理器是可以被动态扩展的,或者说,他们是可以由用户自己定制的,今天我们就是动手试试,怎么做这部分的实践,当然,在阅读本篇之前,至少要阅读过笔记三. 下面我们先来动态扩展一个类装载器,当然这只是一个比较小的demo,旨在让大家有个比较形象的概念. 第一步,首先定义自己的类装载器,从Clas