LPC-tutorial

LPC语言基本变量类型

2019-04-25  本文已影响0人  秋林拾叶

本文节选自本人写作的《 LPC 语言基础教程:从零学习游戏开发》第二章第一节。

第一节 LPC语言基本变量类型

LPC语言的数据类型基本和C语言类似,但是和C语言相同和相似的数据类型只有:整型(int)、浮点型(float)、字符串(string)、数组(array)、结构体(class)。

1. 有符号整型(int)

在C语言中的整数分 short、int、long,还有无符号数 unsigned,但在LPC语言中,只有 int 类型

在 fluffos 驱动中,int 的长度是8个字节,长度和C语言64 位 Linux 及 Mac OS 下的长整型一样,取值范围是-263 ~ 263-1,即-9223372036854775808~9223372036854775807。

在C语言中,如果数值超出取值范围会超成数据溢出,如下C语言代码,在64位 Linux 系统下,因为溢出,输出结果变成:-9223372036854775808

#include <stdio.h>
int main()
{
    long i = 9223372036854775808;
    printf("%ld\n", i);
    return 0;
}

但是在LPC中,对数据溢出的处理方式不太一样,如果变量赋值超出最大值,取最大值,如果赋值小于最小值,取最小值(截止2019-04-20,fluffos最小值有BUG,实际值为-9223372036854775807),但是对赋值后的变量做运算,会和C语言一样的溢出。

我们修改 test.c 的代码如下:

int main(object me, string arg)
{
    // MAX_INT、MIN_INT 为 fluffos 定义宏
    int a = MAX_INT, b = MIN_INT;
    // 赋值时做运算
    int c = MAX_INT + 1, d = MIN_INT - 1, e = d - 1;
    // 赋值超出取值范围(10倍)
    int f = 92233720368547758070, g = -92233720368547758080;
    printf("a = %d\nb = %d\nc = %d\nd = %d\ne = %d\nf = %d\ng = %d\n", a, b, c, d, e, f, g);
}

然后输入指令 test ,输出结果如下:

a = 9223372036854775807
b = -9223372036854775807
c = -9223372036854775808
d = -9223372036854775808
e = 9223372036854775807
f = 9223372036854775807
g = -9223372036854775807

代码中的 MAX_INTMIN_INT 是 fluffos 宏定义的整形最大值和最小值,变量 a、f都取最大值,变量 b、g都取最小值,但从结果看,最小值输出比实际大了1,这是 fluffos 的 BUG。变量 c 运算后溢出,变量 d 理论上应该溢出,但因为系统 BUG ,造成实际取了真实的最小值,变量 e 运行后再次溢出。

注意,如果修改了 test.c 的代码,需要重新编译文件(指令:update /cmds/test)或者重启系统(指令:shutdown)才能生效。

示例代码中的 printf() 函数是LPC语言的 efun,可以理解为类似C语言的库函数(Library Function),因为LPC是从C语言衍生的,语法结构和C语言基本一样,但是C语言的库函数在LPC中不存在,千万不要直接把C语言中的函数在LPMUD开发中直接用,LPC语言中有很多和C语言同名且功能相同或相似的函数,如 printf(),但也有很多和C语言同名但功能不太一样的函数,如 sizeof(),还有更多C语言中没有的函数,如 this_player(),这些独有的函数让LPC开发网络游戏变的更为便捷,关于函数,后续章节会重点介绍。

为方便学习,这里先补充一下格式化输出函数 printf()的用法,在LPC语言中的用法和C语言中的同名函数基本一样,只是格式控制修饰符和数据类型有一些差异。printf() 格式控制符的完整形式如下:

%[flag][width][.precision]type

[ ] 表示此处的内容可有可无,是可以省略的。

和C语言相比,标志控制符(flag)增加了“|”(列居中对齐)、“=”(字符串纵列模式)、“'X'”(使用指定字符填充空位,X代表任意字符)和 “@”(指明参数为数组);修改了“#”的作用为字符串输出为表格模式。

输出类型(type)只有 "c"、"s"、"d/i"、"f"、"o"、"x/X",作用和C语言一致,同时新增了"O"表示LPC语言数据类型(即可输出LPC语言的任何类型),关于 printf函数 的说明文档可看这里:printf/sprintf

这里需要特别说明的是,LPC语言中没有字符型(char),但是在输出中可以把整型输出为字符,而且和C语言一样,可以直接把字符赋值给整型变量。C语言中的整数除了可以使用十进制,还可以使用二进制、八进制和十六进制,但是在LPC语言中,整数只能用十进制和十六进制赋值,可以用八进制输出,但不能用八进制赋值,而且即不能用二进制赋值也无法用二进制输出。

修改 test.c 代码如下,并 update /cmds/test

int main(object me, string arg)
{
    int a = 8, b = 'a', c = 0xF;

    printf("a = %o b = %d c = %d\n", a, b, c);

    return 1;
}

输出结果为:

a = 10 b = 97 c = 15

如果修改代码使用八进制赋值会发现其实还是十进制,而用二进制赋值的话,代码编译报错。

2. 单精度浮点型(float)

在LPC语言中,小数只有 float 类型,但是实际和C语言中的 double 类型一样,取值范围为 ±1.7E-308 和 ±1.7E308 之间。

