免费注册 查看新帖 |

Chinaunix

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

用python来做这个题目,还真好玩! [复制链接]

论坛徽章:
5
丑牛
日期:2014-01-21 08:26:26卯兔
日期:2014-03-11 06:37:43天秤座
日期:2014-03-25 08:52:52寅虎
日期:2014-04-19 11:39:48午马
日期:2014-08-06 03:56:58
11 [报告]
发表于 2014-08-18 19:10 |只看该作者
这个题目 n = 100000
穷举不到半秒搞定还是可以接受的

i have a idea: count
呵呵,程序没还做出来

k在第几个位置?
example: k = 21 => 3
count 1 => total a个
count 2 => total b个

count k in 3 => c
k在第 a + b + c 的位置

效率: 没程序还没测试 ~ {:2_168:}

论坛徽章:
4
白羊座
日期:2013-11-05 10:26:09冥斗士
日期:2015-11-17 14:19:55白银圣斗士
日期:2015-11-17 15:13:0815-16赛季CBA联赛之新疆
日期:2016-04-01 09:10:58
12 [报告]
发表于 2014-08-18 22:45 |只看该作者
个人理解:这个原始题目的目的不是说不允许穷举,毕竟当数据量不是那么大的时候,穷举还是可以解决不少问题的。
所以,原始题目特意指定数据的量是在10^18的范围,来使穷举变得不那么合算。
题目的意思应当是:
让我们通过一个合适的递归分解,来解决一个数可能有哪些排列方式。
比如:当数据在100000内的时候,如果是w(data) = 1,那可能的方式就只有:
1, 10, 100, 1000, 10000, 100000, (共5个)
w(data)=2,可能的方式有:
2, 11, 101, 110......
从而让我们把问题转向数的可能的排列方式以及相应个数的递归分析上,然后,再应用第二条规则,找出第k个位置上的数是谁。

我想,这个是题目希望我们做的。  

论坛徽章:
4
金牛座
日期:2013-10-11 16:12:50卯兔
日期:2014-07-31 09:17:19辰龙
日期:2014-08-08 09:28:02狮子座
日期:2014-09-14 20:32:05
13 [报告]
发表于 2014-08-19 09:20 |只看该作者
回复 12# icymirror


    你说的这个方法我考虑过,但是如果楼主给一个2897575896这样的数字,而不是整数的时候,这个方法就复杂多了。

论坛徽章:
4
白羊座
日期:2013-11-05 10:26:09冥斗士
日期:2015-11-17 14:19:55白银圣斗士
日期:2015-11-17 15:13:0815-16赛季CBA联赛之新疆
日期:2016-04-01 09:10:58
14 [报告]
发表于 2014-08-19 09:40 |只看该作者
回复 13# ssfjhh
针对这个题目来说,没有特定的最好解决方案。
比如:就您刚刚给出的数字,如果数字的范围在0~3000000000 (2897575896快到结尾了)。
因为在枚举可能的排列时,要从1开始,除去列出所有可能组合之外,还需要记数当前一共出现了多少个排列。
这时,穷举反而不会在效率上比枚举所有可能的排列组合慢多少,反而更有可能是比较快的方法。
所以,我的说法只是针对题目的期望,并不是说什么样的方法是好的,呵呵。

论坛徽章:
5
丑牛
日期:2014-01-21 08:26:26卯兔
日期:2014-03-11 06:37:43天秤座
日期:2014-03-25 08:52:52寅虎
日期:2014-04-19 11:39:48午马
日期:2014-08-06 03:56:58
15 [报告]
发表于 2014-08-20 18:10 |只看该作者
呵呵,程序没还做出来,
but i have a 穷举法的优化代码!
come here!

http://bbs.chinaunix.net/forum.p ... id=4150973#lastpost
ssfjhh:
  1. 10000000, 10000 == 1m53.790s
  2. 100000, 10000     == 0m0.994s
复制代码
pitonas:
  1. 10000 10000000  == 0m47.963s
  2. 10000 100000   == 0m0.443s
复制代码

论坛徽章:
5
丑牛
日期:2014-01-21 08:26:26卯兔
日期:2014-03-11 06:37:43天秤座
日期:2014-03-25 08:52:52寅虎
日期:2014-04-19 11:39:48午马
日期:2014-08-06 03:56:58
16 [报告]
发表于 2014-08-21 11:01 |只看该作者
本帖最后由 pitonas 于 2014-08-21 12:17 编辑

这时,穷举法的优化:
比如:就 k, n = 22, 100 来说
22      100     11      24

1 .. 100 generate a new ARRAY

ARRAY
index0: undef
index1
1 10 100

index2
2 11 20

index3
3 12 21 30

index4
4 13 22 31 40

22 ( 2 + 2 = 4 ) => index4
index1 ~ index3 = 3 + 3 + 4 => 10
[SORT!!] index4 [ 13 22 31 4 40 ]
22 's index = 1
10 + 1 = [ index = 11 ]

