题目描述
一共有 nnn 个石子,每次 AliceAliceAlice 可以拿 AAA 的倍数个石子,BobBobBob 可以拿 BBB 的倍数个石子
求从 N ∈\in∈ [1,n][1, n][1,n] 每一轮中 AliceAliceAlice 赢的次数
输入样例
27182818284 59045 23356
输出样例
10752495144
算法
(思维题) O(1)O(1)O(1)
分类讨论
1. 如果 n<an < an<a ,那么无论如何 AliceAliceAlice 都不可能赢,输出 000 即可
2. 如果 n≥an{\geq} an≥a
(1) 如果 a≤ba {\leq} ba≤b
那么第一次 AliceAliceAlice 直接拿 p∗ap * ap∗a 个石子,其中 $p = {\lfloor n/a \rfloor} $
剩下的石子 left=n mod aleft = {n {\bmod a}}left=nmoda,此时 left<bleft < bleft<b,所以剩下的情况 BobBobBob 不可能赢都是 AliceAliceAlice 赢
AliceAliceAlice 输的情况只有 n∈[1,a−1]n \in [1, a-1]n∈[1,a−1] 一共 aaa 次,那么AliceAliceAlice 可以赢 n−a−1n - a - 1n−a−1次
(2) 如果 a>ba > ba>b
同样的第一次 AliceAliceAlice 直接拿 p∗ap * ap∗a 个石子,其中 $p = {\lfloor n/a \rfloor} $
剩下的石子 left=n mod aleft = {n {\bmod a}}left=nmoda
如果 left<bleft < bleft<b,那么 AliceAliceAlice 还可以赢的区间为 [p∗a,p∗a+b)[p * a, p * a + b)[p∗a,p∗a+b) ,一共 bbb 次
此时 p∈[2,⌊n/a⌋]p \in [2, {\lfloor n/a \rfloor}]p∈[2,⌊n/a⌋] 一共 ⌊n/a⌋−1{\lfloor n/a \rfloor} - 1⌊n/a⌋−1 次
为什么 ppp 要从 222 开始?
因为在第一个 aaa 区间内,也就是 [1,a−1][1, a-1][1,a−1] AliceAliceAlice 是输的,所以需要把这一段给减去
此时 AliceAliceAlice 可以获胜 (⌊n/a⌋−1)∗b({\lfloor n/a \rfloor} - 1) * b(⌊n/a⌋−1)∗b 次
接下来看最后一段 leftleftleft 个石子
如果 left≥bleft {\geq} bleft≥b,那么 AliceAliceAlice 还可以获胜的区间为 left∈[0,b−1]left \in [0, b-1]left∈[0,b−1] 一共 bbb 次,那么获胜的总次数就为 (⌊n/a⌋−1)∗b+b({\lfloor n/a \rfloor} - 1) * b + b(⌊n/a⌋−1)∗b+b 次,也可以写为⌊n/a⌋∗b{\lfloor n/a \rfloor} * b⌊n/a⌋∗b 次
如果 left<bleft < bleft<b,那么 AliceAliceAlice 还可以获胜的区间为 left∈[0,n mod a]left \in [0, n {\bmod} a]left∈[0,nmoda] ,一共 n mod a+1n {\bmod} a + 1nmoda+1次,那么获胜的总次数为 (⌊n/a⌋−1)∗b+n mod a+1({\lfloor n/a \rfloor} - 1) * b + n {\bmod} a + 1(⌊n/a⌋−1)∗b+nmoda+1 次
C++ 代码
#include<iostream>
#include<cstring>
#include<algorithm>
#include<queue>
#include<unordered_map>
#include<unordered_set>
#include<cmath>
#include<map>
#include<set>
#include<stack>
#include<vector>
#include<deque>
#include<cmath>
#include<ctime>
using namespace std;
typedef long long ll;
int main()
{
ll n,a,b;
cin>>n>>a>>b;
if(n<a) puts("0");
else
{
if(b>=a)
{
cout<<(n-a+1)<<endl;
}
else
{
ll res=(n/a-1)*b;
if(n%a>=b) res+=b;
else res+=n%a+1;
cout<<res<<endl;
}
}
}
博客围绕石子游戏展开,给定n个石子,Alice可拿A的倍数个,Bob可拿B的倍数个,要求计算从N ∈ [1,n]每一轮中Alice赢的次数。通过分类讨论给出O(1)算法,并给出不同情况的计算方法,最后展示了C++代码实现。
1048

被折叠的 条评论
为什么被折叠?



