Chinaunix

标题: 【求解】如何将2维数组转成多层哈希 [打印本页]

作者: yiten    时间: 2010-11-13 11:03
标题: 【求解】如何将2维数组转成多层哈希
通用的方法,传进任意的2维数组
如:把[ [1,2,3], ['a','b','c'], [ 1, 'b', '3']]转成
{ 1 => { 2 => 3, b => 3 }, a => { b => c } }
作者: zhlong8    时间: 2010-11-13 11:51
  1. sub a2h {
  2.     my $array_ref = shift;
  3.     my $hash_ref;
  4.     for (@$array_ref) {
  5.         my($k1, $k2, $v) = @$_;
  6.         $hash_ref->{$k1}{$k2} = $v;
  7.     }
  8.     return $hash_ref;
  9. }
复制代码

作者: flw    时间: 2010-11-13 11:55
$hash->{$_->[0]}{$_->[1]} = $_->[2] foreach @$array;
作者: yiten    时间: 2010-11-13 12:00
我说的是通用的
如果是[ [1,2,3,4], ['a','b','c',4], [ 1, 'b', '3',4]]呢
作者: yiten    时间: 2010-11-13 12:02
就是[$array_1,...,$array_n]
$array_n为相同长度的数组长度>=2,且至少有一个元素不同
作者: zhlong8    时间: 2010-11-13 12:11
本帖最后由 zhlong8 于 2010-11-13 12:31 编辑

用了递归。下次问题描述清楚点啊,通用什么的
  1. use strict;
  2. use warnings;
  3. use Data::Dumper;

  4. sub l2h;

  5. sub a2h {
  6.     my $array_ref = shift;
  7.     my $hash_ref;
  8.     for (@$array_ref) {
  9.         my $k = shift @$_;
  10.         $hash_ref->{$k} = l2h @$_;
  11.     }
  12.     return $hash_ref;
  13. }

  14. sub l2h {
  15.     if (@_ == 1) {
  16.         return shift;
  17.     } else {
  18.         my $key = shift;
  19.         return {$key => &l2h(@_)};
  20.     }
  21. }

  22. print Dumper a2h [ [1,2,3,4], ['a','b','c',4], [ 1, 'b', '3',4]]
复制代码
因为你有重复的 key 1,所以输出为
  1. $VAR1 = {
  2.           '1' => {
  3.                    'b' => {
  4.                             '3' => 4
  5.                           }
  6.                  },
  7.           'a' => {
  8.                    'b' => {
  9.                             'c' => 4
  10.                           }
  11.                  }
  12.         };
复制代码

作者: yiten    时间: 2010-11-13 13:15
多谢啊,果然是高手
作者: yiten    时间: 2010-11-13 13:20
好像不对
print Dumper a2h [ [1,2,3,4,5], ['a','b','c',4, 6], [ 'a', 'b', 2,3,4], [ 1, 2, 3,'b', 4] ]结果是
$VAR1 = {
          '1' => {
                   '2' => {
                            '3' => {
                                     'b' => 4
                                   }
                          }
                 },
          'a' => {
                   'b' => {
                            '2' => {
                                     '3' => 4
                                   }
                          }
                 }
        };
作者: yiten    时间: 2010-11-13 13:22
正确的结果应该是
$VAR1 = {
          '1' => {
                   '2' => {
                            '3' => {
                                     'b' => 4
                                     '4' => 5
                                   }
                          }
                 },
          ....
          ....
        };
作者: zhlong8    时间: 2010-11-13 13:34
本帖最后由 zhlong8 于 2010-11-13 13:38 编辑

回复 9# yiten


    ……没看上面我说你有重复key所以显示只有两个。原来你每一层key可以随意在后面添加的。我要哭了,说清楚嘛
作者: yiten    时间: 2010-11-13 13:50
恩,是的。
你的方法在第一层是分开了,但是第二层开始都是覆盖了
作者: zhlong8    时间: 2010-11-13 13:50
本帖最后由 zhlong8 于 2010-11-13 14:01 编辑

可以处理任意的里面元素长度超过2的二维数组,不需要等长,不过要注意不能有重复key
  1. use strict;
  2. use warnings;
  3. use Data::Dumper;

  4. sub l2h;

  5. sub a2h {
  6.     my $array_ref = shift;
  7.     my %hash;
  8.     for (@$array_ref) {
  9.         l2h \%hash, $_;
  10.     }
  11.     return \%hash;
  12. }

  13. sub l2h {
  14.     my($hash_ref, $array_ref) = @_;
  15.     my $key = shift @$array_ref;

  16.     if (@$array_ref == 1) {
  17.         $hash_ref->{$key} = shift @$array_ref;
  18.         return;
  19.     }
  20.     elsif (exists $hash_ref->{$key} and ref($hash_ref->{$key}) eq 'HASH') {
  21.         &l2h($hash_ref->{$key}, $array_ref);
  22.     }
  23.     else {
  24.         $hash_ref->{$key} = {};
  25.         &l2h($hash_ref->{$key}, $array_ref);
  26.     }
  27. }


  28. print Dumper a2h [[1,2,3,4,5], ['a','b','c',4, 6], [ 'a', 'b', 2,3,4], [ 1, 2, 3,'b', 4]]
