bzoj-2093 Frog

题意:

数轴上有n个点,有一个青蛙在这些点上跳;

规则是每次向距当前点第k小的点跳,如果有相同距离则向下标较小的跳;

求从每个点出发跳了m次后在哪里;

1<=k<n<=1000000,m<=10^18,1<=坐标<=10^18;

题解:

我真是不知道这题出long long 的意义是啥。。

数据范围比较极限,略卡,加点读入优化啥的卡常数才过;

不过Poi上还是挺良心的,除了偶尔乱入的波兰文以外;

这题后半部分显然就是置换的快速幂;

主要问题是前面如何求出一次的置换;

对于一个点来说,离它k小的点是一个区间;

如果包括这个点的话,那么这个区间有k+1个元素;

这个东西满足一个单调性,第一个点的区间l=1,r=k+1;

而当前点变化之后,区间长度不会改变,只会右移;

那么每次l++,r++就好了;

具体画个图理解一些就好,单调性啥的真是太神了!

这一步是O(n)的,后面置换快速幂O(nlogn);

所以一百万为啥要出带log的算法啊(摔);

代码:

#include<cctype>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 1401420
#define LEN 1<<16
using namespace std;
typedef long long ll;
int n;
struct node
{
    int to[N];
    int &operator[](int x)
    {
        return to[x];
    }
    void operator *=(node &a)
    {
        static node temp;
        for(int i=1;i<=n;i++)
            temp[i]=a[to[i]];
        memcpy(this,&temp,sizeof(node));
    }
}T,ans;
ll a[N];
void pow(node &x,ll y)
{
    while(y)
    {
        if(y&1)
            ans*=x;
        x*=x;
        y>>=1;
    }
}
char getc()
{
    static char *S,*T,buf[LEN];
    if(S==T)
    {
        T=(S=buf)+fread(buf,1,LEN,stdin);
        if(S==T)
            return EOF;
    }
    return *S++;
}
ll read()
{
    static char ch;
    static ll D;
    while(!isdigit(ch=getc()));
    for(D=ch-'0';isdigit(ch=getc());)
        D=D*10+ch-'0';
    return D;
}
int main()
{
    int i,j,k,l,r;
    ll m;
    scanf("%d%d%lld",&n,&k,&m);
    for(i=1;i<=n;i++)
        a[i]=read();
    for(i=1,l=1,r=k+1;i<=n;i++)
    {
        ans[i]=i;
        T[i]=a[i]-a[l]>=a[r]-a[i]?l:r;
        while(r<n&&a[i+1]-a[l]>a[r+1]-a[i+1])
            l++,r++;
    }
    pow(T,m);
    for(i=1;i<=n;i++)
    {
        if(i!=1)    putchar(' ');
        printf("%d",ans[i]);
    }
    return 0;
}
时间: 2024-10-15 00:09:33

bzoj-2093 Frog的相关文章

BZOJ 2093: [Poi2010]Frog

Description 从一个点到达与他距离第 \(k\) 小的点,问从每个点跳 \(m\) 次到达那个点. Sol 队列+倍增. 保持队列里的元素个数为 \(k\) ,从前往后扫不难发现左右端点都是单调的. 求跳 \(m\) 次就是倍增了,滚一下数组. Code /************************************************************** Problem: 2093 User: BeiYu Language: C++ Result: Accep

BZOJ 题目整理

bzoj 500题纪念 总结一发题目吧,挑几道题整理一下,(方便拖板子) 1039:每条线段与前一条线段之间的长度的比例和夹角不会因平移.旋转.放缩而改变,所以将每条轨迹改为比例和夹角的序列,复制一份翻转后的序列,直接上AC自动机即可.注意特判 1125:hash+splay 1183:digit-product只可能是2,3,5,7的积,枚举digit-product进行dp即可 1301:每个点和每个边只能被删除一次,随便搞 1313:上下界最大流 1471:考虑容斥,枚举两条路径相交的位置

BZOJ 1013: [JSOI2008]球形空间产生器sphere

二次联通门 : BZOJ 1013: [JSOI2008]球形空间产生器sphere /* BZOJ 1013: [JSOI2008]球形空间产生器sphere 高斯消元 QAQ SB的我也能终于能秒题了啊 设球心的坐标为(x,y,z...) 那么就可以列n+1个方程,化化式子高斯消元即可 */ #include <cstdio> #include <iostream> #include <cstring> #define rg register #define Max

