油漆那个栅栏


9

您是汤姆·索耶(Tom Sawyer),必须粉刷一下102400 m长的篱笆。幸运的是,您的朋友决定帮助您交换各种东西。每个朋友会画画大号米,从开始小号颜色Ç小号大号是米整数量和1≤ ç ≤97感到厌烦你决定要找出你有各种颜色的多少米。

输入项

从标准输入读取输入。如上所述,每行包含三个数字SLC

乌普特

输出被写入标准输出。对于最终栅栏上出现的每种颜色,请打印颜色编号及其出现的次数。按颜色排序。

例子

输入0

                           ..............
0 3 1                      111...........
2 4 2                      112222........
1 2 3                      133222........
0 4 1                      111122........
7 3 5                      111122.555....

输出0

1 4
2 2
5 3

输入1

 0 100 1
 50 150 2

输出1

 1 50
 2 150

输入2

 500 1000 1
 0 2000 2

输出2

 2 2000

更多例子

这是一个小型发电机:

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>


/* From http://en.wikipedia.org/wiki/Random_number_generation */
unsigned m_w;
unsigned m_z;

unsigned get_random()
{
  m_z = 36969 * (m_z & 65535) + (m_z >> 16);
  m_w = 18000 * (m_w & 65535) + (m_w >> 16);
  return (m_z << 16) + m_w;  /* 32-bit result */
}

int main(int argc, char **argv)
{
  int i;

  assert(argc == 2);
  m_w = 0xbabecafe;
  m_z = atoi(argv[1]);

  i = 10;
  while (i--);
    get_random();

  i = atoi(argv[1]);
  while (i--) {
    int s = (int) ((get_random() << 8) % 102397);
    int l = (int) ((get_random() << 8) % (102397 - s));
    int c = (int) ((get_random() << 8) % 97 + 1);
    printf("%d %d %d\n", s, l, c);
  }

  return 0;
}

运行示例:

$ ./gen 1 | ./paint
6 535
$ ./gen 10 | ./paint
28 82343
36 3476
41 1802
49 4102
82 1656
$ ./gen 100 | ./paint
2 2379
22 17357
24 4097
25 1051
34 55429
42 9028
45 9716
66 1495
71 196
85 640
97 706
$ ./gen 1000 | ./paint
16 719
26 29
28 24
33 1616
55 371
65 35
69 644
74 16
84 10891
86 36896
87 50832
89 19
$ ./gen 10000 | ./paint
3 800
6 5712
14 3022
17 16
26 1
29 18770
31 65372
37 387
44 40
49 37
50 93
55 11
68 278
70 19
71 64
72 170
77 119
78 6509
89 960
97 15
$ ./gen 100000 | ./paint
2 6
8 26
12 272
24 38576
26 1
34 1553
35 8
36 19505
43 2
45 11
46 2
47 9
49 27339
50 139
53 3109
69 11744
92 89
$ ./gen 1000000 | ./paint
1 1
3 4854
6 523
13 1
16 11
18 416
22 7
24 3920
25 96
31 10249
32 241
37 1135
45 10
57 758
62 2348
65 11
66 7422
78 6
85 13361
87 3833
88 187
91 46
93 7524
96 45436

您的程序必须在合理的时间内运行。在上一个示例中,我的解决方案只需几秒钟即可运行。

最短的代码胜出。

包括运行时间和上次测试的输出。

编辑:此问题不旨在被强行使用,所以一个简单的解决方案是不可接受的。


在我看来,最简单的方法(分配数组,填充数组,计算数组中每种颜色的数量,输出)将在合理的时间内运行。不过,您似乎想提出一种算法来应对挑战,但我想错了吗?
马修(Matthew)

我当时以为1000000 ops X 25000平均长度= 25 * 10 ^ 9不会在合理的时间内运行。否则,我可以增加围栏的长度。
Alexandru

啊,我很想念输入的内容是一百万行,这很糟糕。
马修(Matthew)

1
@基思:ITYM 英制单位en.wikipedia.org/wiki/Imperial_units
Paul R

1
基思:我想您可以假设今天的汤姆·索亚(Tom Sawyer)会更明智,并使用SI。
乔伊(Joey)

Answers:


2

Python,221 239个字符

import sys
F=[]
for L in sys.stdin:s,l,c=map(int,L.split());F=sum([[(a,min(b,s),d)]*(a<s)+[(max(a,s+l),b,d)]*(b>s+l)for a,b,d in F],[(s,s+l,c)])
C=[0]*98
for a,b,c in F:C[c]+=b-a
for c in range(98):
 if C[c]:print c,C[c]

保持F为代表栅栏当前状态的三元组的无序列表(运行开始,运行结束,颜色)。由于生成器中的随机绘制会相当频繁地覆盖大量区域,因此此列表永远不会太长(通常在15至40范围内)。

在1M示例中运行37秒。


您可以使用G+=[(a,min(b,s),d)]*(a<s)等在一行上获取for循环
gnibbler 2011年

for C in sorted(f[2] for f in F):print C,sum(b-a for a,b,c in F if c==C)节省超过你的最后四行几个字符,并避免了需要知道的神奇数字98

@Gareth:我认为如果同一颜色被多个范围使用,将打印出重复的图像。那里某处需要唯一性……
Keith Randall

