标签:res 相等 procedure int while 证明 需要 for ...
题目描述
对于给定的正整数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倍,进行下面的操作,直到填完为止。
~\(≧▽≦)/~
标签:res 相等 procedure int while 证明 需要 for ...
原文地址:http://www.cnblogs.com/fengjunjie/p/6021125.html