index5
5 14 23 32 41 50

index1 ~ index5
3 + 3 + 4 + 5 + 6 = 21 ( 21 < 22 )
index1 ~ index6
21 + 7 = 28 ( 28 > 22 )
22 IN index6
[SORT!!] index6

index6
15 24 33 42 51 6 60 [SORTED!!]

index 22 = count 21 + index6[1]
index6[1] => [ number = 24 ]

......

论坛徽章:
5
丑牛
日期:2014-01-21 08:26:26卯兔
日期:2014-03-11 06:37:43天秤座
日期:2014-03-25 08:52:52寅虎
日期:2014-04-19 11:39:48午马
日期:2014-08-06 03:56:58
17 [报告]
发表于 2014-08-21 14:16 |只看该作者
本帖最后由 pitonas 于 2014-08-21 12:14 编辑

因此,这是我的代码 ;
  1. #!/usr/bin/perl

  2. while (<DATA>) {
  3.     my ( $K, $N ) = split;
  4.     my ( $K_sum, @mind );
  5.     $K_sum += $_ for split //, $K;

  6.     for my $i ( 1 .. $N ) {
  7.         my $sum;
  8.         $sum += $_ for split //, $i;
  9.         push @{ $mind[$sum] }, $i;
  10.     }

  11.     my $index = 0;
  12.     $index += @$_ for @mind[ 1 .. $K_sum - 1 ];

  13.     for my $n ( sort { $a cmp $b } @{ $mind[$K_sum] } ) {
  14.         last if $K == $n;
  15.         $index++;
  16.     }

  17.     my ( $INDEX, $I );

  18.     for my $i ( @mind[ 1 .. $#mind ] ) {
  19.         last if $INDEX > $K;
  20.         $I++;
  21.         $INDEX += @$i;
  22.     }

  23.     $mind[$I] = [ sort { $a cmp $b } @{ $mind[$I] } ];
  24.     my $number = $mind[$I][ $K - $INDEX ];
  25.     print join( "\t", $K, $N, $index, $number ), "\n";
  26. }
  27. __DATA__
  28. 10000 100000
复制代码
change it for speed test
__DATA__
10000 10000000

论坛徽章:
7
荣誉版主
日期:2011-11-23 16:44:17子鼠
日期:2014-07-24 15:38:07狮子座
日期:2014-07-24 11:00:54巨蟹座
日期:2014-07-21 19:03:10双子座
日期:2014-05-22 12:00:09卯兔
日期:2014-05-08 19:43:17卯兔
日期:2014-08-22 13:39:09
18 [报告]
发表于 2014-08-22 13:40 |只看该作者
等楼主放代码

论坛徽章:
26
2015亚冠之胡齐斯坦钢铁
日期:2015-06-25 21:40:202015亚冠之柏斯波利斯
日期:2015-08-31 17:03:192015亚冠之柏斯波利斯
日期:2015-11-07 13:10:00程序设计版块每日发帖之星
日期:2015-11-10 06:20:00每日论坛发贴之星
日期:2015-11-10 06:20:00程序设计版块每日发帖之星
日期:2015-11-26 06:20:00程序设计版块每日发帖之星
日期:2015-12-02 06:20:00黄金圣斗士
日期:2015-12-07 17:57:4615-16赛季CBA联赛之天津
日期:2015-12-23 18:34:14程序设计版块每日发帖之星
日期:2016-01-02 06:20:00程序设计版块每日发帖之星
日期:2016-01-06 06:20:00每日论坛发贴之星
日期:2016-01-06 06:20:00
19 [报告]
发表于 2014-08-24 11:40 |只看该作者
我想等楼主来放代码

论坛徽章:
11
2015年迎新春徽章
日期:2015-03-04 09:55:282017金鸡报晓
日期:2017-02-08 10:39:4215-16赛季CBA联赛之辽宁
日期:2016-12-15 10:24:1715-16赛季CBA联赛之佛山
日期:2016-11-30 09:04:2015-16赛季CBA联赛之江苏
日期:2016-04-29 15:56:1215-16赛季CBA联赛之同曦
日期:2016-04-12 13:21:182016猴年福章徽章
日期:2016-02-18 15:30:3415-16赛季CBA联赛之山东
日期:2016-02-16 11:37:52每日论坛发贴之星
日期:2016-02-07 06:20:00程序设计版块每日发帖之星
日期:2016-02-07 06:20:0015-16赛季CBA联赛之新疆
日期:2018-01-09 16:25:37
20 [报告]
发表于 2014-08-24 15:18 |只看该作者
回复 10# ssfjhh


那请你计算一下1到109999999999999999产生权中的第 7684342753的位置和第7684342753个位置的权是哪个数产生的
不知你没有看问题还是思维有问题啊,题目很明白的说明不要用穷举
当然机器好一点可以掩盖一些低劣的算法,也能让IT的门槛降低,退潮后就知道谁在裸泳了

我的电脑很破,是80年代产的古董,算到10万就吃不消了
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP