免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: yiten
打印 上一主题 下一主题

【求解】如何将2维数组转成多层哈希 [复制链接]

论坛徽章:
0
11 [报告]
发表于 2010-11-13 13:50 |只看该作者
恩,是的。
你的方法在第一层是分开了,但是第二层开始都是覆盖了

论坛徽章:
46
15-16赛季CBA联赛之四川
日期:2018-03-27 11:59:132015年亚洲杯之沙特阿拉伯
日期:2015-04-11 17:31:45天蝎座
日期:2015-03-25 16:56:49双鱼座
日期:2015-03-25 16:56:30摩羯座
日期:2015-03-25 16:56:09巳蛇
日期:2015-03-25 16:55:30卯兔
日期:2015-03-25 16:54:29子鼠
日期:2015-03-25 16:53:59申猴
日期:2015-03-25 16:53:29寅虎
日期:2015-03-25 16:52:29羊年新春福章
日期:2015-03-25 16:51:212015亚冠之布里斯班狮吼
日期:2015-07-13 10:44:56
12 [报告]
发表于 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.         };
复制代码

论坛徽章:
0
13 [报告]
发表于 2010-11-13 13:56 |只看该作者
这个是对的了,厉害,呵呵{:3_193:}

论坛徽章:
0
14 [报告]
发表于 2010-11-13 14:01 |只看该作者
我的思路是到了,要把外面的哈希引用传到递归里去填充。
但是就是出不来正确结果,看了这个果然觉悟了,哈哈

论坛徽章:
46
15-16赛季CBA联赛之四川
日期:2018-03-27 11:59:132015年亚洲杯之沙特阿拉伯
日期:2015-04-11 17:31:45天蝎座
日期:2015-03-25 16:56:49双鱼座
日期:2015-03-25 16:56:30摩羯座
日期:2015-03-25 16:56:09巳蛇
日期:2015-03-25 16:55:30卯兔
日期:2015-03-25 16:54:29子鼠
日期:2015-03-25 16:53:59申猴
日期:2015-03-25 16:53:29寅虎
日期:2015-03-25 16:52:29羊年新春福章
日期:2015-03-25 16:51:212015亚冠之布里斯班狮吼
日期:2015-07-13 10:44:56
15 [报告]
发表于 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.         };
复制代码

论坛徽章:
0
16 [报告]
发表于 2010-11-13 15:04 |只看该作者
向zhlong8致敬
自己感觉这个问题还是比较有深度的和技巧的,有人会发现这个转化还是比较有用的,哈哈

论坛徽章:
0
17 [报告]
发表于 2010-11-13 15:14 |只看该作者
这2个算法已经很不错了,$hash_ref有类似游标的作用
但是里头有对数组的shift操作,而且相当于shift了mxn次,我觉得在数据量比较大的情况下,还不是很轻快。

论坛徽章:
46
15-16赛季CBA联赛之四川
日期:2018-03-27 11:59:132015年亚洲杯之沙特阿拉伯
日期:2015-04-11 17:31:45天蝎座
日期:2015-03-25 16:56:49双鱼座
日期:2015-03-25 16:56:30摩羯座
日期:2015-03-25 16:56:09巳蛇
日期:2015-03-25 16:55:30卯兔
日期:2015-03-25 16:54:29子鼠
日期:2015-03-25 16:53:59申猴
日期:2015-03-25 16:53:29寅虎
日期:2015-03-25 16:52:29羊年新春福章
日期:2015-03-25 16:51:212015亚冠之布里斯班狮吼
日期:2015-07-13 10:44:56
18 [报告]
发表于 2010-11-13 15:20 |只看该作者
本帖最后由 zhlong8 于 2010-11-13 15:25 编辑

回复 17# yiten


    shift 很快的不用担心(内部用偏移量实现,时间复杂度为 O(1)),需要历遍所有元素这个复杂度必须的吧?

论坛徽章:
0
19 [报告]
发表于 2010-11-13 15:32 |只看该作者
回复 18# zhlong8


    恩,我的思路也只有这个,但是能不能也借鉴用类似游标来做,尽量不要修改原数组,不知道能不能行的通。有空再研究吧,呵呵

论坛徽章:
46
15-16赛季CBA联赛之四川
日期:2018-03-27 11:59:132015年亚洲杯之沙特阿拉伯
日期:2015-04-11 17:31:45天蝎座
日期:2015-03-25 16:56:49双鱼座
日期:2015-03-25 16:56:30摩羯座
日期:2015-03-25 16:56:09巳蛇
日期:2015-03-25 16:55:30卯兔
日期:2015-03-25 16:54:29子鼠
日期:2015-03-25 16:53:59申猴
日期:2015-03-25 16:53:29寅虎
日期:2015-03-25 16:52:29羊年新春福章
日期:2015-03-25 16:51:212015亚冠之布里斯班狮吼
日期:2015-07-13 10:44:56
20 [报告]
发表于 2010-11-13 19:28 |只看该作者
回复 19# yiten


    用下标索引速度反而慢了。数组结构也不复杂,可以用 Clone 复制份来操作
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP