题目大意

  给你三个数\(a,b,c\),每次你可以选择一个数\(s_1\),再选择一个数\(s_2\),把\(s_1\)变成\(2s_2-s_1\),但要求\(s_3\)不在\(s_1\)到\(2s_2-s_1\)之间。

  再给你三个数\(x,y,z\),问你是否能把\(a,b,c\)变成\(x,y,z\)。

  \(|a|,|b|,|c|,|x|,|y|,|z|\leq {10}^9\)

题解

  首先三个数\(a,b,c(a<b<c)\)只能有三种转移:

\[(a,b,c)\rightarrow
\begin{cases}
(2a-b,a,c)\\
(a,c,2c-b)\\
(b,2b-a,c)~~~~~(c-b>b-a)\\
(a,2b-c,b)~~~~~(c-b<b-a)
\end{cases}
\]

  可以发现,这些状态构成了一棵树。

  转移可以用辗转相除法加速。

  直接暴力往上跳找LCA即可。

  时间复杂度:\(O(\log (\max_a-\min_a))\)

代码

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
void sort(ll &a,ll &b,ll &c)
{
if(b<a)
swap(a,b);
if(c<a)
swap(a,c);
if(c<b)
swap(b,c);
}
void open(const char *s)
{
#ifdef DEBUG
char str[100];
sprintf(str,"%s.in",s);
freopen(str,"r",stdin);
sprintf(str,"%s.out",s);
freopen(str,"w",stdout);
#endif
}
ll gcd(ll a,ll b)
{
return b?gcd(b,a%b):a;
}
ll getmid(ll a,ll b,ll c)
{
while(b-a!=c-b)
if(b-a>c-b)
{
ll j=(b-a)/(c-b);
ll k=c-b;
c-=j*k;
b-=j*k;
if(b==a)
{
b+=k;
c+=k;
}
}
else
{
ll j=(c-b)/(b-a);
ll k=b-a;
a+=j*k;
b+=j*k;
if(b==c)
{
b-=k;
a-=k;
}
}
return b;
}
pll a1[110];
int d1[110];
pll a2[110];
int d2[110];
int cnt1,cnt2;
int main()
{
open("c");
ll a,b,c,x,y,z;
scanf("%lld%lld%lld%lld%lld%lld",&a,&b,&c,&x,&y,&z);
sort(a,b,c);
sort(x,y,z);
if(gcd(b-a,c-b)!=gcd(y-x,z-y))
{
printf("NO\n");
return 0;
}
if(getmid(a,b,c)!=getmid(x,y,z))
{
printf("NO\n");
return 0;
}
printf("YES\n");
pll s1(b-a,c-b);
pll s2(y-x,z-y);
cnt1=cnt2=0;
d1[0]=d2[0]=0;
while(s1.first!=s1.second)
{
a1[++cnt1]=s1;
if(s1.first>s1.second)
{
d1[cnt1]=s1.first/s1.second;
s1.first%=s1.second;
if(!s1.first)
{
s1.first=s1.second;
d1[cnt1]--;
}
}
else
{
d1[cnt1]=s1.second/s1.first;
s1.second%=s1.first;
if(!s1.second)
{
s1.second=s1.first;
d1[cnt1]--;
}
}
}
a1[++cnt1]=s1;
d1[cnt1]=0;
reverse(a1+1,a1+cnt1+1);
reverse(d1+1,d1+cnt1+1);
while(s2.first!=s2.second)
{
a2[++cnt2]=s2;
if(s2.first>s2.second)
{
d2[cnt2]=s2.first/s2.second;
s2.first%=s2.second;
if(!s2.first)
{
s2.first=s2.second;
d2[cnt2]--;
}
}
else
{
d2[cnt2]=s2.second/s2.first;
s2.second%=s2.first;
if(!s2.second)
{
s2.second=s2.first;
d2[cnt2]--;
}
}
}
a2[++cnt2]=s2;
d2[cnt2]=0;
reverse(a2+1,a2+cnt2+1);
reverse(d2+1,d2+cnt2+1);
ll ans=0;
int i,j;
for(i=1;i<=cnt1;i++)
d1[i]+=d1[i-1];
for(i=1;i<=cnt2;i++)
d2[i]+=d2[i-1];
i=cnt1,j=cnt2;
while(a1[i-1]!=a2[j-1])
if(d1[i-1]>d2[j-1])
{
ans+=d1[i]-d1[i-1];
i--;
}
else
{
ans+=d2[j]-d2[j-1];
j--;
}
if(a1[i].first==a2[j].first||a1[i].second==a2[j].second)
{
if(d1[i]>d2[j])
ans+=d1[i]-d2[j];
else
ans+=d2[j]-d1[i];
}
else
ans+=d1[i]-d1[i-1]+d2[j]-d2[j-1];
printf("%lld\n",ans);
return 0;
}