bzoj 3309 DZY Loves Math - 莫比乌斯反演 - 线性筛

对于正整数n,定义f(n)为n所含质因子的最大幂指数.例如f(1960)=f(2^3 * 5^1 * 7^2)=3, f(10007)=1, f(1)=0. 给定正整数a,b,求sigma(sigma(f(gcd(i,j)))) (i=1..a, j=1..b). Input 第一行一个数T,表示询问数. 接下来T行,每行两个数a,b,表示一个询问. Output 对于每一个询问,输出一行一个非负整数作为回答. Sample Input 4 7558588 9653114 6514903 445

【BZOJ】[HNOI2009]有趣的数列

[算法]Catalan数 [题解] 学了卡特兰数就会啦>_<! 因为奇偶各自递增,所以确定了奇偶各自的数字后排列唯一. 那么就是给2n个数分奇偶了,是不是有点像入栈出栈序呢. 将做偶数标为-1,做奇数标为+1,显然当偶数多于奇数时不合法,因为它压不住后面的奇数. 然后其实这种题目,打表就可知啦--QAQ 然后问题就是求1/(n+1)*C(2n,n)%p了,p不一定是素数. 参考bzoj礼物的解法. 看到网上清一色的素数筛+分解质因数解法,不解了好久,感觉写了假的礼物-- 后来觉得礼物的做法才比

洛谷 P2709 BZOJ 3781 小B的询问

题目描述 小B有一个序列,包含N个1~K之间的整数.他一共有M个询问,每个询问给定一个区间[L..R],求Sigma(c(i)^2)的值,其中i的值从1到K,其中c(i)表示数字i在[L..R]中的重复次数.小B请你帮助他回答询问. 输入输出格式 输入格式: 第一行,三个整数N.M.K. 第二行,N个整数,表示小B的序列. 接下来的M行,每行两个整数L.R. 输出格式: M行,每行一个整数,其中第i行的整数表示第i个询问的答案. 输入输出样例 输入样例#1: 6 4 3 1 3 2 1 1 3

BZOJ 1012: [JSOI2008]最大数maxnumber(线段树)

012: [JSOI2008]最大数maxnumber Time Limit: 3 Sec  Memory Limit: 162 MB Description 现在请求你维护一个数列,要求提供以下两种操作:1. 查询操作.语法:Q L 功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值.限制:L不超过当前数列的长度.2. 插入操作.语法:A n 功能:将n加上t,其中t是最近一次查询操作的答案(如果还未执行过查询操作,则t=0),并将所得结果对一个固定的常数D取模,将所得答案插入到数列

【BZOJ】【1016】【JSOI2008】最小生成树计数

Kruskal/并查集+枚举 唉我还是too naive,orz Hzwer 一开始我是想:最小生成树删掉一条边,再加上一条边仍是最小生成树,那么这两条边权值必须相等,但我也可以去掉两条权值为1和3的,再加上权值为2和2的,不也满足题意吗?事实上,如果这样的话……最小生成树应该是1和2,而不是1和3或2和2!!! 所以呢?所以对于一个图来说,最小生成树有几条边权为多少的边,都是固定的!所以我们可以做一遍Kruskal找出这些边权,以及每种边权出现的次数.然后,对于每种边权,比方说出现了$v_i$

【BZOJ】【2844】albus就是要第一个出场

高斯消元解XOR方程组 srO  ZYF  Orz 膜拜ZYF…… http://www.cnblogs.com/zyfzyf/p/4232100.html 1 /************************************************************** 2 Problem: 2844 3 User: Tunix 4 Language: C++ 5 Result: Accepted 6 Time:252 ms 7 Memory:2052 kb 8 *******

[BZOJ 1066] [SCOI2007] 蜥蜴 【最大流】

题目链接:BZOJ - 1066 题目分析 题目限制了高度为 x 的石柱最多可以有 x 只蜥蜴从上面跳起,那么就可以用网络流中的边的容量来限制.我们把每个石柱看作一个点,每个点拆成 i1, i2,从 i1 到 i2 连一条边,容量为这个石柱 i 的高度,即跳跃次数限制.来到这个石柱就是向 i1 连边,从这个石柱跳起就是从 i2 向外连边,这样只要从石柱 i 跳起就一定会消耗 i1 到 i2 的边的容量.如果 i 有蜥蜴,就从 S 到 i1 连一条容量为 1 的边,如果从石柱 i 能跳出边界,就从