本文系 Creating JVM language 翻译的第 12 篇。
原文中的代码和原文有不一致的地方均在新的代码仓库中更正过,建议参考新的代码仓库。

源码

Github

为什么需要命名参数

在 Java 中(多数语言中也是如此)方法调用的参数匹配是通过索引值,如果方法调用的参数比较少并且参数的类型有差别的情况,是合理的。不幸的是,如果方法调用的参数有很多个,并且类型相同,这是个悲剧。

例如:
Rect createRectangle(int x1,int y1,int x2, int y2) //createRectangle signature

我打赌你很有可能会传错参数。

你发现问题了吗?这种情况开发者很容易搞混参数的顺序,由于是相同类型,编译器也没办帮你检查问题。

这就是命名参数的有点,你可以给参数指定名字,而不是仅仅通过索引值来指定参数。
使用命名参数有很多好处:

  • 参数的顺序不受限制
  • 代码可读性提高
  • 不用再两个文件中跳转对比方法的签名和实际传参是否一致

语法规则更改

1
2
3
functionCall : functionName '('argument? (',' argument)* ')';
argument : expression //unnamed argument
| name '->' expression ; //named argument

方法调用的参数之间用逗号分割。argument 有两种格式,命名参数和未命名参数,这两种格式不允许同时存在。

记录参数

在第七部分描述到,方法的解析分为两个步骤: 首先记录所有的方法签名(方法的声明),下一步是解析方法体,这样保证在解析方法体的时候,所有的方法签名都已经被解析过了。

实现命名参数的思路是把命名参数的调用转换成未命名参数的调用,参数索引位置通过方法签名去获得:

  • 在方法签名中查找匹配的参数名字
  • 获得参数的索引
  • 如果参数的索引值和实际不一致,记录下来

上图中的示例,x1 的索引和 y1 对调。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
{
//other stuff
@Override
public Expression visitFunctionCall(@NotNull EnkelParser.FunctionCallContext ctx) {
String funName = ctx.functionName().getText();
FunctionSignature signature = scope.getSignature(funName);
List<EnkelParser.ArgumentContext> argumentsCtx = ctx.argument();
//Create comparator that compares arguments based on their index in signature
Comparator<EnkelParser.ArgumentContext> argumentComparator = (arg1, arg2) -> {
if(arg1.name() == null) return 0; //If the argument is not named skip
String arg1Name = arg1.name().getText();
String arg2Name = arg2.name().getText();
return signature.getIndexOfParameter(arg1Name) - signature.getIndexOfParameter(arg2Name);
};
List<Expression> arguments = argumentsCtx.stream() //parsed arguments (wrong order)
.sorted(argumentComparator) //Order using created comparator
.map(argument -> argument.expression().accept(this)) //Map parsed arguments into expressions
.collect(toList());
return new FunctionCall(signature, arguments);
}
}

这种方式对字节码的生成是透明的,字节码生成阶段无需了解方法调用参数是命名还是未命名

示例

如下的 Enkel 代码:

1
2
3
4
5
6
7
8
9
10
NamedParamsTest {

main(string[] args) {
createRect(x1->25,x2->-25,y1->50,y2->-50)
}

createRect (int x1,int y1,int x2, int y2) {
print "Created rect with x1=" + x1 + " y1=" + y1 + " x2=" + x2 + " y2=" + y2
}
}

编译后的字节码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class NamedParamsTest {
public static void main(java.lang.String[]);
Code:
0: bipush 25 //x1 (1 index in call)
2: bipush 50 //y1 (3 index in call)
4: bipush -25 //x2 (2 index in call)
6: bipush -50 //y2 (4 index in call)
8: invokestatic #10 // Method createRect:(IIII)V
11: return

public static void createRect(int, int, int, int);
Code:
//normal printing code
}

输出:
Created rect with x1=25 y1=50 x2=-25 y2=-50