您说对了:这将sorted(set(...))不再是一种进步。

1

Haskell中,251个261 269 294字符

import Data.List
w@(y@(f,d,e):z)§x@[a,b,c]|e<=d=z§x|a+b>d=(f,d,e`min`a):((f,a+b,e):z)§x
w§[a,b,c]=(c,a,a+b):w
r(c,a,b)=replicate(b-a)c
f l=shows(l!!0)" "++shows(length l)"\n"
main=interact$(>>=f).group.(>>=r).sort.foldl'(§)[].map(map read.words).lines

这样做不太正确,因为它花费了太多时间和堆栈...但是它确实产生了正确的答案:

$> ghc -O3 --make -rtsopts -with-rtsopts -K32m 3095-PaintTheFence.hs 
Linking 3095-PaintTheFence ...

$> ./3095-gen 1000000 | time ./3095-PaintTheFence
1 1
3 4854
6 523
13 1
16 11
18 416
22 7
24 3920
25 96
31 10249
32 241
37 1135
45 10
57 758
62 2348
65 11
66 7422
78 6
85 13361
87 3833
88 187
91 46
93 7524
96 45436
       43.99 real        43.42 user         0.46 sys

  • 编辑(294→269)replicate,它group是计数涂料的有效方法,并且比自定义功能所需的代码更少s
  • 编辑(269→261)无需max通话
  • 编辑(261→251)无需剔除0 in f

这是太慢了
亚历山德鲁

是代码高尔夫球,不是吗?对于代码高尔夫球,通常,诸如“合理时间”之类的限制意味着对于目标输入大小,它不会花费几天。是否有一些标准可以使37秒(另一个答案的时间)还可以,但是44秒太慢了?如果您愿意,我可以将我的时间花在更快的CPU上!
MtnViewMark

在这种情况下,应该花几秒钟的时间*语言减速。如果我错了,请纠正我,但是Haskell是否比Python快得多?(这就是为什么我没有对Keith的解决方案投反对票的原因)。张贴了C蛮力解决方案,该解决方案花了大约相同的时间,根据规则,这是不允许的。
亚历山德鲁

测得的同一台机器上,它确实比Python的解决方案运行得更快。Python解决方案在我的计算机上耗时133.556秒。Haskell解决方案的速度提高了3倍。另外,请注意,此Haskell解决方案不是“强力”解决方案(我猜您的意思是简单地在墙的长度上构建颜色阵列。)
MtnViewMark 2011年

0

Perl-148字节

似乎Perl是最好的选择。易于编码,更快更短。;)

码:

#!perl -n
($a,$b,$c)=split;substr($s,$a,$b,chr($c)x$b)}BEGIN{$s="z"x102400}{$s=~s/([^z])\1*/$H{$1}+=length$&/ge;print ord," $H{$_}
"for sort keys%H

时间:

time ./gen 1000000 | perl paint.pl
...
real    0m9.767s
user    0m10.117s
sys 0m0.036s

0
$ cat input.txt
0 3 1
2 4 2
1 2 3
0 4 1
7 3 5

$ cat input.txt  | perl -anE '@a[$F[0]..$F[0]+$F[1]]=($F[2])x$F[1];END{$i[$_]++for@a;$i[$_]&&say"$_ $i[$_]"for 1..$#i}'
1 4
2 1
5 3


$ cat input2.txt
500 1000 1
0 2000 2

$ cat input2.txt  | perl -anE '@a[$F[0]..$F[0]+$F[1]]=($F[2])x$F[1];END{$i[$_]++for@a;$i[$_]&&say"$_ $i[$_]"for 1..$#i}'
2 2000

0

JavaScript,183个字符,1.3秒

可悲的是,我确实不得不削减了标准输入/输出的部分,JavaScript不支持该部分。取而代之的是,我从文件上传中获取输入<input>(尽管我可能应该这样做,但我不将其计入字符数)。

这是非高尔夫版本。这个函数需要完整的输入字符串...全部14MB!这是需要1.3秒的时间;高尔夫版本大约需要两倍的时间-但仍然胜过其他解决方案!有趣的是,它在Firefox中的速度是在Chrome中的两倍。现场演示

function Q(input) {

    var c = [];
    var l = input.trim().split(/\s/g);
    input = null
    var length = l.length;

    // Loop through each meter of the wall...
    for (var i = 0; i <= 102400; i++) {

        // ...and loop through each of the friends, finding
        // the last one who painted this meter...
        for (var j = length; j > 0; ) {
            j -= 3;

            // Start = +l[j]
            // Length = +l[j + 1]
            // Color = +l[j + 2]

            var S = +l[j];      
            if (S <= i && +l[j + 1] + S > i) {

                // ...and incrementing the color array.
                var C = +l[j + 2];
                if (!++c[C])
                    c[C] = 1;

                break;
            }
        }
    }

    console.log(c.map(function (a,b) {return b + ' ' + a}).filter(function (a) { return a }).join('\n'));
}

这是高尔夫版本。

function G(i){l=i.trim(c=[]).split(/\s/)
for(i=0;i<102401;i++)for(j=l.length;j>0;)if(l[j-=3]<=i&&i-l[j+1]<l[j]){if(!++c[l[j+2]])c[l[j+2]]=1
break}for(k in c)console.log(k+' '+c[k])}

屏幕截图

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.