题目描述 

          对于给定的正整数N,我们把[1, N]中的整数按照字符串的字典序排序得到N 项数列A(N)。

          例如,N = 11的时候,A(N) = {1, 10, 11, 2, 3, 4, 5, 6, 7, 8, 9}。二元函数 Q(N, K)的定义域为N,

          K∈Z+ 且 N≥K,其值为K 在A(N) 中的位置。例如从上面给出的A(11)中可以看出Q(11, 2) = 4。

          现在你的任务是,对于给定的正整数 K 和M,求最小的正整数N 满足Q(N, K) = M。
输入格式
          仅有一行,包含两个正整数K 和M。
输出格式
          输出一个正整数N 表示答案。如果不存在这样的N,输出0。
样例输入 1
  12 7
样例输出1
  102
样例输入 2
  100000001 1000000000
样例输出2
  100000000888888879
数据范围与约定
  对于30% 的数据,满足M, K≤100。
  对于 100% 的数据,满足1≤M, K≤10^9。
乍一看.......没思路,但是好像是数学;
再乍......果然是数学,没思路(@_@);
先说一下,这道题最后是我看题解会的,这里就讲一下题解的思路(题解太简单了,简直虐尽天下像我一样的蒟蒻)

首先,先和大家普及一下有关字典序的知识,
对于数字n,它第一次出现在数列中时,它的位置的计算方法(以12345为例):
先拆位,将12345拆为(1),(12),(123),(1234),(12345)。
然后用每一位的数减去与其位数相等的最小数+1,为其在这一位上的位置
1——1 (1-1+1=1)
12——10 (12-10+1=3)
123——100 (123-100+1=24)
1234——1000 (1234-1000+1=235)
12345——10000 (12345-10000+1=2346)
最后,将得到的所有数加起来,得到 (1+3+24+235+2346=2609);
就是这个数第一次出现时他的位置(证明请看推导过程)
然后......在代码里讲吧

program ex02;
var ten:array[0..30] of int64;
      m,k,p:int64;
procedure init; //读入,预处理
var i:longint;
begin 
  ten[0]:=1;
  for i:=1 to 18 do ten[i]:=ten[i-1]*10;
  readln(m,k);
end;

function rank(x:int64):int64; //求某数在数列中第一次出现的位置;(用上面公式)
var i:int64;
begin 
  i:=1;
  while ten[i]<=x do inc(i); //每一位分别处理,再依次相加;
  exit(x-ten[i-1]+1);
end;

function solve:int64;
var i,len,r,n:int64;
begin
  n:=m; r:=0;
  while n>0 do //求输入的数在数列中第一次出现的位置;
  begin 
    inc(r,rank(n));
    n:=n div 10;
  end;
  if r=k then exit(m); //正好相等,直接输出;
  if r>k then exit(0); //小,不可能达成;
  len:=0; n:=m;
  while n>0 do //求位数
  begin 
    inc(len);
    n:=n div 10;
  end;
  dec(k,r); i:=0; //求中间差了几个数
  while true do 
  begin
    inc(i);
    if k<=rank(ten[i]*m)-1 then exit(k+ten[i+len-1]-1); //关键的一步,,,卡了我好长时间,,,讲解在下面
    dec(k,rank(ten[i]*m-1));
  end;
end;

function check(m:int64):int64;
begin
  check:=0;
  while m>=1 do
  begin 
    if (m<>1) and (m mod 10<>0) then exit(-1);
    m:=m div 10;
    inc(check);
  end;
end;
begin
  assign(input,'sec.in'); reset(input);
  assign(output,'sec.out')' rewrite(output);
  init;
  p:=check(m);
  if (p>0) then 
  begin 
    if (p=k) then 
      writeln(m) 
    else 
      writeln(0); 
  end
  else 
    writeln(solve);
  close(input);
  close(output);
end.

 

 

key点

       我们知道要将这个题所求的范围置于某一特定的位数

       那么,这个位数则么得到?

       易得一点:1,10,100,1000,10000.....出现后,它们前面不会多出数字,那么,我们只需要知道rank(k*10^d)>m>rank(10^l) 的l的值

       那么,m-rank(k)就是需要在k前面插入的数的个数

       如果在 10^l~k*10^d中的数用不完就填满了,直接输出, 但如果不够,就将其扩大10倍,进行下面的操作,直到填完为止。

    ~\(≧▽≦)/~      

posted on 2016-11-01 21:40  艾路雷朵  阅读(371)  评论(0编辑  收藏  举报