题目重点是每次不能跳过两个棋子

即对于每一个棋子的状态(a,b,c) (a<b<c)

最多有两种移动的方式

1.中间往两边跳 (a,b,c)-->(2b-a,a,c)或(a,c,2b-c)

2.a或c往中间跳 当然要满足不跳过两个棋子

b-a<c-b a可以跳过b (a,b,c)-->(b,2b-a,c)

c-b<b-a c可以跳过b (a,b,c)-->(a,2b-c,b)

当然当c-b=b-a时就不能再往中间跳了

此时可以想到对于一个状态(a,b,c)如果一直往中间跳 最终的状态一定是一定的

于是可以把这个最终的状态看做是根节点 中间向左向右跳的状态分别是左右节点

这样就是一棵二叉树了,而询问两个状态能否相互跳到就是树上距离了

于是回顾lca求树上距离的过程,我们首先要确定一个状态的k层祖先的状态是什么

如果暴力跳的话就会超时,所以我们想一种特殊的情况

(1,100000000,100000001) 显然暴力跳会一直跳1的长度是不行的

但我们看这时b-a很大但c-b很小就意味着接下来会很多次都是c往b跳

能跳几次呢?因为棋子实际上没有区别 假如c跳过b 那么实际上就是c和b一起向左平移了c-b的距离

所以c往b跳的次数就是(b-a)/(c-b)

这样对于本题就可以达到取模gcd的速度快速求出一个状态的k层祖先状态

这样用lca的手法先让两点到达同一高度,再二分一个距离k,如果两种状态的k层祖先一直 则将k缩小,反之增大求出他们的最近公共祖先

#include<bits/stdc++.h>
using namespace std;
int ans,a,b,c,x,y,z,x1,yy1,z1,a1,b1,c1,t1,t2;
int read()
{
int f=1,x=0;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(x=x*10+ch-48,ch=getchar(),ch>='0'&&ch<='9');
return x*f;
}
void SWAP(int &a,int &b,int &c){if(a>b){swap(a,b);if(a>c)swap(a,c);}if(b>c)swap(b,c);}
void back(int &a,int &b,int &c,int k)//将状态a,b,c回到k步前的状态 即它的k层祖先
{
int A=b-a,B=c-b;
if(A<B)
{
int t=B/A;if(B%A==0)t--;
if(t>=k){a=a+k*A;b=b+k*A;}
else {a=a+t*A;b=b+t*A;back(a,b,c,k-t);}
}else
{
int t=A/B;if(A%B==0)t--;
if(t>=k){c=c-k*B;b=b-k*B;}
else {c=c-t*B;b=b-t*B;back(a,b,c,k-t);}
}
}
int getfa(int &a,int &b,int &c)//寻找状态a,b,c的根节点
{
int A=b-a,B=c-b;
if(A==B)return 0;
if(A<B)
{
int t=B/A;if(B%A==0)t--;
a=a+t*A;b=b+t*A;
return t+getfa(a,b,c);
}else
{
int t=A/B;if(A%B==0)t--;
c=c-t*B;b=b-t*B;
return t+getfa(a,b,c);
}
}
int comp(int a,int b,int c,int x,int y,int z){if(a==x&&b==y&&c==z)return 1;else return 0;}
int main()
{
a=read();b=read();c=read();
x=read();y=read();z=read();
SWAP(a,b,c);SWAP(x,y,z);
x1=x;yy1=y;z1=z;a1=a;b1=b;c1=c;
t1=getfa(x1,yy1,z1);t2=getfa(a1,b1,c1);
if(!comp(x1,yy1,z1,a1,b1,c1)){puts("NO");return 0;}
if(t1<t2) back(a,b,c,t2-t1);else back(x,y,z,t1-t2);
int l=0,r=min(t1,t2);
while(l<=r)
{
int mid=(l+r)>>1;
x1=x;yy1=y;z1=z;a1=a;b1=b;c1=c;
back(x1,yy1,z1,mid);back(a1,b1,c1,mid);
if(comp(a1,b1,c1,x1,yy1,z1)) ans=mid,r=mid-1;else l=mid+1;
}
puts("YES");
printf("%d",2*ans+max(t2,t1)-min(t2,t1));
return 0;
}
/*
(a,b,c) if b-a<c-b --> (b,2b-a,c)
if c-b<b-a --> (a,2b-c,b) */

