hdu 3859 Inverting Cups

题意是给出A个杯子,一开始都朝上,每次可以翻B个杯子,问最少需要翻转多少次可以让所有杯子都朝下。

分类讨论:

首先对于A%B==0一类情况,直接输出。

对于A>=3B,让A减到[2B,3B)区间内,翻转次数累加上A/B-2。

当A>=2B时,分奇偶讨论:A为奇数B为偶数显然无解;AB同奇偶时最多需要3次,A偶数B奇数最多需要4次。

当A<2B时,分奇偶讨论:AB同奇偶时最多需要3次,A奇数B偶数无解,A偶数B奇数时,有F(A,B)=F(A,A-B)成立,可以转换成上面的情况求解即可。

具体证明画画图就知道了,将两个B分别放到对称的位置上,想办法调整使得每次改变自己需要的杯子就行。对于A偶B奇的F(A,B)=F(A,A-B),其实挺好想的,因为A是偶数,B是奇数,而每个杯子一共翻转了奇数次,而一共一定是要翻转偶数轮,因此每个杯子不翻转的次数也是奇数次,也就相当于对“翻转”操作“取反”,每次翻转A-B个,结果是一样的,因此F(A,B)=F(A,A-B)成立。

 

 1 #include<iostream>
 2 #include<stdio.h>
 3 #include<algorithm>
 4 #include<iomanip>
 5 #include<cmath>
 6 #include<cstring>
 7 #include<vector>
 8 #define ll __int64
 9 #define pi acos(-1.0)
10 #define MAX 5000001
11 using namespace std;
12 ll solve(ll n,ll m)
13 {
14     if(n%m==0) return n/m;
15     if(n%2==1&&m%2==0) return -1;
16     if(n>=3*m) return n/m-2+solve(n%m+2*m,m);
17     if(n%2==m%2) return 3;
18     if(n>=2*m) return 4;
19     return solve(n,n-m);
20 }
21 int main(){
22     ll n,m;
23     while(cin>>n>>m){
24         ll ans=solve(n,m);
25         if(ans<0) cout<<"No Solution!"<<endl;
26         else cout<<ans<<endl;
27     }
28     return 0;
29 }
View Code

 

 

 

posted @ 2013-08-06 10:34  _随心所欲_  阅读(322)  评论(0编辑  收藏  举报