zoukankan      html  css  js  c++  java
  • SECHS

    题目描述 

              对于给定的正整数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倍,进行下面的操作,直到填完为止。

        ~(≧▽≦)/~      

  • 相关阅读:
    oracle客户端plsql安装配置
    vue基础-vue-cli(vue脚手架
    ES5、6、7浅析
    webservice的使用
    使用intellj idea的hibernate生成注解实体类
    spring源码分析
    Total Eclipse(并查集)
    《大道至简》读后感
    2020年8月3日Java学习日记
    2020年8月2日Java学习日记
  • 原文地址:https://www.cnblogs.com/fengjunjie/p/6021125.html
Copyright © 2011-2022 走看看