[luogu]P1852跳跳棋的更多相关文章

  1. P1852 跳跳棋 [LCA思想+二分答案]

    题目描述 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 我们用跳跳棋来做一个简单的游戏:棋盘上有\(3\)颗棋子,分别在\(a,b,c\)这三个位置.我们要通过最少的跳动 ...

  2. 【题解】P1852 跳跳棋

    link 题意 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子.棋盘上有3颗棋子,分别在 \(a,b,c\) 这三个位置.我们要通过最少的跳动把他们的位置移动成 \(x,y, ...

  3. 洛谷 P1852 [国家集训队]跳跳棋 解题报告

    P1852 [国家集训队]跳跳棋 题目描述 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在\(a\),\(b\), ...

  4. P1852 [国家集训队]跳跳棋

    P1852 [国家集训队]跳跳棋 lca+二分 详细解析见题解 对于每组跳棋,我们可以用一个三元组(x,y,z)表示 我们发现,这个三元组的转移具有唯一性,收束性 也就是说,把每个三元组当成点,以转移 ...

  5. 洛谷 P1852 [国家集训队] 跳跳棋

    题目描述 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.我们要通过最少的跳动把他们的位置移动 ...

  6. 【洛谷】1852:[国家集训队]跳跳棋【LCA】【倍增?】

    P1852 [国家集训队]跳跳棋 题目背景 原<奇怪的字符串>请前往 P2543 题目描述 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 我们用跳跳棋来做一个 ...

  7. 【LCA】bzoj 2144:跳跳棋

    2144: 跳跳棋 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 248  Solved: 121[Submit][Status][Discuss] ...

  8. bzoj2144 【国家集训队2011】跳跳棋

    Description 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子.我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.我们要通过最少的跳动把他 ...

  9. 跳跳棋(9018_1563)(BZOJ_2144)

    题目: Hzwer的跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 某一天,黄金大神和cjy用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.他们要 ...

随机推荐

  1. vue 父子组件

    组件 什么是组件? 组件 (Component) 是 Vue.js 最强大的功能之一.组件可以扩展 HTML 元素,封装可重用的代码.在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊 ...

  2. NFS客户端阻塞睡眠问题与配置调研

    Linux NFS客户端需要很小心地配置,否则在NFS服务器崩溃时,访问NFS的程序会被挂起,用ps查看,进程状态(STAT)处于D,意为(由于IO阻塞而进入)不可中断睡眠(如果是D+,+号表示程序运 ...

  3. 2017第十三届湖南省省赛A - Seating Arrangement CSU - 1997

    Mr. Teacher老师班上一共有n个同学,编号为1到n. 在上课的时候Mr. Teacher要求同学们从左至右按1, 2, …, n的顺序坐成一排,这样每个同学的位置是固定的,谁没来上课就一目了然 ...

  4. Solr基本操作

    /update 使用/update进行索引维护,进入Solr管理界面SolrCore下的Document下: 我们进行更新操作可以用json和xml多种格式,这里以xml格式为例说明.先来看看界面上的 ...

  5. ORA-12052: cannot fast refresh materialized view

    SQL> execute dbms_mview.refresh ('TX_FAIL_LOG_DAY_MV', 'f'); BEGIN DBMS_MVIEW.REFRESH ('TX_FAIL_L ...

  6. 苹果笔记本充不进电怎么办_macbook充不进电解决办法

    使用苹果Macbook的用户可能会遇到这种情况,使用一段时间后自己的苹果笔记本充不进电了,虽然充电器指示灯依然亮着,但是电池电脑一直充不进去,断开充电器后就直接关机的情况.通常碰到这种情况,很多用户都 ...

  7. 微信小程序细节

    微信小程序开发几个细节: 1.界面传值 ①全局参数传值 <!--结果--> <view wx:for="{{data}}" class="case pr ...

  8. Java版 家政服务 社区服务 家装服务平台 源码 有案例 可定制

    产品说明: 家装服务平台.社区服务平台.服务类型的平台--公司成熟产品 包括工匠注册.资质认证.发布服务产品.会员注册.预约服务.工匠定价.执行服务.服务完毕填写工作日志上传现场照片.会员确认服务.返 ...

  9. Java利用JNI调用C/C++写成的DLL

    前言 由于学期作业的要求,笔者需要开发一个语音识别系统.出于对Java的热爱,笔者非常想用Java来写上层程序(前台+数据库的三层),又要用到Microsoft Speech SDK,所以在这些条件下 ...

  10. ubuntu上解决访问github慢的方法

    1.进入终端命令行模式,输入sudo vi /etc/hosts 2.输入i进入编辑命令,英文输入法输入G,vim编辑器跳到hosts文件的最后一行 3.用浏览器访问 IPAddress.com 使用 ...