5

我知道这里已经涵盖了类似的主题,但是我遇到了一个问题,我认为这是由于我误解了数组切片是如何在 foreach 循环的上下文中插入的。我无法弄清楚我哪里出错了,所以我正在寻找一些见解。

我有一个行数可变的二维数组。例如目的:

@2DArray = (['str1', 1, 2, 'E', val1, val2, val3]
            ['str2', 3, 4, 'E', val4, val5, val6]
            ['str4', 5, 6, 'F', val7, val8, val9])   

我想构建一个带有附加列的新数组,仅当原始数组的某些行在第 3 列中包含字符串“E”时才包含它们。此外,对于我希望合并到新数组中的行,我只想要列的一个子集,并且我希望该子集以不同的顺序排列。最终目标是生成下游脚本所需的正确格式的输出。

这是我的尝试:

my $projName = 'test';

my $i = 1;
my @Newarray
my @Newarray_element;
     foreach (@2DArray) {
         if  (${$_}[3] eq 'E') {
             ${$_}[3] = $i; 
             ${$_}[5] = '+'; 
             @Newarray_element = ("$projName$i", @$_[0,1,2,5,3], 'STR', 11, 11);
             $i++;
             push (@Newarray, \@Newarray_element);
         }

         next;
     }

print (join("\t", @$_), "\n") for @Newarray;

但是,如果我这样做,我得到的是:

#(original) col nums:      0       1    2    5    3

                  test2    str2    3    4    +    2    STR    11    11
                  test2    str2    3    4    +    2    STR    11    11

即,我的新数组将在原始数组中的每一行都有一行,在第 3 列中带有一个“E”,但每一行都填充了要由循环处理的最后一行的值。

我认为这个问题与在 foreach 循环中切片二维数组有关的原因是我知道我是否只是循环遍历一个二维数组,在第 3 列中找到所有带有“E”的行,修改其他列中的一些值对于这些行,然后将其返回到一个新数组,这一切都很好。也就是说,如果我改为这样做:

my @Newarray;
my $i = 1;
foreach (@2Darray) {
    if  (${$_}[3] eq "E") {
        ${$_}[3] = $i;
        ${$_}[5] = '+';
        $i++;
        push (@Newarray, \@$_);
    }
    next;   
}
print (join("\t", @$_), "\n") for @Newarray;

我得到了我期望的输出:

                  *            &
str1    1    2    1    val1    +    val3
str2    3    4    2    val4    +    val6

其中标有 * 和 & 的列是修改后的第 3 列和第 5 列。让猛攻开始:我的新手自己哪里出错了?

4

1 回答 1

4

该变量@Newarray_element指向整个程序的相同内存空间,因此您在一次迭代中所做的更改会传播到您在分配中使用该变量的先前迭代。

两个可能的修复:

。更改变量的范围,使其在每次迭代中使用不同的内存。改变

my @Newarray_element;
foreach (@2DArray) {
    ...

foreach (@2DArray) {
    my @Newarray_element;
    ...

甚至

foreach (@2DArray) {
    ...
    my @Newarray_element = ("$projName$i", @$_[0,1,2,5,3], 'STR', 11, 11);

:重用@Newarray_element但将其数据的副本分配给每一行@Newarray。改变

push (@Newarray, \@Newarray_element);

push (@Newarray, [ @Newarray_element ]);

后一个调用创建并附加一个新的匿名数组引用到@Newarray.

于 2013-04-18T22:43:33.130 回答