和C语言不同的是,LPC语言中的浮点数只能用十进制形式表示,不可以使用指数形式,如果使用指数形式赋值会报错。另外,因为LPC语言中只有一种小数类型,所以,也就没有C语言中的数字加后缀L或F指明类型了。

在 fluffos 中,也定义了二个宏表示特定的浮点数,MAX_FLOAT 表示浮点数的最大值,但是 MIN_FLOAT 却是表示 0.00000,这很奇怪,不知道是 BUG,还是特意这样设定的。另外,在不同的操作系统下,浮点数的最大值和最小值是不一样的。

test.c 中使用以下代码试试看,会输出一个大到你数不出来的数字(整数部分309位)。

int main(object me, string arg)
{
    float f = MAX_FLOAT;
    printf("f = %f\n",f);

    return 1;
}

3. 字符串类型string)

和C语言差别比较大的一种数据类型,我们知道,C语言中的字符串类型实际上是字符数组,可以使用以下几种方式赋值:

#include <stdio.h>

int main(int argc, char const *argv[])
{
    char str1[] = {"mud.ren"};
    char str2[] = "mud.wiki";
    char *str = "www.mud.ren";

    printf("str1 = %s\nstr2 = %s\nstr = %s\n", str1, str2, str);
    return 0;
}

在LPC语言中,没有字符类型,无法使用这种方式赋值字符串,但是单独增加了字符串类型(string),可以直接赋值并使用。

int main(object me, string arg)
{
    string s = "www.mud.ren";
    printf("s = %s\n",s);

    return 1;
}

在LPC语言中,对字符串取子字符串的操作很别致而灵活,使用子字符串操作符,形式是:

str[n1..n2];

正整数代表从左开始,负整数代表从右开始,如果数值超过字符串的长度,视为字符串的长度。如果二个数值相等(str[n1..n1]),则传回字符串的第 n1 个字符。如果二个数值都超过字符串的长度,或者第一个数字所指的位置在第二个数字所指位置之后,会传回空字符串“”。
测试如下代码:

int main(object me, string arg)
{
    string s = "www.mud.ren";

    printf("s = %s\n",s);
    printf("s[4..4] = %s\n",s[4..4]);
    printf("s[4..6] = %s\n",s[4..6]);
    printf("s[0..20] = %s\n",s[0..20]);
    printf("s[20..30] = %s\n",s[20..30]);
    printf("s[1..0] = %s\n",s[1..0]);

    return 1;
}

输出结果:

s = www.mud.ren
s[4..4] = m
s[4..6] = mud
s[0..20] = www.mud.ren
s[20..30] =
s[1..0] =

4. 数组类型(array)

数组类型,LPC语言和C语言的差别还是比较大的,在C语言中数组定义方式为:

dataType  arrayName[length];

并且可以使用指针操作数组。例如:

#include <stdio.h>

int main(int argc, char const *argv[])
{
    int arr[] = {1, 2, 3, 4};
    int *p = arr;
    printf("arr[0] = %d\n", *p);
    return 0;
}

而在LPC语言中,数组用 ({}) 表示,并且使用类似指针的方式赋值,另外还有使用 外部函数 allocate() 初始化的方式。具体定义数组的方式如下:

dataType *arrayName;

我们继续测试如下代码:

int main(object me, string arg)
{
    string *str = ({"www", "mud", "ren"});
    int *arr = allocate(1 + random(9), (: $1 :));

    printf("str = %@-4s\n", str);
    printf("str[0] = %s, str[1] = %s, str[2] = %s\n", str[0], str[1], str[2]);
    printf("arr = %@','-2d\n", arr);

    return 1;
}

数组严格上说不是一种数据类型,只是一种数据集合,关于数组的使用,后续章节会详细讲解,这里先了解LPC语言中数组定义的方式。以上代码中第二种定义中使用了 random() 外部函数,所以使用 test 测试时每次输出是随机的。

5. 结构体类型(class)

结构体本质上也不是一种数据类型,而是一种数据集合,在C语言中数组是一组具有相同类型的数据的集合,而结构体用来存放一组不同类型的数据,但是在LPC语言中完全可以使用混合类型数组来实现存放不同类型的数据,或用使用更灵活的映射类型,所以结构体基本没什么大用,这里只做简单的介绍。

在C语言中的结构体类型关键字是 struct,LPC语言中,把关键词改成 class 了,结构体变量的定义方法和C语言有一点不一样的地方是C语言中结构体定义时 } 后面必须有 分号(;),而LPC语言中不需要,另外LPC语言中结构体赋值取值只能使用 ->

这里给一个具体的示例:

class example {
    string name;
    int age;
    string *lover;
}

int main(object me, string arg)
{
    // 变量
    class example instance = new(class example);
    // 赋值
    instance->age = 24;
    instance->name = "Ivy";
    instance->lover = ({"Alice", "Vivian", "Jessica"});
    // 取值
    printf("name : %s,age : %d, lover : %@-7s\n", instance->name, instance->age, instance->lover);
    return 1;
}

因为结构体类型在LPC语言中的定义和声明的方法不太一样,所以推荐使用 class 关键字,其实 struct 关键字也可以用,以上代码中 class 改成 struct 也可以,甚至混用也没错。

上一篇 下一篇

猜你喜欢

热点阅读