分配给列表容器混淆

Assignment to a List Container Confusion

我可能患有脑衰退,但根据有关项目和列表分配的文档 (https://docs.raku.org/language/variables#Item_and_list_assignment),

对列表容器(列表上下文)的分配总是触发列表分配。

然而,这似乎与我从代码中得到的内容冲突(此处转载于 raku repl)..

> my %syns-by-name = %(Bq => ["Bq", "becquerel", "becquerels"], C => ["C", "coulomb", "coulombs"],)
    {Bq => [Bq becquerel becquerels], C => [C coulomb coulombs]}
> my Str @res = %syns-by-name{'Bq'};
    Type check failed in assignment to @res; expected Str but got Array (["Bq", "becquerel", ...) in block <unit> at <unknown file> line 1
> my Str @res = [|%syns-by-name{'Bq'}];
    [Bq becquerel becquerels]

这是一个错误,还是我误解了意图...?

Welcome to ™ v2020.10.
Implementing the ™ programming language v6.d.
Built on MoarVM version 2020.10.

TL;DR 如果非本地项通过赋值是可变的,则它是 Scalar。列表分配不会展平 Scalar 项。 [1]

一点头绪

考虑这段代码:

my %map = :a;
%map<a> = 42;
say %map<a>; # 42

作业有效,因为:

say %map<a>.VAR.WHAT; # (Scalar)

“列表分配”

考虑这段代码:

my $scalar = 1,2;       # Useless use of constant integer 2
say $scalar;            # 1

my @list1 = 1,2;        # "list assignment", so RHS is iterated
say @list1;             # [1 2]

这就是项目分配和列表分配之间的一个区别。

my @list3 = [1,2];      # Again, RHS is iterated
say @list3;             # [1 2]

my @list2 = (1,2);      # Again, RHS is iterated
say @list2;             # [1 2]

my @list4 = 1,(2,3);    # Again, RHS is iterated. Second element stays as a `List`.
say @list4;             # [1 (2 3)]

my @list5 = 1,[2,3];    # Again, RHS is iterated. Second element stays as an `Array`.
say @list5;             # [1 [2 3]]

如果 RHS 上只列出了一个项目,并且它不是 Scalar,列表赋值会将其展平。但在所有其他情况下,列表分配不会展平项目。

my @list6 = $[1,2];     # RHS is a `Scalar`. So it doesn't get flattened.
say @list6;             # [[1 2]]

我很困惑!

打球情况Q:

my Str @res = %( :a[42,99] )<a>;

这会产生相同类型的错误。

因为:

say .VAR.WHAT given :a[42,99]<a>;       # (Array)
say .VAR.WHAT given (% = :a[42,99])<a>; # (Scalar)

脚注

[1] 当推测产生惊喜,并将其转化为学习时,您就会实现并理想化对 ERNing.

注意:阅读不会出错。但是,我将尝试解释这里发生的事情并提出可能的解决方案。

在 Raku,context 就是一切。这意味着不同的上下文会在不同的上下文中触发,这意味着您正在使用的数据结构具有不同的 boxingunboxing 功能。

让我们看看这个

my %syns-by-name = %(Bq => ["Bq", "becquerel", "becquerels"], C => ["C", "coulomb", "coulombs"],)

我们确实知道我们处于“关联”上下文中,因为等号两边的百分比标记。我们甚至不需要在右侧使用它:

my %syns-by-name = Bq => ["Bq", "becquerel", "becquerels"], C => ["C", "coulomb", "coulombs"]

因为,看lhs,还是一个Associative,所以我们知道我们手里拿的是什么。此上下文中的相同 rhs 代码:

my @list-of-signs = Bq => ["Bq", "becquerel", "becquerels"], C => ["C", "coulomb", "coulombs"] # [Bq => [Bq becquerel becquerels] C => [C coulomb coulombs]]

将生成一个配对列表。因此,我们检查您在 OP 中提到的内容:

Assignment to a List container (list-context) always triggers list assignment.

是真实的。这是一个“列表”容器,我们只是通过上下文将 rhs 变成一个列表。这里发生的事情没有歧义:rhs 是一个逗号分隔的列表,lhs 是一个位置。所以给你。

您的代码中的情况略有不同。如果我们使用 2020.12

可能会更清楚一些
 my %syns-by-name = Bq => ["Bq", "becquerel", "becquerels"], C => ["C", "coulomb", "coulombs"]
my Str @res = %syns-by-name{'Bq'};
# Type check failed in assignment; expected Positional[Str] but got Array ($["Bq", "becquerel", ...)

第一眼可能看不出区别,所以我在这里强调一下:

Type check failed in binding; expected Positional[Str] but got Array ($["Bq", "becquerel", ...)

这表明该数组是 itemized,它已被 装箱 以便它可以适合类似标量的事物,例如哈希键的值。默认情况下,哈希值是标量的,在它们的定义中组成:

role Associative[::TValue = Mu, ::TKey = Str(Any)] { }

所以你提到的仍然成立:列表赋值被触发,就像在“列表上下文”中一样。但是,单个项目只能通过一种方式转换为列表:通过使其成为列表中的单个元素。如果您仔细查看错误报告,它就是这样说的:嘿,您告诉我您要给我一个 Strs 列表。这不是那个!这是一个(逐项列出的)数组列表!这有效:

my List @res = %syns-by-name{'Bq'};
# [(Bq becquerel becquerels)]

我们需要做的是对这个东西进行«unbox»,即取回寄居在Scalar中的Array。简单易行:

my Str @res = %syns-by-name{'Bq'}<>;
# [Bq becquerel becquerels]

de-cont operator (which maybe should be called "unbox" or "de-Scalarize" or "de-itemize" operator) 为您做正确的事。但是,这可能不是您想要的。你希望你的数组是数组。然而,事实证明这有点棘手,我将需要另一个 SO 问题来解决它。