codeforces 13EE. Holes(分块&动态树)

E. Holes
time limit per test
1 second
memory limit per test
64 megabytes
input
standard input
output
standard output

Little Petya likes to play a lot. Most of all he likes to play a game «Holes». This is a game for one person with following rules:

There are N holes located in a single row and numbered from left to right with numbers from 1 to N. Each hole has it's own power (hole number i has the power ai). If you throw a ball into hole i it will immediately jump to hole i + ai, then it will jump out of it and so on. If there is no hole with such number, the ball will just jump out of the row. On each of the M moves the player can perform one of two actions:

  • Set the power of the hole a to value b.
  • Throw a ball into the hole a and count the number of jumps of a ball before it jump out of the row and also write down the number of the hole from which it jumped out just before leaving the row.

Petya is not good at math, so, as you have already guessed, you are to perform all computations.

Input

The first line contains two integers N and M (1 ≤ N ≤ 1051 ≤ M ≤ 105) — the number of holes in a row and the number of moves. The second line contains N positive integers not exceeding N — initial values of holes power. The following M lines describe moves made by Petya. Each of these line can be one of the two types:

  • 0 a b
  • 1 a
Type  0 means that it is required to set the power of hole  a to  b, and type  1 means that it is required to throw a ball into the  a-th hole. Numbers  a and  b are positive integers do not exceeding  N.
Output

For each move of the type 1 output two space-separated numbers on a separate line — the number of the last hole the ball visited before leaving the row and the number of jumps it made.

Sample test(s)
input
8 5
1 1 1 1 1 2 8 2
1 1
0 1 3
1 1
0 3 4
1 2
output
8 7
8 5
7 3
题意:
给你n个编号1到N的洞(N<=1e5)。每个洞有个能量值po[i]。当你在i号洞里放一个球时.这个球会弹到i+po[i]号洞内。然后弹到i+po[i]+po[i+po[i]]。。。。也就是说球每到一个洞i就会弹到i+po[i]号洞内。现在有两种操作。
1.0 a b。把a号洞的po改成b
2.1 a。询问当把球放到a号洞内时。它是从几号洞弹出界的。和它一共弹了几次。

思路:
如果题目没有修改操作。这题就会很简单。我们只需要把每个洞的出界点和弹跳数预处理出来就可以了。现在关键是怎么处理修改操作。如果还是按照上述预处理方式肯定时间复杂度下不来。所以我们要想办法使每次修改操作更新尽量少的信息。于是可以想到分块处理。就是把整个序列分成sqrt(N)块。序列中每个节点记录next[i]表示i结点要跳到下个块的位置。ed[i]表示放到i号洞时的出界位置。st[i]表示。i跳到下个块需要的步数。这样预处理后每次修改操作只会影响同块内且标号比当前小的位置的信息。所以最多修改sqrt(n)个位置的信息。对于每次查询操作。由于结点指针只会指向不同的块所以只需要顺着指针统计一遍就好了。最多跳sqrt(n)次。总时间发杂度O(M*sqrt(N))在可以接受的范围内。
详细见代码:
#include<bits/stdc++.h>
using namespace std;
const int INF=0x3f3f3f3f;
const int maxn=100010;
typedef long long ll;
int st[maxn],next[maxn],ed[maxn],po[maxn];
int block,n,m,ans,ansp;
void update(int x,int y)
{
    if(y>n)
        ed[x]=x,st[x]=1,next[x]=y;
    else
    {
        ed[x]=ed[y];//ed[x]为从x处放球的终点
        if(x/block==y/block)
            next[x]=next[y],st[x]=st[y]+1;//next[i]表示i跳到另外的块的位置
        else
            next[x]=y,st[x]=1;
    }
}
void qu(int x)
{
    ans=0;
    while(1)
    {
        ans+=st[x];
        if(next[x]>n)
        {
            ansp=ed[x];
            break;
        }
        x=next[x];
    }
}
int main()
{
    int i,cmd,a,b;

    while(~scanf("%d%d",&n,&m))
    {
        block=ceil(sqrt(1.0*n));
        for(i=1;i<=n;i++)
            scanf("%d",&po[i]);
        for(i=n;i>=1;i--)
            update(i,i+po[i]);
        while(m--)
        {
            scanf("%d%d",&cmd,&a);
            if(cmd)
            {
                qu(a);
                printf("%d %d\n",ansp,ans);
            }
            else
            {
                scanf("%d",&po[a]);
                b=(a/block)*block;
                b=max(b,1);
                for(i=a;i>=b;i--)
                    update(i,i+po[i]);
            }
        }
    }
    return 0;
}


你可能感兴趣的:(c,算法,ACM)