免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2387 | 回复: 9
打印 上一主题 下一主题

请教欧拉计划第18题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-06-11 10:57 |只看该作者 |倒序浏览

欧拉计划第18题(http://projecteuler.net/problem=18)

论坛徽章:
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
2 [报告]
发表于 2012-06-11 13:49 |只看该作者
本帖最后由 zhlong8 于 2012-06-11 13:49 编辑

回复 1# sequencing

use 5.010;
use strict;
use warnings;
use List::Util 'max';
use Data::Dumper;

my @data = (
[qw(75)],
[qw(95 64)],
[qw(17 47 82)],
[qw(18 35 87 10)],
[qw(20 04 82 47 65)],
[qw(19 01 23 75 03 34)],
[qw(88 02 77 73 07 63 67)],
[qw(99 65 04 28 06 16 70 92)],
[qw(41 41 26 56 83 40 80 70 33)],
[qw(41 48 72 33 47 32 37 16 94 29)],
[qw(53 71 44 65 25 43 91 52 97 51 14)],
[qw(70 11 33 28 77 73 17 78 39 68 17 57)],
[qw(91 71 52 38 17 14 91 43 58 50 27 29 48)],
[qw(63 66 04 68 89 53 67 30 73 16 69 87 40 31)],
[qw(04 62 98 27 23 09 70 98 73 93 38 53 60 04 23)],
);


my $pre = shift @data;

for my $line (@data) {
    my @max;
    for my $idx (0 .. @$line-1) {
        my($v1, $v2) = @$pre[$idx-1, $idx];
        $v1 = 0 if $idx == 0;
        $v2 = 0 if $idx == @$line-1;
        push @max, $line->[$idx] + max($v1, $v2);
    }
    $pre = \@max;
    say Dumper $pre;
}


思路是从第一行开始计算每个位置能达到的最大值。

论坛徽章:
0
3 [报告]
发表于 2012-06-11 14:14 |只看该作者
回复 2# zhlong8
  1. @$line-1   ->    $#$line
复制代码

论坛徽章:
0
4 [报告]
发表于 2012-06-11 17:12 |只看该作者
zhlong8 发表于 2012-06-11 13:49
回复 1# sequencing

use 5.010;
多谢版主!

论坛徽章:
0
5 [报告]
发表于 2012-06-12 10:50 |只看该作者
版主方法不对,以这个例子为例:
   3
  7 4
2 4 6
8 5 9 3

用版主的方法可以计算得到3+7+4+9 = 23,如果将左下角的8换成99,这种方法就得不出正确结论了。
我的办法是假定最后一个元素是$data[$n][$i],那么就有sum[$n][$i] = max($data[$n][$i] +sum[$n-1][$i],$data[$n][$i]+sum[$n-1][$i-1],代码如下:
  1. #!/usr/bin/perl

  2. use strict;
  3. use warnings;
  4. use 5.010;

  5. my @data = (
  6. [qw(75)],
  7. [qw(95 64)],
  8. [qw(17 47 82)],
  9. [qw(18 35 87 10)],
  10. [qw(20 04 82 47 65)],
  11. [qw(19 01 23 75 03 34)],
  12. [qw(88 02 77 73 07 63 67)],
  13. [qw(99 65 04 28 06 16 70 92)],
  14. [qw(41 41 26 56 83 40 80 70 33)],
  15. [qw(41 48 72 33 47 32 37 16 94 29)],
  16. [qw(53 71 44 65 25 43 91 52 97 51 14)],
  17. [qw(70 11 33 28 77 73 17 78 39 68 17 57)],
  18. [qw(91 71 52 38 17 14 91 43 58 50 27 29 48)],
  19. [qw(63 66 04 68 89 53 67 30 73 16 69 87 40 31)],
  20. [qw(04 62 98 27 23 09 70 98 73 93 38 53 60 04 23)],
  21. );

  22. sub max{

  23.         my ($x,$y) = @_;
  24.         if($x>$y){
  25.                 return $x;
  26.         }else{
  27.                 return $y;
  28.         }
  29. }

  30. sub sum{

  31.         my ($row,$column) = @_;
  32.         my ($sum1,$sum2);
  33.         if(0>$column){
  34.                 say 'stop';
  35.                 return 0;
  36.         }elsif($column>$row){
  37.                 say 'stop';
  38.                 return 0;
  39.         }
  40.         if(0==$row){
  41.                 say 'error';
  42.                 return undef;
  43.         }elsif(1==$row){
  44.                 $sum1 = $data[0][0]+$data[1][0];
  45.                 $sum2 = $data[0][0]+$data[1][1];
  46.                 say "row\t$row\tcolumn\t$column\t",$data[$row][$column];
  47.                 return max($sum1,$sum2);
  48.         }else{
  49.                 $sum1 = sum($row-1,$column)+$data[$row][$column];
  50.                 $sum2 = sum($row-1,$column-1)+$data[$row][$column];
  51.                 say "row\t$row\tcolumn\t$column\t",$data[$row][$column];
  52.                 return max($sum1,$sum2);
  53.         }

  54. }

  55. my @sums;
  56. for(0..14){

  57.         push @sums,sum(14,$_);
  58.         say $sums[-1];

  59. }


  60. say join "\n",@sums;
复制代码

论坛徽章:
0
6 [报告]
发表于 2012-06-12 11:20 |只看该作者
我这种办法还是不行,全部遍历了一次,数据量大后,会无法处理

论坛徽章:
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
7 [报告]
发表于 2012-06-12 11:39 |只看该作者
slayer_axe 发表于 2012-06-12 11:20
我这种办法还是不行,全部遍历了一次,数据量大后,会无法处理


你的方法就是历遍啊是 O(N!)   没有归约掉不可能的路径。另外99怎么不对了

论坛徽章:
0
8 [报告]
发表于 2012-06-12 12:00 |只看该作者
我弄错了,抱歉,我把问题搞复杂了。

论坛徽章:
0
9 [报告]
发表于 2012-06-13 08:56 |只看该作者
本帖最后由 dgtnk 于 2012-06-15 09:33 编辑

回复 1# sequencing


    另一种思路 从下到上
  1. #!/usr/bin/perl -w
  2. use strict;
  3. use 5.010;
  4. use List::Util 'max';


  5. my @array;
  6. my $index = 0;

  7. while (<DATA>) {
  8.         my @temp = split;
  9.         $array[$index] = \@temp;
  10.         $index ++;
  11. }

  12. for  (my $i = $#array; $i > 0; $i --) {
  13.         for (my $j = 0; $j < @{$array[$i - 1]}; $j ++) {
  14.                 $array[$i - 1][$j] += max($array[$i][$j], $array[$i][$j + 1]);
  15.         }
  16. }

  17. say $array[0][0];

  18. __DATA__
  19. 75
  20. 95 64
  21. 17 47 82
  22. 18 35 87 10
  23. 20 04 82 47 65
  24. 19 01 23 75 03 34
  25. 88 02 77 73 07 63 67
  26. 99 65 04 28 06 16 70 92
  27. 41 41 26 56 83 40 80 70 33
  28. 41 48 72 33 47 32 37 16 94 29
  29. 53 71 44 65 25 43 91 52 97 51 14
  30. 70 11 33 28 77 73 17 78 39 68 17 57
  31. 91 71 52 38 17 14 91 43 58 50 27 29 48
  32. 63 66 04 68 89 53 67 30 73 16 69 87 40 31
  33. 04 62 98 27 23 09 70 98 73 93 38 53 60 04 23
复制代码

论坛徽章:
0
10 [报告]
发表于 2012-06-14 19:30 |只看该作者
受到启发。
应该从最下面开始算。

   63 66 04 68 89 53 67 30 73 16 69 87 40 31
04 62 98 27 23 09 70 98 73 93 38 53 60 04 23

倒数第二行 第一个 63 可以和倒数第一行第一个 和第二个结合 ,那么取最大的 ,以此类推。
倒数第二行就变成了
63+62 66+98 04+98 68+27 89+23 53+70 67+98 30+98 73+93 16+93 69+53 87+60 40+60 31+23
然后一层一层往上走
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP