免费注册 查看新帖 |

Chinaunix

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

【问题】$K 的 INDEX 是几? [复制链接]

论坛徽章:
145
技术图书徽章
日期:2013-10-01 15:32:13戌狗
日期:2013-10-25 13:31:35金牛座
日期:2013-11-04 16:22:07子鼠
日期:2013-11-18 18:48:57白羊座
日期:2013-11-29 10:09:11狮子座
日期:2013-12-12 09:57:42白羊座
日期:2013-12-24 16:24:46辰龙
日期:2014-01-08 15:26:12技术图书徽章
日期:2014-01-17 13:24:40巳蛇
日期:2014-02-18 14:32:59未羊
日期:2014-02-20 14:12:13白羊座
日期:2014-02-26 12:06:59
41 [报告]
发表于 2014-08-27 01:16 |只看该作者
本帖最后由 jason680 于 2014-08-27 17:04 编辑

回复 9# pitonas

[ 问题的限制 ] : 不能使用穷举法 <==不是不能使用,是在数值很大时......太慢了......
(这跟(破)解码一样,不是不能用暴力解,但如果暴力解
要花好几年,几十年,上百年,上千年,上万年....这个解,有解跟没解是一样的....)

所以我们要找出方法,用产生出来的....
例:K =4,N =200  

W=1,有
1,10,100

W=2
101, 11, 110
2, 20, 200

W=3
102, 111, 12, 120
201(X), 21, 210(X)
3, 30, 300(X)

W=4
103, 112, 121, 13, 130
202(X), 211(X), 22, 220(X)
301(X), 31, 310(X)
4, 40, 400(X)

Note: for (X), the length match the N(Number 200) and summary match the W, but value is over than Number(200)
ex: 201(X)
1. the length is 3 and match the length of Number(200)
2. and summary(2+0+1=3) is matched the W=3,
3. but value(201) is over than 200

---------------------------------------------
$ cat nk.pl
#!/usr/bin/perl
use strict;
use warnings;

my $sCnt;
my $X=0;
my $sMatch = 0;
my $sCount_str = "";
my $sLocation  = "";
while (<DATA>) {
  my ( $sK, $sN ) = split;

  $sCnt = 0;
  kn($sK, $sN);  

}

sub kn{
  my($sK, $sN, $sW, $sStr, $sSum) = @_;
  
  my $sDigit = int(log($sN)/log(10)+1);

  if(!defined $sW){
    foreach(1 .. $sK){
      kn($sK, $sN, $_);
    }
    return;
  }
  if(!defined $sStr){
    my $sStart = ($sW>9 ? 9 : $sW);
    foreach(1 .. $sStart){
      kn($sK, $sN, $sW, $_, $_);
    }
    return;
  }
  my $sLen = length($sStr);
  return if($sLen > $sDigit);

  my $sDiff = $sW - $sSum;
  print STDERR "debug K=$sK, N=$sN($sDigit), W=$sW, str=$sStr($sLen), Sum=$sSum,Diff=$sDiff\n";
  return if($sN < $sStr);
  #exit if($X++ > 20000);
  if($sDiff > 0){
    if($sDigit - $sLen > 1){
      foreach(0 .. ($sDiff>9?9:$sDiff)){
        last if($sW < $sSum+$_);
        kn($sK, $sN, $sW, "$sStr$_", $sSum+$_);
      }
    }
    elsif($sDigit - $sLen == 1){
      
      print STDERR "$sK, $sN, $sW, $sStr$sDiff, " . ($sSum+$sDiff) ."\n";
      kn($sK, $sN, $sW, "$sStr$sDiff", $sSum+$sDiff);
    }
    return;
  }
  if($sDiff == 0 ){
    foreach( 0 .. $sDigit - $sLen){
      if($sStr <= $sN){
        ++$sCnt;
        print STDERR "***match $sCnt: $sK, $sN, $sW, $sStr, $sSum\n";
        printf("\nmatch %2d: ",$sCnt) if($sCnt %10 == 1);
        print sprintf( "%5d, ",$sStr);
        if($sCnt == $sK){
           $sCount_str = "\n count $sK is number $sStr\n";
          ++$sMatch;
        }
        if($sStr == $sK){
          $sLocation = "\n the number $sStr is on $sCnt\'th\n";
          ++$sMatch;
        }  
        if($sMatch == 2){
          print "$sCount_str $sLocation";
          exit;
        }
        $sStr .= "0";
      }
    }
      
  }
}

__DATA__
4 200

$ perl nk.pl 2> nk.log

match  1:     1,    10,   100,   101,    11,   110,     2,    20,   200,   102,
match 11:   111,    12,   120,    21,     3,    30,   103,   112,   121,    13,
match 21:   130,    22,    31,     4,
count 4 is number 101

the number 4 is on 24'th

Note: please check the nk.log for detail information...


   

论坛徽章:
0
42 [报告]
发表于 2014-08-27 11:34 |只看该作者
回复 41# jason680

膜拜@jason680

學習了

论坛徽章:
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
43 [报告]
发表于 2014-08-27 16:42 |只看该作者
我读到,但看起来不是那么容易,
谢谢你, my $师傅,我读它,一次 + 一次

回复 41# jason680


   
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP