AtCoder arc145. B - AB Game题解

博客围绕石子游戏展开,给定n个石子,Alice可拿A的倍数个,Bob可拿B的倍数个,要求计算从N ∈ [1,n]每一轮中Alice赢的次数。通过分类讨论给出O(1)算法,并给出不同情况的计算方法,最后展示了C++代码实现。

题目描述

一共有 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} ana
(1) 如果 a≤ba {\leq} bab
那么第一次 AliceAliceAlice 直接拿 p∗ap * apa 个石子,其中 $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,a1] 一共 aaa 次,那么AliceAliceAlice 可以赢 n−a−1n - a - 1na1
(2) 如果 a>ba > ba>b
同样的第一次 AliceAliceAlice 直接拿 p∗ap * apa 个石子,其中 $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)[pa,pa+b) ,一共 bbb
此时 p∈[2,⌊n/a⌋]p \in [2, {\lfloor n/a \rfloor}]p[2,n/a] 一共 ⌊n/a⌋−1{\lfloor n/a \rfloor} - 1n/a1
为什么 ppp 要从 222 开始?
因为在第一个 aaa 区间内,也就是 [1,a−1][1, a-1][1,a1] AliceAliceAlice 是输的,所以需要把这一段给减去
此时 AliceAliceAlice 可以获胜 (⌊n/a⌋−1)∗b({\lfloor n/a \rfloor} - 1) * b(n/a1)b
接下来看最后一段 leftleftleft 个石子
如果 left≥bleft {\geq} bleftb,那么 AliceAliceAlice 还可以获胜的区间为 left∈[0,b−1]left \in [0, b-1]left[0,b1] 一共 bbb 次,那么获胜的总次数就为 (⌊n/a⌋−1)∗b+b({\lfloor n/a \rfloor} - 1) * b + b(n/a1)b+b 次,也可以写为⌊n/a⌋∗b{\lfloor n/a \rfloor} * bn/ab
如果 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/a1)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;
		}
	}
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值