[BZOJ2144][国家集训队2011]跳跳棋
题目描述
跳跳棋是在一条数轴上进行的。棋子只能摆在整点上。
每个点不能摆超过一个棋子。
我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在\(a\),\(b\),\(c\)这三个位置。
我们要通过最少的跳动把他们的位置移动成\(x\),\(y\),\(z\)。(棋子是没有区别的)跳动的规则很简单,任意选一颗棋子,对一颗中轴棋子跳动。
跳动后两颗棋子距离不变。
一次只允许跳过\(1\)颗棋子。
\(a,b,c,x,y,z \le 1e9\)
Input
第一行包含三个整数,表示当前棋子的位置\(a\), \(b\) ,\(c\)。(互不相同)
第二行包含三个整数,表示目标位置\(x\),\(y\) ,\(z\)。(互不相同)
Output
如果无解,输出一行\(NO\)。如果可以到达,第一行输出\(YES\),第二行输出最少步数。
Sample Input
1 2 3
0 3 5
Sample Output
YES
2
一道非常不错的想法题。
非常强大的建模。。。
对于一个状态\((a,b,c)\),我们保证\(a \le b \le c\);
对于当前状态\((a,b,c)\)可以转移的状态为
中间的\(b\)往两边跳,即\((2*a-b,b,c)\)和\((a,b,2*c-b)\)为\((a,b,c)\)的子节点。
由于,一个棋子只能跳过一个棋子。
所以,\((a,b,c)\)由两边的棋子跳动的转移,只能由距离中轴最近的棋子跳动,将其状态定义为其父亲节点。
由此,我们可以把一个状态到另一个状态的过程转换为树上一个节点到另一个节点的距离。
很显然我们只知道初状态和末状态。
我们并不知道树上的所有节点,因此,我们不能直接利用常规方法求\(LCA\)。
但是我们发现,我们可以求出一个节点的\(K\)祖先的状态。
由此,我们可以将一个节点移至于另一个节点深度相同的位置。
再二分答案,二分两个节点向上走的步数。
现在,问题成功转换为如何快速的求出一个节点的\(k\)祖先的状态。
我们可以发现设前两个数的差值为\(t_1\),后两个数的差值为\(t_2\),
左边的节点最多往右边跳\((t_2-1)/t_1\)次,然后变成右边跳。
这是一个辗转相除的过程,时间复杂度为\(O(log{n})\),问题也就解决了。
详见代码。
代码如下
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define LL long long
#define reg register
#define debug(x) cerr<<#x<<" = "<<x<<endl;
#define rep(a,b,c) for(reg int a=(b),a##_end_=(c); a<=a##_end_; ++a)
#define ret(a,b,c) for(reg int a=(b),a##_end_=(c); a<a##_end_; ++a)
#define drep(a,b,c) for(reg int a=(b),a##_end_=(c); a>=a##_end_; --a)
#define erep(i,x) for(reg int i=Head[x]; i; i=Nxt[i])
inline int Read() {
int res = 0, f = 1;
char c;
while (c = getchar(), c < 48 || c > 57)if (c == '-')f = 0;
do res = (res << 3) + (res << 1) + (c ^ 48);
while (c = getchar(), c >= 48 && c <= 57);
return f ? res : -res;
}
template<class T>inline bool Min(T &a, T const&b) {
return a > b ? a = b, 1 : 0;
}
template<class T>inline bool Max(T &a, T const&b) {
return a < b ? a = b, 1 : 0;
}
const int N = 20, M = 5e5 + 5;
struct node {
int A[4];
bool operator!=(node _)const {
rep(i, 1, 3)if (A[i] != _.A[i])return true;
return false;
}
};
int tot;
node Find(int *A, int step) {
node Ans;
rep(i, 1, 3)Ans.A[i] = A[i];
int step1 = A[2] - A[1], step2 = A[3] - A[2];
if (step1 == step2)return Ans;
if (step1 < step2) {
int t = min(step, (step2 - 1) / step1);
step -= t, tot += t;
Ans.A[1] += t * step1, Ans.A[2] += t * step1;
} else {
int t = min(step, (step1 - 1) / step2);
step -= t, tot += t;
Ans.A[2] -= t * step2, Ans.A[3] -= t * step2;
}
if (step)return Find(Ans.A, step);
else return Ans;
}
int A[5], B[5];
signed main(void) {
rep(i, 1, 3)A[i] = Read();
rep(i, 1, 3)B[i] = Read();
sort(A + 1, A + 4), sort(B + 1, B + 4);
node a = Find(A, 1e9); int step1 = tot; tot = 0;
node b = Find(B, 1e9); int step2 = tot; tot = 0;
if (a != b)return !puts("NO");
if (step1 > step2) {
swap(step1, step2);
rep(i, 1, 3)swap(A[i], B[i]);
}
int Ans = step2 - step1;
node T = Find(B, Ans);
rep(i, 1, 3)B[i] = T.A[i];
int L = 0, R = step1;
while (L <= R) {
int mid = (L + R) >> 1;
if (Find(A, mid) != Find(B, mid)) L = mid + 1;
else R = mid - 1;
}
puts("YES");
printf("%d", Ans + 2 * L);
return 0;
}
[BZOJ2144][国家集训队2011]跳跳棋的更多相关文章
- AC日记——[国家集训队2011]旅游(宋方睿) cogs 1867
[国家集训队2011]旅游(宋方睿) 思路: 树链剖分,边权转点权: 线段树维护三个东西,sum,max,min: 当一个区间变成相反数时,sum=-sum,max=-min,min=-max: 来, ...
- cogs 1901. [国家集训队2011]数颜色
Cogs 1901. [国家集训队2011]数颜色 ★★★ 输入文件:nt2011_color.in 输出文件:nt2011_color.out 简单对比时间限制:0.6 s 内存限制 ...
- BZOJ 2150 cogs 1861 [国家集训队2011]部落战争
题目描述 lanzerb的部落在A国的上部,他们不满天寒地冻的环境,于是准备向A国的下部征战来获得更大的领土. A国是一个M*N的矩阵,其中某些地方是城镇,某些地方是高山深涧无人居住.lanzerb把 ...
- bzoj2144 【国家集训队2011】跳跳棋
Description 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子.我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.我们要通过最少的跳动把他 ...
- [BZOJ2144]国家集训队 跳跳棋
题目描述 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.我们要通过最少的跳动把他们的位置移动 ...
- happiness[国家集训队2011(吴确)]
[试题来源] 2011中国国家集训队命题答辩 [问题描述] 高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友.这学期要分文理科了,每个同学对于选择文科 ...
- COGS1882 [国家集训队2011]单选错位
★ 输入文件:nt2011_exp.in 输出文件:nt2011_exp.out 简单对比时间限制:1 s 内存限制:512 MB [试题来源] 2011中国国家集训队命题答辩 [问题 ...
- 1893. [国家集训队2011]等差子序列(bitset)
★★ 输入文件:nt2011_sequence.in 输出文件:nt2011_sequence.out 简单对比时间限制:0.3 s 内存限制:512 MB [试题来源] 2011中国 ...
- COGS.1901.[模板][国家集训队2011]数颜色(带修改莫队)
题目链接 COGS BZOJ2120 洛谷P1903 /* Add和Subd函数中的vis不能直接设为=1或=0 比如 l=1,r=0 -> l=3,r=5 时,[1,5]的vis标记全都是1 ...
随机推荐
- marquee跑马灯效果的相关属性
介绍HTML设置滚动文字marquee的相关属性 ㈠文字滚动标签 设置文字滚动使用<marquee></marquee>标签,可以再标签里面设置字体的颜色,字号,字体大小等. ...
- jquery设置css属性几种方式
用css()方法返回元素的样式属性 $("div").css("padding-left")); 用css()设置样式 $("div").c ...
- 静态库lib、动态库dll基础
首先从hello world!开始 //main.cpp文件 void cpu(); int main() { put(); ; } 在main.cpp中定义了一个cpu():函数,但没有实现其功能, ...
- Nginx配置记录【例2】
B服务器,例: [root@localhost conf.d]# egrep -v "^#|^$" /etc/nginx/nginx.conf user nginx; worker ...
- Mongo Cursor
简介 在使用 Java 对数据库进行连接时,都会获取到一个 cursor ,cursor 实际指到的是我们查询数据库的query,而并不是 query 查询到的数据集. 此次在使用 mongo 的 c ...
- vue开发多页面应用
1.添加多页面配置 在工程根路径下(package.json同目录)添加添加vue.config.js配置文件,内容为: module.exports = { pages: { index: 'src ...
- JS基础_Null和Undefind
1.Null Null类型的值只有一个值,就是null null专门用来表示一个为空的对象 var a=null; console.log(a);//nulltypeof a //object 2.U ...
- leetcode 198. House Robber 、 213. House Robber II 、337. House Robber III 、256. Paint House(lintcode 515) 、265. Paint House II(lintcode 516) 、276. Paint Fence(lintcode 514)
House Robber:不能相邻,求能获得的最大值 House Robber II:不能相邻且第一个和最后一个不能同时取,求能获得的最大值 House Robber III:二叉树下的不能相邻,求能 ...
- 获取手机本地IP地址
public String getLocalIpAddress() { try { for (Enumeration<NetworkInterface> en = NetworkInter ...
- redhat下配置SEED DVS6446开发环境2
---恢复内容开始--- 1.rpcbind步骤 linux包:portmap安装包 libgssglue-0.1-8.1.el6.i686.rpm libtirpc-0.2.1-1.el6.i68 ...