当前位置: 首页 > 编程笔记 >

Lua中调用函数使用点号和冒号的区别

空慈
2023-03-14
本文向大家介绍Lua中调用函数使用点号和冒号的区别,包括了Lua中调用函数使用点号和冒号的区别的使用技巧和注意事项,需要的朋友参考一下

本文是面向对象预热篇,讲解函数两种调用方式的区别,初学者比较容易被坑。

1.初学者最易混乱Top1——调用函数时用点号还是用冒号?

我们来看看下面的两句代码:


mSprite.setPosition(100, 20);

mSprite:setPosition(100, 20);

对于初次接触Lua的朋友来说,这简直就是噩梦,为嘛函数的调用有两种形式,是让我们随便挑的意思吗?
 
这两种形式是有区别的,区别很大,但只有一个。
不过,暂时不解释,后面再介绍。

2.最简单的类

我们先来看看简单的,来创建一个“类”试试,如下代码:


    TSprite = {

        x = 0,

        y = 0,

    }

    function TSprite.setPosition(x, y)

        TSprite.x = x;

        TSprite.y = y;

    end

   

    TSprite.setPosition(1, 2);

    print("TSprite坐标(" .. TSprite.x .. "," .. TSprite.y .. ")");

其实就是创建了一个table,给这个table添加一些字段而已。
输出结果如下:


[LUA-print] TSprite坐标(1,2)


大家留意一下setPosition函数,函数里其实也是通过TSprite来调用x和y字段的。
并且,我们使用setPosition的方式是,使用点号,这是正宗的函数调用方式,记住了。

3.不用真实姓名可以吗?——self的作用

如果大家比较敏感的话,就会发现,刚刚的例子很有问题,如果我们这样调用的话:


    local who = TSprite;

    TSprite = nil;

   

    who.setPosition(1, 2);

这么做一定会报错,虽然通过who确实可以成功调用setPosition函数,但函数里需要用到TSprite,而此时的TSprite已经为nil了。
 
于是,聪明的我们可以这么做:


    TSprite = {

        x = 0,

        y = 0,

    }

    function TSprite.setPosition(self, x, y)

        self.x = x;

        self.y = y;

    end

   

    local who = TSprite;

    TSprite = nil;

     

    who.setPosition(who, 1, 2);

    print("TSprite坐标(" .. who.x .. "," .. who.y .. ")");

输出结果仍然是:


[LUA-print] TSprite坐标(1,2)


留意setPosition的第一个参数,我们强制要求传入一个参数,这个参数就是TSprite本身。

于是,在调用setPosition函数时,传入who,who的内容就是TSprite的内容,于是,setPosition就能正常执行了。

4.发挥偷懒的传统美德——默认的self参数,以及默认传递self参数

如果你让一个高(chao)智(ji)商(lan)猿人每次创建函数和调用函数都要这么去处理self,那他一定会说“你过来一下下,我保证不打死你”。

所以,Lua提供了一个新的使用方式,没错,那就是冒号。

看好了,我说的是,用冒号调用函数。
如下代码:


    TSprite = {

        x = 0,

        y = 0,

    }

    function TSprite:setPosition(x, y)

        self.x = x;

        self.y = y;

    end

    local who = TSprite;

    TSprite = nil;

     

    who:setPosition(1, 2);

第一,留意setPosition函数的定义,使用了冒号;
第二,留意setPosition函数的调用,使用了冒号。

冒号的作用就是:定义函数时,给函数的添加隐藏的第一个参数self;调用函数时,默认把当前调用者作为第一个参数传递进去。

使用了冒号之后,就相当于我们刚刚使用点号时一样,只是我们不再需要显式地定义self参数以及主动地传递who参数。
 
好了,这就是点号和冒号的区别了,可以说,冒号是为了给我们偷懒而诞生的。
如果是使用Cocos2d-x lua开发的话,大部分情况下都是使用冒号的。
原因很简单,因为大部分情况下我们都要使用到self参数,就像C++的this关键字一样。

5.结束

下一篇正式进入面向对象的内容,希望大家还没有忘记元表和元方法等基础,面向对象中会用到。

 类似资料:
  • 本文向大家介绍Lua中使用.和:调用函数的区别,包括了Lua中使用.和:调用函数的区别的使用技巧和注意事项,需要的朋友参考一下 tb.print和tb:print的含义略微不同,:调用函数会多传递进去一个self进去,差不多相当于OOP里面的成员函数调用. 但是和OOP里面成员函数/非成员函数不同的是,lua里面的非成员函数.调用,他不是静态的. tb对象有一个print function成员,当

  • 我想知道冒号与Perl 6中的方法和函数调用有什么关系。郑重声明,我使用的是基于MoarVM版本2015.05构建的perl6版本2015.05-55-gd84bbbc。 我刚刚在Perl6规范测试(S32-io)中看到了以下内容(我添加了评论): 据我所知,这相当于: 这两者似乎都需要多个论点并很好地扁平化列表: 这两种不同的语法肯定是有原因的。是否有任何理由使用一种或另一种语法? 我还注意到,

  • 看下面示例代码: local str = "abcde" print("case 1:", str:sub(1, 2)) print("case 2:", str.sub(str, 1, 2)) 执行结果: case 1: ab case 2: ab 冒号操作会带入一个 self 参数,用来代表 自己。而点号操作,只是 内容 的展开。 在函数定义时,使用冒号将默认接收一个 self 参数,而使用点

  • 问题内容: 我碰到了以下内容 在这种情况下,和是什么意思? 问题答案: 它称为条件运算符(是三元运算符)。 它有以下形式:?: 认为“ then”和“ else”。 您的代码等同于

  • 问题内容: 我正在使用itertools.chain以这种方式“拉平”列表列表: 这跟说的有什么不同? 问题答案: 是“ splat”运算符:它接受一个列表作为输入,并将其扩展为函数调用中的实际位置参数。 所以如果是,那就等于说 这与传递just显然不同。对于您的情况,您有一个想要拼合的列表列表;什么确实是在所有你传递给它的位置参数,其中每个位置参数是在自己的权利迭代拼接返回一个迭代。 换句话说,

  • 我有一个类分别组成两个方法静态和非静态,根据我有限的知识提交方法接受可运行的,可调用的实例直接或通过lamba表达式。 今天我知道了,我们甚至可以直接使用双冒号调用或触发静态和非静态方法,这是在Java8中添加的。 我只是想知道这是如何工作的,我的类中没有run方法,它也没有实现runnable,甚至我也没有使用Lamba? 使用::是一个好的实践吗?还是应该传递runnable或callable