【BZOJ2144】Throw 数论的更多相关文章

  1. 数学概念——I - 数论,线性方程

    I - 数论,线性方程 Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit  ...

  2. 浅谈Java的throw与throws

    转载:http://blog.csdn.net/luoweifu/article/details/10721543 我进行了一些加工,不是本人原创但比原博主要更完善~ 浅谈Java异常 以前虽然知道一 ...

  3. C++异常处理:try,catch,throw,finally的用法

    写在前面 所谓异常处理,即让一个程序运行时遇到自己无法处理的错误时抛出一个异常,希望调用者可以发现处理问题. 异常处理的基本思想是简化程序的错误代码,为程序键壮性提供一个标准检测机制. 也许我们已经使 ...

  4. Codeforces Round #382 Div. 2【数论】

    C. Tennis Championship(递推,斐波那契) 题意:n个人比赛,淘汰制,要求进行比赛双方的胜场数之差小于等于1.问冠军最多能打多少场比赛.题解:因为n太大,感觉是个构造.写写小数据, ...

  5. java中的throw与throws的区别

    什么时运行时异常?什么是非运行时异常? 通俗的讲: 运行时异常:就是编译通过,运行时就崩了,比如数组越界. 非运行时异常:就是编译不通过,这时就得必须去处理了.不然就没法运行了. 全面的讲: Thro ...

  6. js 利用throw 写的一个小程序

    在下边的小程序中比较特殊的是使用isNaN()函数判断一个参数是不是数字, <!DOCTYPE html> <!DOCTYPE html> <html> <h ...

  7. C++异常处理: try,catch,throw,finally的用法

    写在前面 所谓异常处理,即让一个程序运行时遇到自己无法处理的错误时抛出一个异常,希望调用者可以发现处理问题. 异常处理的基本思想是简化程序的错误代码,为程序键壮性提供一个标准检测机制. 也许我们已经使 ...

  8. Java 中 手动抛出异常: throw new Exception("错误信息") 错误信息的获得

    当然需要先用try catch捕获,但注意new Exception("")括号里的字符串其实是异常原因,所以获取是要用ex.getCause().getMessage() int ...

  9. NOIP2014 uoj20解方程 数论(同余)

    又是数论题 Q&A Q:你TM做数论上瘾了吗 A:没办法我数论太差了,得多练(shui)啊 题意 题目描述 已知多项式方程: a0+a1x+a2x^2+..+anx^n=0 求这个方程在[1, ...

随机推荐

  1. Bootstrap 基础讲解2

    -------------------------------------------思想是行动的先导,心理问题直接作用并影响人的思想. 知识预览 bootstrap简介 CSS栅格系统 四 表格 表 ...

  2. OO博客作业4:第13-14周作业总结

    一.论述测试与正确性论证的效果差异,比较其优缺点 测试是设计若干组测试用例,运行程序并检验其是否完成预期功能.测试是一种直接发现BUG的方法,可以准确断定什么样的BUG会发生,并通过辅助调试进一步确定 ...

  3. poj 1486 纸张与数字匹配(二分图+割边处理)

    题目来源:http://poj.org/problem?id=1486 题意: 算出所有独一无二的字母与数字的组合,使二分图完全匹配 我以为所有点都要独一无二匹配时输出匹配方法 题解: 先得到一个完全 ...

  4. 牛客练习赛35 C.函数的魔法

    链接 [https://ac.nowcoder.com/acm/contest/32] 题意 题目描述 一位客人来到了此花亭,给了女服务员柚一个数学问题:我们有两个函数,F(X)函数可以让X变成(XX ...

  5. CSS颜色代码 颜色值 颜色名字大全

    颜色值 CSS 颜色使用组合了红绿蓝颜色值 (RGB) 的十六进制 (hex) 表示法进行定义.对光源进行设置的最低值可以是 0(十六进制 00).最高值是 255(十六进制 FF).从 0 到 25 ...

  6. javascript重定向页面并用post方法传递消息

    javascript中重定向页面得方法很多,同时能传递消息的也不少:但可用post方法传递的我只找到两种: 第一种方法:用document.write在 JavaScript函数中,用document ...

  7. Shell脚本2

      5 Shell传递参数 我们可以在执行 Shell 脚本时,向脚本传递参数, 脚本内获取参数的格式为:$n.n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,以此类推…… ...

  8. js原生实现div渐入渐出

    jq对渐入渐出进行封装,简单的使用连个方法就可以实现.fadeIn(),fadeOut();如果我们界面没有使用jq那么原生怎么实现呢? 我们讲解一下,这个原理.当我们要实现渐入的时候,首先是让隐藏的 ...

  9. C# DataTable详解

    添加引用 using System.Data; 创建表 //创建一个空表 DataTable dt = new DataTable(); //创建一个名为"Table_New"的空 ...

  10. Git命令以及常见注意事项

    命令: git init -> 初始化一个git仓库 git clone -> 克隆一个本地库 git pull -> 拉取服务器最新代码 git fetch –p -> 强行 ...