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

 

 

 

你可能感兴趣的:(HDU)