复制代码
输出为
  1. $VAR1 = {
  2.           '1' => {
  3.                    '2' => {
  4.                             '3' => {
  5.                                      '4' => 5,
  6.                                      'b' => 4
  7.                                    }
  8.                           }
  9.                  },
  10.           'a' => {
  11.                    'b' => {
  12.                             'c' => {
  13.                                      '4' => 6
  14.                                    },
  15.                             '2' => {
  16.                                      '3' => 4
  17.                                    }
  18.                           }
  19.                  }
  20.         };
复制代码

作者: yiten    时间: 2010-11-13 13:56
这个是对的了,厉害,呵呵{:3_193:}
作者: yiten    时间: 2010-11-13 14:01
我的思路是到了,要把外面的哈希引用传到递归里去填充。
但是就是出不来正确结果,看了这个果然觉悟了,哈哈
作者: zhlong8    时间: 2010-11-13 14:11
本帖最后由 zhlong8 于 2010-11-13 14:14 编辑

回复 14# yiten


    写了个循环版的,FP用多习惯递归了
  1. use strict;
  2. use warnings;
  3. use Data::Dumper;

  4. sub a2h {
  5.     my $array_ref = shift;
  6.     my %hash;

  7.     for my $item (@$array_ref) {
  8.         my $hash_ref = \%hash;
  9.         while (@$item > 2) {
  10.             my $key = shift @$item;
  11.             if (exists $hash_ref->{$key} and ref($hash_ref->{$key}) eq 'HASH') {
  12.                 $hash_ref = $hash_ref->{$key};
  13.             }
  14.             else {
  15.                 $hash_ref->{$key} = {};
  16.                 $hash_ref = $hash_ref->{$key};
  17.             }
  18.         }
  19.         my($key, $val) = @$item;
  20.         $hash_ref->{$key} = $val;
  21.     }

  22.     \%hash;
  23. }

  24. print Dumper a2h [[1,2,3,4,5], ['a','b','c',4, 6], ['a', 'b', 2,3,4], [ 1, 2, 3,'b', 4],[1,'kk','afkd'],[1,'kk', 'afkd',2]]
复制代码
输出为
  1. $VAR1 = {
  2.           '1' => {
  3.                    'kk' => {
  4.                              'afkd' => 2
  5.                            },
  6.                    '2' => {
  7.                             '3' => {
  8.                                      '4' => 5,
  9.                                      'b' => 4
  10.                                    }
  11.                           }
  12.                  },
  13.           'a' => {
  14.                    'b' => {
  15.                             'c' => {
  16.                                      '4' => 6
  17.                                    },
  18.                             '2' => {
  19.                                      '3' => 4
  20.                                    }
  21.                           }
  22.                  }
  23.         };
复制代码

作者: yiten    时间: 2010-11-13 15:04
向zhlong8致敬
自己感觉这个问题还是比较有深度的和技巧的,有人会发现这个转化还是比较有用的,哈哈
作者: yiten    时间: 2010-11-13 15:14
这2个算法已经很不错了,$hash_ref有类似游标的作用
但是里头有对数组的shift操作,而且相当于shift了mxn次,我觉得在数据量比较大的情况下,还不是很轻快。
作者: zhlong8    时间: 2010-11-13 15:20
本帖最后由 zhlong8 于 2010-11-13 15:25 编辑

回复 17# yiten


    shift 很快的不用担心(内部用偏移量实现,时间复杂度为 O(1)),需要历遍所有元素这个复杂度必须的吧?
作者: yiten    时间: 2010-11-13 15:32
回复 18# zhlong8


    恩,我的思路也只有这个,但是能不能也借鉴用类似游标来做,尽量不要修改原数组,不知道能不能行的通。有空再研究吧,呵呵
作者: zhlong8    时间: 2010-11-13 19:28
回复 19# yiten


    用下标索引速度反而慢了。数组结构也不复杂,可以用 Clone 复制份来操作
作者: yiten    时间: 2010-12-24 10:10
这几天有空,把zhlong8的第二个算法改写了下,改写后不会修改原数组,也快了一些。
如果有人发现这个功能有用,可以参考下,我是个好人

  1. sub array2hash {
  2.     my $array = shift;
  3.    
  4.     my $result_hash = {};
  5.     my $temp_hash;
  6.    
  7.     for my $item (@$array) {
  8.         $temp_hash = $result_hash;
  9.         
  10.         for my $i ( 0..$#$item - 2 ) {
  11.             my $key = $item->[$i];
  12.             
  13.             $temp_hash->{$key} = {} unless ref $temp_hash->{$key};
  14.             
  15.             $temp_hash = $temp_hash->{$key};
  16.         }
  17.         
  18.         $temp_hash->{$item->[-2]} = $item->[-1] if exists $item->[-2];
  19.     }

  20.     return $result_hash;
  21. }
复制代码

作者: yiten    时间: 2010-12-24 10:11
本帖最后由 yiten 于 2010-12-24 10:12 编辑






欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2