【poj1019】 Number Sequence

http://poj.org/problem?id=1019 (题目链接)

题意

  给出一个数:1 12 123 1234 12345 123456 1234567 12345678 123456789 12345678910(当然中间是没有空格的)求它从左往右第n位上是多少。

solution

  水题一道。我们可以发现,这个数可以分成若干串数,记为i,那么每串数i就是从1~i。我们可以用数组x[i]来记录串i所占的空间也就是位数,数组sum[i]来记录串i的末尾位于整个串中的哪个位置。

  对于每次输入的数n,我们先二分找到它位于哪个串中,之后再递归到当前串中处理。在这里我们发现x[i]=x[i-1]+log10(i)+1,所以就很好做了。

代码

// poj1019
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<vector>
#define MOD 1000000007
#define inf 2147483640
#define LL long long
#define free(a) freopen(a".in","r",stdin);freopen(a".out","w",stdout);
using namespace std;
inline LL getint() {
    LL x=0,f=1;char ch=getchar();
    while (ch>'9' || ch<'0') {if (ch=='-') f=-1;ch=getchar();}
    while (ch>='0' && ch<='9') {x=x*10+ch-'0';ch=getchar();}
    return x*f;
}

LL x[1000000],sum[1000000];
vector<int> v;

int main() {
    int T;scanf("%d",&T);
    sum[0]=x[0]=0;
    for (int i=1;i<=9;i++) x[i]=x[i-1]+1;
    for (int i=10;i<=99;i++) x[i]=x[i-1]+2;
    for (int i=100;i<=999;i++) x[i]=x[i-1]+3;
    for (int i=1000;i<=9999;i++) x[i]=x[i-1]+4;
    for (int i=10000;i<=99999;i++) x[i]=x[i-1]+5;
    for (int i=1;i<=99999;i++) sum[i]=sum[i-1]+x[i];
    while (T--) {
        LL n;scanf("%lld",&n);
        int i=lower_bound(sum,sum+100000,n)-sum;
        n-=sum[i-1];
        i=lower_bound(x,x+i+1,n)-x;
        n-=x[i-1];
        v.clear();
        while (i) {
            v.push_back(i%10);
            i/=10;
        }
        reverse(v.begin(),v.end());
        printf("%d
",v[n-1]);
    }
    return 0;
}

  

原文地址:https://www.cnblogs.com/MashiroSky/p/5914633.html